虚函数与纯虚函数

虚函数:在某基类中声明为virtual并在一个或多个派生类中被重新定义的成员函数,virtual  函数返回类型  函数名(参数表){函数体;} ,实现多态性,通过指向派生类的基类指针或引用,访问派生类中同名覆盖成员函数。注意虚函数在基类中是有定义的,即便定义是空。

纯虚函数:在基类中是没有定义的,必须由派生类重定义实现,否则不能由对象进行调用。

看下面的例子:

#include<iostream>
using namespace std;
class Cshape
{
public:
void SetColor(int color){m_nColor=color;}
virtual void Display(void){};
private:
int m_nColor;
}; class Crectangle:public Cshape{ //公有继承
public: }; void main()
{
Crectangle obRectangle;
Cshape* pShape=&obRectangle;
pShape->Display();
}

上面例子中,Display是虚函数,虽然是空定义,并且在派生类中没有重写该函数,但是上面的代码在编译阶段不会出错,而是在链接时出错,而如果将Display声明为纯虚函数,即 virtual void Display( )=0; 则该代码将在编译阶段出错,即编译器需要确保纯虚函数必须在子类中进行重写。注意:如果在基类中定义了纯虚函数,那么基类是不能实例化对象的,它只是提供一个接口,就想老板造房子,我就告诉你干嘛,具体怎么干还得靠自己,但是完全不说,你也当然不知道要干嘛。

虚函数怎么实现多态

前面讲到虚函数的作用主要是实现多态,那么具体是怎么实现的?为什么需要用基类指针来指向派生类对象?

多态:C++中实现多态有虚函数,抽象类,重载,覆盖,模板等方法,虚函数是最重要的一种。把不同的子类对象当做父类来看,可以屏蔽不同子类对象之间的差异,写出通用的代码,做出通用的编程,以适应需求的不断变化,实际上是通过基类来访问子对象的。那么子对象在程序中是怎么通过基类来找到与该对象对应的方法呢?这就要说到虚函数表

虚函数表

虚函数是通过虚函数表来实现的,在有虚函数的类的实例中这个表被分配在了这个实例的内存中,所以当用父类的指针来操作一个子类的时候,这张虚函数表就像一张地图,指明实际应该调用的函数。

