- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章一文读懂C++ 虚函数 virtual由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
C++ 中的虚函数用于解决动态多态问题,虚函数的作用是允许在派生类中重新定义与积累同名的函数,并且可以通过基类指针或引用来访问基类和派生类中的同名函数.
首先写两个简单的类,类 B 继承自类 A,即 A 是基类,B 是派生类.
class A{public: void print(){ cout << "A" << endl; }};class B : public A {public: void print(){ cout << "B" << endl; }};int main(){ B b; //创建一个 B 类对象 b; A &a = b; //a 是 b 的一个 A 类引用; A *pa = &b; //pa 是一个指向 A 类对象的指针; a.print(); pa->print(); b.print(); return 0;}
程序中,A 类和 B 类均定义了一个同名函数 print ,但两个函数的功能不同,编译系统按照同名覆盖原则决定调用对象.
另外一点,引用的本质是指针常量,可以认为 a,pa 都指向了 b。( 注意区分常量指针与指针常量,常量指针可以类比于整型指针,即指向一个常量的指针,指针的指向可以修改;指针常量类比于整型常量,即一个指针是个常量,也就是指针只能固定的指向某一单元,指针常量的指向不可改而指向的值可以修改。) 。
int a, b;int * const p1 = &a; //指针常量const int *p2 = &b; //常量指针
执行函数后,我们发现结果为 。
因为 a 是 A 类的一个引用,所以 a 的 print( ) 依旧是 A 类的成员函数;pa 是 A 类的指针,同理;而 b 是 B 类的对象,调用的 print( ) 为 B 类的成员函数。简言之就是,没有 virtual 时,调用哪一类的成员函数取决于调用对象 a ,pa,b 在定义时的类型。而此时,若 B 类对象 b 想调用直接基类 A 的 print 函数,则应当 b.A::print( ).
这种 a,pa,b 能调用哪个同名函数在对象定义时已经确定好了的多态,我们称之为静态多态。什么是多态?同一个 print 函数在不同的对象中有不同的作用,这就呈现了多态.
这里再提一点,原本基类指针是用来指向基类对象的,如果用它指向派生类对象,此时基类指针指向的是派生类对象中的基类部分。在没有虚函数时,基类指针是无法调用派生类对象中的成员函数的.
而当我们在 A 类中 print( ) 前加上关键字 virtual,变成虚函数时 。
class A{public: virtual void print(){ cout << "A" << endl; }};class B : public A{public: void print(){ cout << "B" << endl; }};
再次执行主函数,结果为 。
这是因为 virtual 跟着对象走,即调用的 print( ) 究竟是 A 类还是 B 类的成员函数取决于“ 调用者 ” a,pa 所指的对象 b 属于哪一类,而不再是取决于 a,pa 本身在定义时的类型了.
这种用基类指针或引用指向某一派生类对象,从而能够调用指针指向的派生类对象中的函数的多态,我们称为动态多态,virtual 正是实现动态多态的关键字.
接着刚才的话题,在 A 类中有虚函数的前提下,我们继续讨论 。
class C{public: void print(){ cout << "C" << endl; }};int main(){ cout << "sizeof(A): " << sizeof(A) << endl; cout << "sizeof(B): " << sizeof(B) << endl; cout << "sizeof(C): " << sizeof(C) << endl; A a; B b; C c; cout << "sizeof(a): " << sizeof(a) << endl; cout << "sizeof(b): " << sizeof(b) << endl; cout << "sizeof(c): " << sizeof(c) << endl; return 0;}
执行结果为 。
为什么有虚函数的 A 类大小为 8 字节,继承 A 的 B 为 8 字节,而没有虚函数的 C 类是 1 字节呢?联想到 64 位操作系统下指针占8个字节内存,而 A 大小也是 8 字节,是巧合吗?事实上,在包含虚函数的类中,在该类的存储空间中,会有一个指向虚函数表的指针,正是这个指针使 A 的大小变为 8 字节。而指针所指的虚函数表本质上是 A 类中定义的所有虚函数名构成的列表。A 中只定义了一个虚函数 print( ) ,所以虚函数表中也只有一个虚函数名 print,通过这个虚函数名,再找到整个虚函数 print( ) 在内存中的存储位置 ( B 同理 ) .
虚函数表看不见摸不着,怎么确定它的存在呢?
int main(){ A a; B b; a.print(); b.print(); cout << "-------------------" << endl; typedef void (*func)(); //利用函数指针 func; ((func**)(&a))[0][0](); //((func**)(&a))[0] 代表对象 a 的内存空间中的第一个元素:指向虚函数表的指针; ((func**)(&b))[0][0](); //((func**)(&a))[0][0] 表示虚函数表中第一个函数名; return 0;}
从结果中我们可以发现,((func**)(&a))[0][0](); 等效于 a.print();,即确实证明了对象 a 的内存空间中存有一个指向虚函数表的指针,虚函数表的第一个函数名正是 print .
到此这篇关于一文读懂C++ 虚函数 virtual的文章就介绍到这了,更多相关C++ 虚函数 virtual内容请搜索我以前的文章或继续浏览下面的相关文章希望大家以后多多支持我! 。
原文链接:https://blog.csdn.net/weixin_44636463/article/details/114336411 。
最后此篇关于一文读懂C++ 虚函数 virtual的文章就讲到这里了,如果你想了解更多关于一文读懂C++ 虚函数 virtual的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!