当前位置: 编程技术>c/c++/嵌入式
C++虚析构函数的使用分析
来源: 互联网 发布时间:2014-10-16
本文导语: 在C++中,不能声明虚构造函数,但可以声明虚析构函数。多态性是指不同的对象对同一消息有不同的行为特性。虚函数作为运行时多态性的基础,主要是针对对象的,而构造函数是在对象产生之前运行的,因此虚构造函数是没...
在C++中,不能声明虚构造函数,但可以声明虚析构函数。多态性是指不同的对象对同一消息有不同的行为特性。虚函数作为运行时多态性的基础,主要是针对对象的,而构造函数是在对象产生之前运行的,因此虚构造函数是没有意义的。
析构函数的功能是在该类对象消亡之前进行一些必要的清理工作,析构函数最好都是virtual的。
首先解释一下虚构函数和指针之间是如何交互的,以及虚析构函数的具体含义。例如以下代码,其中SomeClass是含有非virtual析构函数的一个类:
SomeClass *p= new SomeClass;
. . . . . .
delect p;
为p调用delect时,会自动调用SomeClass类的析构函数。现在,再来看看将析构函数标记为virtual之后,会发生什么事情。
描述析构函数与虚函数机制的交互时,最简单的表述是:将所有析构函数都视为具有相同的名字(即使它们并非真的同名)。例如,假定Derived类是Base类的一个派生类,并假定Base类中的析构函数标记为virtual。现在来分析以下代码:
Base *pBase= new Derived;
. . . . . .
delect pBase;
为Base调用delect时,会调用一个析构函数。由于Base类中的析构函数标记为virtual,而且指向的对象属于Derived类型,所以会调用Derived类中的析构函数。
应注意的一点是,将析构函数标记为virtual后,派生类所有的析构函数都自动成为virtual的(不管时候用virtual来标记它们)。同样地,这种行为就好象所有析构函数都具有相同的名字(即使事实上不同名)。
下面说的是将所有析构函数都标记为virtual的好处。假定Base类有一个指针类型的成员变量pB,Base类的构造函数会创建由pB指向的一个动态变量,而Base类的析构函数会删除pB指向的动态变量。另外,假定Base类没有标记为virtual,并假定Derived类(它从Base派生)有一个指针类型的成员变量pD,Derived类的构造函数会创建有pD指向的一个动态变量,而Derived类的析构函数会删除pD指向的动态变量。分析一下以下代码:
Base *pBase= new Derived;
. . . . . .
delect pBase;
由于基类中的析构函数没有标记为virtual,所以只会调用Base类的析构函数。它会将pB指向的动态变量占用的内存返还给自由存储。但是,对于pD指向的动态变量来说,它占用的内存永远不会返还给自由存储(除非程序终止)。
另一方面,如果基类Base的析构函数标记为virtual,那么为pBase调用delect时,就会调用Derived类的析构函数(因为指向的对象属于Derived类型)。Derived类的析构函数会删除pD指向的动态变量,再自动调用基类Base的析构函数。后者会删除pB指向的动态变量。因此,将基类析构函数标记为virtual之后,所有内存都能成功地由自由存储回收。为了准备好迎接这种情况,最好总是将析构函数标记为virtual。
举个例子说明一下:
#include "stdafx.h"
#include
using namespace std;
class Base
{
public:
Base(){cout
C++ Maps 成员 key_comp():返回比较元素key的函数
怎么样在C++里调用Fortran函数??????
C++ Maps 成员 value_comp():返回比较元素value的函数
SOS ,在UNIX下编译C++程序时出现链接错误:未下义的符号,但函数库里有这些函数,不知道为什么?
C++ MultiMaps 成员 key_comp():返回比较key的函数
求linux C++的时间相关函数及用法,急用,谢谢
C++ MultiMaps 成员 value_comp():返回比较元素value的函数
C++ 函数编程库 FC++
C++ Strings(字符串) 成员 Constructors:构造函数,用于字符串初始化
C++ 程序中,如何增加c函数?
Linux c++虚函数(virtual function)简单用法示例代码
C++中虚拟函数表的问题
C++ STL Bitsets构造函数及成员函数解释及代码示例
linux下编译c++程序,系统头文件中包含的函数的定义源代码可以查看吗?
c++ stl容器set成员函数介绍及set集合插入,遍历等用法举例
Linux中的共享库难道只能导出函数,而不能导出C++类吗?
c++断言assert()函数的头文件及其用法详解
常用C++函数库 Libretta
Windows和Linux下C++类成员方法作为线程函数方法介绍
C++空类默认函数详细解析
C++ STL库中priority_queue介绍,成员函数说明及priority_queue具体用法举例
析构函数的功能是在该类对象消亡之前进行一些必要的清理工作,析构函数最好都是virtual的。
首先解释一下虚构函数和指针之间是如何交互的,以及虚析构函数的具体含义。例如以下代码,其中SomeClass是含有非virtual析构函数的一个类:
SomeClass *p= new SomeClass;
. . . . . .
delect p;
为p调用delect时,会自动调用SomeClass类的析构函数。现在,再来看看将析构函数标记为virtual之后,会发生什么事情。
描述析构函数与虚函数机制的交互时,最简单的表述是:将所有析构函数都视为具有相同的名字(即使它们并非真的同名)。例如,假定Derived类是Base类的一个派生类,并假定Base类中的析构函数标记为virtual。现在来分析以下代码:
Base *pBase= new Derived;
. . . . . .
delect pBase;
为Base调用delect时,会调用一个析构函数。由于Base类中的析构函数标记为virtual,而且指向的对象属于Derived类型,所以会调用Derived类中的析构函数。
应注意的一点是,将析构函数标记为virtual后,派生类所有的析构函数都自动成为virtual的(不管时候用virtual来标记它们)。同样地,这种行为就好象所有析构函数都具有相同的名字(即使事实上不同名)。
下面说的是将所有析构函数都标记为virtual的好处。假定Base类有一个指针类型的成员变量pB,Base类的构造函数会创建由pB指向的一个动态变量,而Base类的析构函数会删除pB指向的动态变量。另外,假定Base类没有标记为virtual,并假定Derived类(它从Base派生)有一个指针类型的成员变量pD,Derived类的构造函数会创建有pD指向的一个动态变量,而Derived类的析构函数会删除pD指向的动态变量。分析一下以下代码:
Base *pBase= new Derived;
. . . . . .
delect pBase;
由于基类中的析构函数没有标记为virtual,所以只会调用Base类的析构函数。它会将pB指向的动态变量占用的内存返还给自由存储。但是,对于pD指向的动态变量来说,它占用的内存永远不会返还给自由存储(除非程序终止)。
另一方面,如果基类Base的析构函数标记为virtual,那么为pBase调用delect时,就会调用Derived类的析构函数(因为指向的对象属于Derived类型)。Derived类的析构函数会删除pD指向的动态变量,再自动调用基类Base的析构函数。后者会删除pB指向的动态变量。因此,将基类析构函数标记为virtual之后,所有内存都能成功地由自由存储回收。为了准备好迎接这种情况,最好总是将析构函数标记为virtual。
举个例子说明一下:
代码如下:
#include "stdafx.h"
#include
using namespace std;
class Base
{
public:
Base(){cout