这里引用一下CSDN上的一篇题为“C++虚函数表解析”博客(博主的地址是http://blog.csdn.net/haoel):

假设我们有这样一个类,

class Base{
public:
virtual void f(){cout << "Base::f" << endl;}
virtual void g(){cout << "Base::g" << endl;}
virtual void h(){cout << "Base::h" << endl;}
};

我们可以通过Base的实例来得到虚函数表:下面是实际例程:

void main()
{
typedef void(*Fun)(void); //这里是定义一个参数为void,返回值为void的函数指针。
Base b;
Fun pFun=NULL;
cout << "虚函数表地址:" << (int*)(&b) << endl;
cout << "虚函数表--第一个函数地址:" << (int*)*(int*)(&b) << endl;
// 调用第一个虚拟函数
pFun=(Fun)*((int*)*(int*)(&b));
pFun();
}

程序运行结果如下:(Windows 7 + Visual C++6.0)

由以上程序可以知道,如果要调用Base::g 和Base::h,代码如下:

   pFun=(Fun)*((int*)*(int*)(&b)+);
pFun=(Fun)*((int*)*(int*)(&b)+);

虚拟函数表的图示如下:

类中的虚函数所占内存为多少?

虚函数要占用4个字节的内存空间,用来指定虚函数的虚拟函数表的入口地址。既然是入口地址,所以一个类的虚函数所占的地址是不变的,和虚拟函数的个数没有关系,如果派生类继承了多个超类的虚拟函数,那个这个派生类的虚拟函数所占用的内存空间等于4*(超类个数),即为每个超类维护一张虚函数表(关于多重继承的虚函数表内存分布,更详细请参考博客http://blog.csdn.net/haoel/article/details/1948051#)。另外要注意的是:父类和子类共享一个虚函数指针。这一点可以从下面的代码中看出:

#include<iostream>
using namespace std;
class Base{
public:
virtual void f(){cout << "Base::f" << endl;}
virtual void g(){cout << "Base::g" << endl;}
virtual void h(){cout << "Base::h" << endl;}
}; class Child:public Base{
virtual void i(){cout << "Child::i" << endl;}
};
void main()
{
typedef void(*Fun)(void);
Child child;
Base* pBase=&child;
Fun pFun=NULL;
cout << sizeof(Base) << " " << sizeof(Child) << endl;
cout << "虚拟函数i()的地址是:" << (int*)*(int*)pBase+ << endl;
pFun=(Fun)*((int*)*(int*)pBase+);
pFun();
}

代码运行结果如下:

C++学习笔记--从虚函数说开去的更多相关文章

  1. 学习笔记---C++虚函数,纯虚函数

    1 .虚函数 假设people是man的父类,people类和man类都定义了实函数walk() people* p = new man(); p->walk(); 这里P执行的是people类 ...

  2. C++学习笔记27,虚函数作品

    C++它指定虚函数的行为,但实现的作者编译器. 通常,编译器处理虚函数的方法是给每个对象加入一个隐藏成员.隐藏成员中保存了一个指向函数地址数组的指针. 这个数组称为虚函数表(virtual funct ...

  3. IOS学习笔记07---C语言函数-printf函数

    IOS学习笔记07---C语言函数-printf函数 0 7.C语言5-printf函数 ------------------------- ----------------------------- ...

  4. IOS学习笔记06---C语言函数

    IOS学习笔记06---C语言函数 --------------------------------------------  qq交流群:创梦技术交流群:251572072              ...

  5. Typescript 学习笔记三:函数

    中文网:https://www.tslang.cn/ 官网:http://www.typescriptlang.org/ 目录: Typescript 学习笔记一:介绍.安装.编译 Typescrip ...

  6. ES6学习笔记<三> 生成器函数与yield

    为什么要把这个内容拿出来单独做一篇学习笔记? 生成器函数比较重要,相对不是很容易理解,单独做一篇笔记详细聊一聊生成器函数. 标题为什么是生成器函数与yield? 生成器函数类似其他服务器端语音中的接口 ...

  7. OpenCV 学习笔记03 findContours函数

    opencv-python   4.0.1 1 函数释义 词义:发现轮廓! 从二进制图像中查找轮廓(Finds contours in a binary image):轮廓是形状分析和物体检测和识别的 ...

  8. canvas学习笔记、小函数整理

    http://bbs.csdn.net/topics/391493648 canvas实例分享 2016-3-16 http://bbs.csdn.net/topics/390582151 html5 ...

  9. Go语言学习笔记七: 函数

    Go语言学习笔记七: 函数 Go语言有函数还有方法,神奇不.这有点像python了. 函数定义 func function_name( [parameter list] ) [return_types ...

随机推荐

  1. 洛谷 P2498 [SDOI2012]拯救小云公主 解题报告

    P2498 [SDOI2012]拯救小云公主 题目描述 英雄又即将踏上拯救公主的道路-- 这次的拯救目标是--爱和正义的小云公主. 英雄来到\(boss\)的洞穴门口,他一下子就懵了,因为面前不只是一 ...

  2. 解题:USACO06DEC Milk Patterns

    题面 初见SA 用了一个常见的按$height$分组的操作:二分答案,然后按$height$分组,遇到一个$height$小于$mid$的就丢进下一组并更新答案,如果最多的那组不少于$k$个说明可行 ...

  3. nodejs使用场景

    NodeJS的工作原理其实就是事件循环.可以说每一条NodeJS的逻辑都是写在回调函数里面的,而回调函数都是有返回之后才异步执行的! 既然NodeJS处理并发的能力强,但处理计算和逻辑的能力反而很弱, ...

  4. 使用Eclipse进行SWT编程

    使用Eclipse进行SWT编程 1. 为什么要使用SWT? SWT是IBM开发一套跨平台的GUI开发框架.为什么IBM要创建另一种GUI呢?为什么他们不使用现有的JavaGUI框架呢?要回答这些问题 ...

  5. [DeeplearningAI笔记]序列模型3.6Bleu得分/机器翻译得分指标

    5.3序列模型与注意力机制 觉得有用的话,欢迎一起讨论相互学习~Follow Me 3.6Bleu得分 在机器翻译中往往对应有多种翻译,而且同样好,此时怎样评估一个机器翻译系统是一个难题. 常见的解决 ...

  6. OpenCV---Canny边缘提取

    一:Canny算法介绍 Canny 的目标是找到一个最优的边缘检测算法,最优边缘检测的含义是: 好的检测- 算法能够尽可能多地标识出图像中的实际边缘. 好的定位- 标识出的边缘要尽可能与实际图像中的实 ...

  7. Google Map API使用详解(一)——Google Map开发背景知识

    一.谷歌地图主页 谷歌地图对应不同的地区都会有一些专门的主页,首次登陆时会显示这些地区.比如,香港的:http://maps.google.com.hk,台湾的:http://maps.google. ...

  8. CF544 C 背包 DP

    n个人写m行代码,第i人写一行代码有a[i]个bug,问总bug数不超过b的不同方案数. 其实就是个背包,dp[i][j][k]代表前i个人写了j行代码用了k个bug限度,然后随便转移一下就好了 /* ...

  9. 开发技巧:高效的使用 Response.Redirect

    我正在评估一个 ASP.NET Web 项目应用.它有一些可扩展性问题.意味着当网站访问量增加的时候.系统将会变得缓慢.当我查看应用日志.我找到了大量的 ThreadAbortException. 这 ...

  10. FastDFS图片服务器java后台的简单调用

    工具类: package com.liveyc.common.fdfs; import org.apache.commons.io.FilenameUtils; import org.csource. ...