关于一个虚函数的疑惑

 
C/C++ code

//============================================================================
// Name        : VirtualTest.cpp
// Author      :
// Version     :
// Copyright   : Your copyright notice
// Description : Hello World in C++, Ansi-style
//============================================================================

#include <iostream>
#include <string>
using namespace std;

class A {
public:
    string name;
    A(char const *n):name(n) {
    }

    ~A() {
        cout << "A析构" << endl;
    }
};

class B : public A {
public:
    int age;
    B(char const *n, int age):A(n), age(age) {
    }

    virtual ~B() {
        cout << "B析构" << endl;
    }
};

int main() {
    A *b = new B("张三", 10);
    cout << b->name.c_str() << endl;
    delete b;
    return 0;
}

问题如下:

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位置

可以如下简单的测试,可以看出来
C/C++ code
        
        
B * pb = new B( " 张三 " , 10 ); A * pa = pb; cout << pb << ' \t ' << pa << endl; // 这里可以看出来pb和pa的值不一样


到这里问题原因就应该很明朗了,
delete b;
会先引起析构函数的调用,而因为基类的析构函数不是虚拟的,所以这里就是静态调用,
这时的this指针位置没有调整,仍是指向name的偏移处,一般在动态申请内存时,会在传回的指针值得前面安插一些这段内存的大小信息,因为这里在delete b时b指针的值和在new时传回的值不一样,这样必然导致这段内存大小信息读取错误,这样在释放时出现错误就不奇怪了
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值