memset初始化类 在调用虚函数时的出错分析

本文探讨了C++中虚函数和memset结合使用时可能引发的问题。具体分析了当使用memset初始化包含虚函数的类实例时,如何导致虚函数指针被覆盖,进而影响虚函数调用的正确性。
偶尔 在群里看见 一段代码的问题,自己敲了下 试了试 总结一下;代码:
#include <cstring>
#include <iostream>
using namespace std;

class Parent
{
    public:
        Parent(){}
        virtual void output();
};
void Parent::output()
{
    cout << " Parent " << endl;
}
class son:public Parent
{
    public:
        virtual void output();
};
void son::output()
{
    cout << " son "  << endl;
}
int main()
{
    son s;
    memset(&s,0,sizeof(s));
    //Parent &p = s; 可以将注释打开 出错
    Parent p;
    p.output();
    s.output();
    return 0;
}



分析如下:
在使用memset作初始化的时候 一般情况下是不会出错的 但是若是 改变了内部结构的话  就会产生 错误;每个包含虚函数的类存在一个指向虚表函数的指针,在memset的时候 ,这个指针会被覆盖,所以在调用运行时绑定的虚函数的情况下 会导致程序崩溃 若不进行运行时绑定的调用 会正常输出;所以在有虚函数 有父类引用 指向了子类对象的时候 使用构造函数来初始化;

ubuntu下的出错情况:



                
在C++中,`memset` 是一个C标准库函数,用于将一块内存区域的每个字节设置为指定的值,通常用于初始化内存。其原型为 `void *memset(void *s, int c, size_t n);`,该函数将指针 `s` 所指向的内存区域的前 `n` 个字节都设置为字符 `c`。 ### 使用方法 使用 `memset` 进行初始化构造函数,主要是在构造函数内部调用 `memset` 函数来初始化对象的内存。以下是一个示例代码: ```cpp #include <iostream> #include <cstring> class ConfigSettings { public: int intValue; char charValue; float floatValue; // 构造函数中使用memset初始化 ConfigSettings() { memset(this, 0, sizeof(ConfigSettings)); } }; int main() { ConfigSettings settings; std::cout << "intValue: " << settings.intValue << std::endl; std::cout << "charValue: " << settings.charValue << std::endl; std::cout << "floatValue: " << settings.floatValue << std::endl; return 0; } ``` 在这个示例中,`ConfigSettings` 的构造函数使用 `memset` 将对象的所有内存初始化为0。 ### 原理 `memset` 的原理是按字节对内存进行操作,将指定内存区域的每个字节都设置为指定的值。当用于初始化构造函数,它会将对象的整个内存空间按字节填充为指定的值。对于POD(Plain Old Data)型的对象,由于其成员变量都是基本数据型,且没有虚函数和虚继承,其内存布局是简单的连续存储,因此可以使用 `memset` 安全地进行初始化,因为POD型的二进制内容可以随便复制、移动、重置而不会改变对象的值 [^1]。 然而,当中包含非POD型的成员变量,如 `std::map` ,使用 `memset` 会导致问题。因为非POD型的对象可能包含内部指针、引用或复杂的数据结构,`memset` 会将这些内部指针设置为 `nullptr`,破坏对象的内部结构,从而在后续操作中引发崩溃 [^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值