C++中的虚函数(一)

2016-01-29 12:23 18 1 收藏

C++中的虚函数(一),C++中的虚函数(一)

【 tulaoshi.com - C语言心得技巧 】

C++中的虚函数(一)


作者:aber

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com)




    虽然很难找到一本不讨论多态性的C++书籍或杂志,但是,大多数这类讨论使多态性和C++虚函数的使用看起来很难。我打算在这篇文章中通过从几个方面和结合一些例子使读者理解在C++中的虚函数实现技术。说明一点,写这篇文章只是想和大家交流学习经验因为本人学识浅薄,难免有一些错误和不足,希望大家批评和指正,在此深表感谢!

一、 基本概念
    首先,C++通过虚函数实现多态."无论发送消息的对象属于什么类,它们均发送具有同一形式的消息,对消息的处理方式可能随接手消息的对象而变"的处理方式被称为多态性。"在某个基类上建立起来的类的层次构造中,可以对任何一个派生类的对象中的同名过程进行调用,而被调用的过程提供的处理可以随其所属的类而变。"虚函数首先是一种成员函数,它可以在该类的派生类中被重新定义并被赋予另外一种处理功能。

二、 虚函数的定义与派生类中的重定义

class 类名{public:       virtual 成员函数说明;}class 类名:基类名{   public:          virtual 成员函数说明;}      
三、虚函数在内存中的结构

1.我们先看一个例子:
#include "iostream.h"#include "string.h"class A {public:virtual void fun0() { cout << "A::fun0" << endl; }};int main(int argc, char* argv[]){A  a;cout << "Size of A = " << sizeof(a) << endl;return 0;}      
结果如下:Size of A = 4

2.如果再添加一个虚函数:virtual void fun1() { cout << "A::fun" << endl;}
得到相同的结果。如果去掉函数前面的virtual修饰符
class A {public:void fun0() { cout << "A::fun0" << endl; }};int main(int argc, char* argv[]){A  a;cout << "Size of A = " << sizeof(a) << endl;return 0;}      
结果如下:Size of A = 1
 
3.在看下面的结果:
class A {public:virtual void fun0() { cout << "A::fun0" << endl; }int a;int b;};int main(int argc, char* argv[]){A  a;cout << "Size of A = " << sizeof(a) << endl;return 0;}      
结果如下:Size of A = 12

其实虚函数在内存中结构是这样的:


图一

   在window2000下指针在内存中占4个字节,虚函数在一个虚函数表(VTABLE)中保存函数地址。在看下面例子。
class A {public:virtual void fun0() { cout << "A::fun0" << endl; }virtual void fun1() { cout << "A::fun1" << endl; }int a;int b;};int main(int argc, char* argv[]){A  a;cout << "Size of A = " << sizeof(a) << endl;return 0;}      
结果如下:结果如下:
Size of A = 4

   虚函数的内存结构如下,你也可以通过函数指针,先找到虚函数表(VTABLE),然后访问每个函数地址来验证这种结构,在国外网站作者是:Zeeshan Amjad写的"ATL on the Hood中有详细介绍"


图二

4.我们再来看看继承中虚函数的内存结构,先看下面的例子
class A {public:virtual void f() { }};class B {public:virtual void f() { }};class C {public:virtual void f() { }};class Drive : public A, public B, public C {};int main() {Drive d;cout << "Size is = " << sizeof(d) << endl;return 0;}      
结果如下:Size is = 12 ,相信大家一看下面的结构图就会很清楚,


图三

5.我们再来看看用虚函数实现多态性,先看个例子:
class A {public:virtual void f() { cout << "A:
                        

来源:http://www.tulaoshi.com/n/20160129/1485882.html

延伸阅读
数学库函数声明在 math.h 中,主要有: 1、abs(x) 求整型数x 的绝对值 2、cos(x) x(弧度)的余弦 3、fabs(x) 求浮点数x 的绝对值 4、ceil(x) 求不小于x 的最小整数 5、floor(x) 求不大于x 的最小整数 6、log(x) 求x 的自然对数 7、log10(x) 求x 的对数(底为10) 8、pow(x, y) 求x 的y 次方 函数说明: pow()用来计算以x为底的y次...
  函数重载是用来描述同名函数具有相同或者相似功能,但数据类型或者是参数不同的函数治理操作的称呼。 !-- frame contents -- !-- /frame contents -- 我们来举一个实际应用中的例子来说明问题:我们要进行两种不同数据类型的和操作为了实现它,在c语言中我们就要写两个不同名称的涵数来进行区分例...
函数参数的求值顺序 当一个函数带有多个参数时,C++语言没有规定在函数调用时实参的求值顺序。而编译器根据对代码进行优化的需要自行规定对实参的求值顺序。 !-- frame contents -- !-- /frame contents -- 有的编译器规定自左至右,有的编译器规定自右至左,这种对求值顺序的不同规定,对一般参数来讲没有影响。但是,假...
/*程序作者:管宁 站点:www.cndev-lab.com 所有稿件均有版权,如要转载,请务必闻名出处和作者*/ #include stdio.h main() { void search_score(); /* 定义自定义涵数类型为不返回型 */ void count_avg(); /* 定义自定义涵数类型为不返回型 */ static float a[3][4]={{97,45.5,66,77},{88,92.5...
《c++编程思想》上说一个类如果没有拷贝函数,那么编译器就会自动创建一个默认的拷贝函数。下面就让我们看一下真实的情况。 首先看一个简单的类X,这个类没有显示定义拷贝构造函数。 c++源码如下: 代码如下: class X { private:     int i;     int j; }; int main() {     X x1;//先定义...

经验教程

666

收藏

71
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部