|
问题如下:
A的虚构函数被调用。
在linux下执行,出现如下错误:
张三
A析构
83 [sig] VirtualTest 4508 open_stackdumpfile: Dumping stack trace to VirtualTest.exe.stackdump
A的析构函数被正确调用,可为何会出现内存错误?
假如将B定义为(删除析构函数的virtual关键字)
-
C/C++ code
-
class B : public A { public : int age; B( char const * n, int age):A(n), age(age) { } ~ B() { cout << " B析构 " << endl; } };
则没有任何错误
原因如下:
C++中的“越位”是指的基类没有虚拟函数,而派生类有虚拟函数
这时你的编译器中的对象模型就很可能是,派生类对象的VPTR在最前面,如下: VPTR name age 当基类指针指向此类的派生类对象时,它是指向的name位置 当派生类的指针指向此类的派生类对象时,它是指向VPTR位置 可以如下简单的测试,可以看出来
到这里问题原因就应该很明朗了, delete b; 会先引起析构函数的调用,而因为基类的析构函数不是虚拟的,所以这里就是静态调用, 这时的this指针位置没有调整,仍是指向name的偏移处,一般在动态申请内存时,会在传回的指针值得前面安插一些这段内存的大小信息,因为这里在delete b时b指针的值和在new时传回的值不一样,这样必然导致这段内存大小信息读取错误,这样在释放时出现错误就不奇怪了 |