Effective C++学习笔记 条款07:为多态基类声明virtual析构函数
一、C++明确指出:当derived class对象经由一个base class指针被删除,而该base class带着一个non-virtual析构函数,其结果未定义——实际执行时通常发生的是对象的derived成分没有被销毁!(注:使用基类引用派生类的方式使用多态,由于引用只是对原对象的一个引用或者叫做别名,其并没有分配内存,对其引用对象内存的销毁,将由原对象自己负责,所以使用引用的时候,如果析构函数不为virtual,那么也不会发生derived成员没有销毁的情况)
例如:
class base
{
public:
base(const string& name): m_baseName(name) {}
private:
string m_baseName;
}; class derived: public base
{
public:
derived(const string& baseName, const string& derivedName): base(baseName), m_derivedName(derivedName) {}
private:
string m_derivedName;
};
int main(void)
{
base *pd = new derived("base", "derived");
delete pd;
}
经由gdb单步调试发现,delete pd的时候,只调用了base的析构函数,销毁了基类的数据成员,但是没有调用派生类的析构函数,导致派生类的数据成员没有被销毁,造成一个诡异的“局部销毁”对象。这会导致资源泄漏。
gdb调试如下:
Breakpoint , main () at clause07_1.cpp:
base *pd = new derived("base", "derived");
(gdb) n
31 delete pd;
(gdb) s
base::~base (this=0x603070, __in_chrg=<optimized out>) at clause07_1.cpp:11
11 class base
(gdb) s
main () at clause07_1.cpp:32
32 }
(gdb) s
0x00007ffff752576d in __libc_start_main () from /lib/x86_64-linux-gnu/libc.so.
注:使用基类引用派生类的方式使用多态,由于引用只是对原对象的一个引用或者叫做别名,其并没有分配内存,对其引用对象内存的销毁,将由原对象自己负责,所以使用引用的时候,如果析构函数不为virtual,那么也不会发生derived成员没有销毁的情况
例如:
class base
{
public:
~base()
{
cout << "base::~base" << endl;
}
virtual void f1()
{
cout << "base::f1()" << endl;
}
void f2()
{
cout << "base::f2()" << endl;
}
}; class derived: public base
{
public:
virtual void f1()
{
cout << "derived::f1()" << endl;
} void f2()
{
cout << "derived::f2()" << endl;
} ~derived()
{
cout << "derived::~derived()" << endl;
}
};
int main(void)
{
derived d;
base &rd = d;
rd.f1();
rd.f2();
}
输出为:
derived::f1()
base::f2()
derived::~derived()
base::~base
derived的析构,是有对象d的销毁产生的。由于rd只是一个引用,没有自己的内存,所以不需要销毁。此时就不会有资源的泄漏(这里的引用的对象是静态对象,自动会销毁)。
二、对于(1)出现的问题的解决方法:给base class一个virtual析构函数。
仅仅将base class的析构函数定义为virtual,如下:
class base
{
public:
base(const string& name): m_baseName(name) {}
virtual ~base() {}
private:
string m_baseName;
};
这样,derived class的析构函数,由于继承的原因,自动变为virtual,此时,通过delete指针,会触发多态机制,由于指针的动态类型是derived class,又派生类的析构函数是虚函数,所以调用derived class的析构函数,这样就会销毁派生类的数据成员,当派生类析构完成,自动会调用基类的析构函数,这样就没有资源的泄漏了。
gdb单步调试验证如下:
Breakpoint , main () at clause07_2.cpp:
base *pd = new derived("base", "derived");
(gdb) n
delete pd;
(gdb) s
derived::~derived (this=0x603070, __in_chrg=<optimized out>) at clause07_2.cpp:
class derived: public base
(gdb) s
derived::~derived (this=0x603070, __in_chrg=<optimized out>) at clause07_2.cpp:
class derived: public base
(gdb) s
base::~base (this=0x603070, __in_chrg=<optimized out>) at clause07_2.cpp:
virtual ~base() {}
(gdb) s
main () at clause07_2.cpp:
}
三、任何class只要带有virtual函数都几乎确定应该也有一个virtual析构函数。
例如:
class base
{
public:
virtual void f1() {}
}; class derived: public base
{
public:
virtual void f1()
{
ptr = operator new(sizeof(int));
}
~derived()
{
delete ptr;
}
private:
void* ptr;
};
int main(void)
{
base *pd = new derived();
pd->f1();
delete pd;
}
虚函数机制使调用派生类函数分配了内存,希望在派生类的析构函数里释放,但是由于析构函数不是virtual,所以delete只会调用基类的析构函数。所以这里有内存泄漏!
解决方法同样是base class的析构函数定义为virtual,这样就可以利用虚函数的多态机制了。
四、如果base class不含virtual函数,通常表示它并不意图被用作一个base class。当class不企图被当作base class,令其析构函数为virtual往往是个馊主意。
因为当一个类有虚函数后,那么它的所以对象的内存布局中都会多包含一个vptr(虚函数表指针),指向vtbl(vritual table)。这样会使对象膨胀,由于其他语言没有vptr,所以这就导致了不再具有移植性。
五、对于继承任何析构函数不是virtual的类,如果通过指针使用多态机制,最后通过delete该指向derived类型的base类型指针销毁原derived对象,就会出现泄漏的问题。
(注,通过引用的方式,引用对象的销毁,由其引用的对象自己负责(静态对象自动销毁,动态分配的需要delete pd(派生类的指针);)。但是如果你打算使用多态,就可能会用到使用指针的多态的方式,所以最好声明基类的析构为virtual)原因就是如(1)所说明的。
六、当你打算使用的多态的时候,base class一般应该有个virtual析构函数。另外此时,不要继承析构不是virtual的类。然而,并非所有的基类都是为了多态 ,对于不打算使用多态的情况,基类不需要virtual析构函数。
总结
a.带多态性质的base class应该声明一个virtual析构函数。
b.如果class带有virtual函数,它就应该拥有一个virtual析构函数。
c.对于一个class的设计目的不是为了作为一个base class使用,或者对于base class设计目的不是为了多态用途,就不该virtual析构函数。
Effective C++学习笔记 条款07:为多态基类声明virtual析构函数的更多相关文章
- 《effective C++》:条款07——为多态基类声明virtual析构函数
在继承中,基类的析构函数需要定义为虚析构函数数否则: (1)当派生类对象经由一个base类指针删除时,而这个base类的析构函数不是虚函数时,其结果是未定义的. (2)这样做会导致derived类部分 ...
- Effective C++_笔记_条款07_为多态基类声明virtual析构函数
(整理自Effctive C++,转载请注明.整理者:华科小涛@http://www.cnblogs.com/hust-ghtao/) 这个规则只适用于polymorphic(带多态性质的)base ...
- Effective C++ 条款七 为多态基类声明virtual析构函数
class TimeKeeper { public: TimeKeeper(); // ~TimeKeeper(); 错误,此作为一个基类,被继承了.其继承类被delete后,基类被销毁,但继承类可能 ...
- 为多态基类声明virtual析构函数
一个函数的返回值为基类指针,而当指针指向一个派生类对象,接下来派生类对象被这个基类指针删除的时候,就出现了局部销毁的问题.因为C++指出,当派生类经由一个基类指针被删除,而该基类指针带着一个non-v ...
- Effective C++ -----条款07:为多态基类声明virtual析构函数
polymorphic(带多态性质的)base classes应该声明一个virtual析构函数.如果class带有任何virtual函数,它就应该拥有一个virtual析构函数. Classes的设 ...
- effective c++(07)之为多态基类声明virtual析构函数
class TimeKeeper { public: TimeKeeper() ; ~TimeKepper() ; ... } ; class AtomicClock:public TimeKeepe ...
- 条款7:为多态基类声明virtual析构函数
C++明确指出:当派生类对象是由一个基类指针释放的,而基类中的析构函数不是虚函数,那么结果是未定义的.其实我们执行时其结果就是:只调用最上层基类的析构函数,派生类及其中间基类的析构函数得不到调用. # ...
- [Effective C++ --007]为多态基类声明virtual析构函数
引言: 我们都知道类的一个很明显的特性是多态,比如我们声明一个水果的基类: class Fruit { public: Fruit() {}; ~Fruit(){}; } 那么我们根据这个Fruit基 ...
- Effective C++(7) 为多态基类声明virtual析构函数 or Not
问题聚焦: 已经对一个对象执行了delete语句,还会发生内存泄漏吗? 先来看个demo: // 计时器类 class TimeKeeper { public: TimeKeeper(); ~Time ...
- 【C++】为多态基类声明virtual析构函数
来自<Effective C++>条款07:为多态声明virtual析构函数 当derived class对象经由一个base class指针被删除,而该base class带着一个non ...
随机推荐
- Java高效读取大文件
1.概述 本教程将演示如何用Java高效地读取大文件.这篇文章是Baeldung (http://www.baeldung.com/) 上“Java——回归基础”系列教程的一部分. 2.在内存中读取 ...
- 简单好用的 AJAX 上传插件,还可以抛弃难看的 file 按钮哦~
在做网页设计的时候,设计师常常会把上传按钮设计得非常漂亮,还用了什么放大镜之类的图标来表达 browse 的效果.可是她们不知道,type="file" 的按钮在不同浏览器上的效果 ...
- Swift和OC,是编译型语言、解释性语言、运行时语言
首先需要明确的一点是,什么是编译型语言和解释性语言 编译型语言,就是在其执行过程中需要先将其经过编译成机器码来给计算机识别的,其执行效率就会比较高这个是显而易见的,常见比如:C.C++ 而解释型语言, ...
- adb 选择设备
在adb中有多个设备时,可以先adb devices列举出设备,然后可以通过adb -s <设备名> [其他参数] 对某个设备进行操作. 例如: adb -s 0123456789ABC ...
- 【Vijos】【1923】漫长的等待
可持久化线段树 这次是询问一段区间内权值 在给定范围内的点的数量,同样是可持久化线段树简单操作…… //Vijos 1923 #include<vector> #include<cs ...
- 基于EBP的栈帧
程序的OEP,一开始以 push ebp 和mov ebp esp这两句开始. 原因:c程序的开始是以一个主函数main()为开始的,而函数在访问的过程中最重要的事情就是要确保堆栈的平衡,而在wi ...
- Mac OS X中MacPorts的安装使用备忘
Mac下面除了用dmg.pkg来安装软件外,比较方便的还有用MacPorts来帮助你安装其他应用程序,跟BSD中的ports道理一样.MacPorts就像apt-get.yum一样,可以快速安装些软件 ...
- 表中相同数据的sql语句
1.查找表中多余的重复记录,重复记录是根据单个字段(peopleId)来判断select * from peoplewhere peopleId in (select peopleId from ...
- 无法将 lambda 表达式 转换为类型“System.Delegate”,因为它不是委托类型
this.BeginInvoke(() => { this.btnQuery.Enabled = false; //禁用查询 }); 跨线程调用时,编译上面的代码将提示 对于Control.In ...
- 暑假学习日记:Splay树
从昨天开始我就想学这个伸展树了,今天花了一个上午2个多小时加下午2个多小时,学习了一下伸展树(Splay树),学习的时候主要是看别人博客啦~发现下面这个博客挺不错的http://zakir.is-pr ...