C++虚函数表
C语言中的函数指针应用

C语言中的函数指针应用在C语言中,函数指针是一种非常强大的工具,它允许我们将函数作为参数传递给其他函数,或者将函数赋值给其他函数指针变量,从而实现更加灵活和动态的编程。
下面我们来看一些C语言中函数指针的常见应用。
1. 回调函数函数指针最常见的用途之一就是实现回调函数。
回调函数是指当某个事件发生时,通过调用事先注册好的函数来处理该事件。
例如,在GUI编程中,当用户点击按钮时,可以通过函数指针来调用相应的处理函数。
这种机制可以实现程序的灵活性和可扩展性。
2. 函数指针数组函数指针还可以用于构建函数指针数组,通过数组来存储多个函数的地址。
这样可以根据需要动态地选择并调用不同的函数。
例如,可以通过函数指针数组来实现一个简单的命令调度器,根据用户输入的命令选择执行对应的函数。
3. 函数指针作为函数返回值在C语言中,函数指针还可以作为函数的返回值。
这种情况通常发生在一些高级的应用场景中,例如函数指针用于实现函数工厂模式,根据不同的参数返回不同的函数指针,从而实现动态创建不同的函数对象。
4. 函数指针作为结构体成员函数指针也可以作为结构体的成员,用于实现结构体的多态。
这种方法类似于面向对象编程中的虚函数表,通过为结构体定义不同的函数指针来实现结构体对象的多态行为。
总的来说,函数指针是C语言中一项非常有用的特性,可以有效提高程序的灵活性和可维护性。
但是由于函数指针的复杂性和难以理解性,需要仔细考虑在何种情况下使用函数指针,以避免造成代码的混乱和难以维护。
同时,通过合理地利用函数指针,可以使程序结构更加清晰,逻辑更加严密,从而提高代码的可读性和可维护性。
C--程序设计--第10章-多态性及虚函数

使用重载函数注意:
不要使用重载函数描述不相干的函数 在类中,构造函数和普通成员函数均可以
重载 避免与函数的默认参数产生二义性
二、运算符重载
运算符重载(operate overloading)就是 赋予已有的运算符多重含义。
运算符重载实质是函数重载,运算符重载 的选择与函数重载类似,会根据运算符的 操作数的类型、个数和顺序来进行运算符 函数的选择。
#include<iostream.h> str#iinngc:l:usdter<isntgr(icnhga.rh>*s) v{}ossccsssc{s{{ittohtttolsstlsssls*drruarrrueptrepttepsi1trii3tc{pn=rin=rrn=pmn.<nn.<lprgncngncign=agp<*ggp<auitepgtepnte'irssrssbv\hwy:hwyghwnsit1ssitsla0=(:=(:=(tnr=ttnrit'scssscs:sc)rt1"rrt3scesss~ivci;thpt1hpsih1(.T23(.t::tttsnohn}ra,r.a,tza()gh(()grrrrttiatlrsilrsrer";eass;eiiiirdre[)ne[1i;[Ttt1ttnnnniglnl;gnl.nlhl)rlggggnep*e(e}(gesgeiei;2e(((gtrsnsnstnp(nsns)ncsi(lipg)gthg)ig(;(htn)en;t;tr;t;nti)a)artnthhih}ths<<ri{((;+n++<p<snd))}1g1s1aere*ige;]]i]nonszl{{;&;z;ddgd)&eercseelrl;s=teo1)m;a;/18etu)om/)0ut..;)构sr<""/;pn<造);//;s;/复}lp函构e<制n<数造ge构tn函hd造;l数};重} 载
C++面试问答100道

C++经典⾯试题100例及答案1. ⾯向对象的程序设计思想是什么?答:把数据结构和对数据结构进⾯操作的⾯法封装形成⾯个个的对象。
2. 什么是类?答:把⾯些具有共性的对象归类后形成⾯个集合,也就是所谓的类。
3. 对象都具有的两⾯⾯特征是什么?分别是什么含义?答:对象都具有的特征是:静态特征和动态特征。
静态特征是指能描述对象的⾯些属性(成员变量),动态特征是指对象表现出来的⾯为(成员函数)4. 在头⾯件中进⾯类的声明,在对应的实现⾯件中进⾯类的定义有什么意义?答:这样可以提⾯编译效率,因为分开的话只需要编译⾯次⾯成对应的.obj⾯件后,再次应⾯该类的地⾯,这个类就不会被再次编译,从⾯⾯⾯的提⾯了编译效率。
5. 在类的内部定义成员函数的函数体,这种函数会具备那种属性?答:这种函数会⾯动为内联函数,这种函数在函数调⾯的地⾯在编译阶段都会进⾯代码替换。
6. 成员函数通过什么来区分不同对象的成员数据?为什么它能够区分?答:通过this指针指向对象的⾯地址来区分的。
7. C++编译器⾯动为类产⾯的四个缺省函数是什么?答:默认构造函数,拷贝构造函数,析构函数,赋值函数。
8. 拷贝构造函数在哪⾯种情况下会被调⾯?答:1.当类的⾯个对象去初始化该类的另⾯个对象时;2.如果函数的形参是类的对象,调⾯函数进⾯形参和实参结合时;3.如果函数的返回值是类对象,函数调⾯完成返回时。
9.构造函数与普通函数相⾯在形式上有什么不同?答:1.构造函数是类的⾯种特殊成员函数,⾯般情况下,它是专门⾯来初始化对象成员变量的。
2.构造函数的名字必须与类名相同,它不具有任何类型,不返回任何值。
10. 什么时候必须重写拷贝构造函数?答:当构造函数涉及到动态存储分配空间时,要⾯⾯写拷贝构造函数,并且要深拷贝。
11. 构造函数的调⾯顺序是什么?答:1.先调⾯基类构造函数2.按声明顺序初始化数据成员3.最后调⾯⾯⾯的构造函数。
12. 哪⾯种情况必须⾯到初始化成员列表?答:1.类的成员是常量成员初始化;2.类的成员是对象成员初始化,⾯该对象没有⾯参构造函数。
CPP复习题答案

13. 派生类的对象对它的基类成员中 ( A ) 是可以采用对象·或者对象指针->的方 B. 公有继承的私有成员 D. 私有继承的公有成员 C ) 。
14. 关于纯虚函数和抽象类的描述中,错误的是( B. 抽象类是指具有纯虚函数的类。
A. 纯虚函数是一种特殊的虚函数,它没有具体的实现。 C. 一个基类中说明有纯虚函数,该基类的派生类一定不再是抽象类。 D. 抽象类只能作为基类来使用,其纯虚函数的实现由派生类给出。 15.下列说法错误的是( C )。 A.若语言只支持类而不支持多态,则不能称为面向对象的。 B.在运行时根据其类型确定调用哪个函数的能力叫多态性。 C.静态多态性也是在运行时根据其类型确定调用哪个函数。 D.C++中的静态多态性是通过函数重载进行实现的。 16. ( D )不是构造函数的特征 A. 构造函数的函数名与类名相同。 B. 构造函数可以重载。 C. 构造函数可以设置缺省参数。 D. 构造函数必须指定类型说明。 17.下列标识符中, A 不是 C++的关键字; A. cin B. private C. this D. operator 18.下列标识符中, A. cout A. cout 18.下列标识符中, A A 不是 C++的关键字; C. this C. this D. template D. sizeof 不是 C++的关键字; 不是 C++的关键字; C. this D. sizeof B. virtual B. public
A. 缩短程序代码,少占用内存空间 B. 既可以保证程序的可读性,又能提高程序的运行效率 C. 占用内存空间少,执行速度快 D. 使程序的结构比较清晰 3. 重载函数是( A ) A. 以函数参数来区分,而不用函数的返回值来区分不同的函数 B. 以函数的返回值来区分,而不用函数参数来区分不同的函数 C. 参数表完全相同而返回值类型不同的两个或多个同名函数 D. 参数表和返回值类型都必须是不同的两个或多个同名函数 4. 在 C++中,数据封装要解决的问题是( A. 数据的规范化 C. 避免数据丢失 5. 下列特性中,( B A.继承 之间的关系是( A. 组合关系 B.内联函数 C )。 B. 间接关系 C. 继承关系 D. 直接关系 D )。 B. 便于数据转换 D. 防止不同模块之间数据的非法访问 )不是面向对象的程序设计的特征。 C.多态性 D.封装
C++语言中的虚函数研究

万方数据 万方数据 万方数据C++语言中的虚函数研究作者:徐启丰, 胡勇, 万玉成, XU Qifeng, HU Yong, WANG Yucheng作者单位:徐州空军学院,江苏,徐州,221000刊名:现代电子技术英文刊名:MODERN ELECTRONICS TECHNIQUE年,卷(期):2010,33(4)参考文献(14条)1.Stanley B Lippman;侯捷Inside the C++ Object Model 20012.蓝雯飞;陆际光C++面向对象程序设计中的多态性研究[期刊论文]-计算机工程与应用 2000(08)3.Bjarne Stroustrup;裘宗燕C++的设计与演化 20024.赵红超;方金云;唐志敏C++的动态多态和静态多态[期刊论文]-计算机工程 2005(20)5.蓝雯飞C++中的多态性及其应用 1998(07)6.袁亚丽;肖桂云C++中虚函数的实现技术研究[期刊论文]-河北北方学院学报(自然科学版) 2006(05)7.Scott Mayers More Effective C++ 19968.和力;吴丽贤关于C++虚函数底层实现机制的研究与分析[期刊论文]-计算机工程与设计 2008(10)9.亚鹏关于C++中虚函数的几个问题 2006(02)10.Terrence W Pratt;傅育熙程序设计语言:设计与实现 200111.张昀C++中的多态性研究 2009(02)12.Bjarne Stroustrup The C++ Programming Language 200113.夏承遗;董玉涛;赵德新C++中虚函数的实现机制[期刊论文]-天津理工学院学报 2004(03)14.蓝雯飞C++语言中的面向对象特征探讨[期刊论文]-计算机工程与应用 2000(09)本文链接:/Periodical_xddzjs201004048.aspx。
C++中虚函数工作原理和(虚)继承类的内存占用大小计算

C++中虚函数工作原理和(虚)继承类的内存占用大小计算一、虚函数的工作原理虚函数的实现要求对象携带额外的信息,这些信息用于在运行时确定该对象应该调用哪一个虚函数。
典型情况下,这一信息具有一种被称为vptr(virtual table pointer,虚函数表指针)的指针的形式。
vptr 指向一个被称为vtbl(virtual table,虚函数表)的函数指针数组,每一个包含虚函数的类都关联到vtbl。
当一个对象调用了虚函数,实际的被调用函数通过下面的步骤确定:找到对象的vptr 指向的vtbl,然后在vtbl 中寻找合适的函数指针。
虚拟函数的地址翻译取决于对象的内存地址,而不取决于数据类型(编译器对函数调用的合法性检查取决于数据类型)。
如果类定义了虚函数,该类及其派生类就要生成一张虚拟函数表,即vtable。
而在类的对象地址空间中存储一个该虚表的入口,占4个字节,这个入口地址是在构造对象时由编译器写入的。
所以,由于对象的内存空间包含了虚表入口,编译器能够由这个入口找到恰当的虚函数,这个函数的地址不再由数据类型决定了。
故对于一个父类的对象指针,调用虚拟函数,如果给他赋父类对象的指针,那么他就调用父类中的函数,如果给他赋子类对象的指针,他就调用子类中的函数(取决于对象的内存地址)。
虚函数需要注意的大概就是这些个地方了,之前在More effective C++上好像也有见过,不过这次在Visual C++权威剖析这本书中有了更直白的认识,这本书名字很牛逼,看看内容也就那么回事,感觉名不副实,不过说起来也是有其独到之处的,否则也没必要出这种书了。
每当创建一个包含有虚函数的类或从包含有虚函数的类派生一个类时,编译器就会为这个类创建一个虚函数表(VTABLE)保存该类所有虚函数的地址,其实这个VTABLE的作用就是保存自己类中所有虚函数的地址,可以把VTABLE形象地看成一个函数指针数组,这个数组的每个元素存放的就是虚函数的地址。
C#虚函数virtual详解

中实现多态的关键之一。
还是上面的例子(C#):
class 飞禽 { public string wing; // 翅膀 public string feather; // 羽毛 …… // 其它属性和行为 public virtual bool Fly() // 利用关键字 virtual 来定义为虚拟函数,这是一个热点 { // 空下来让子类去实现 } } class 麻雀 : 飞禽 // 麻雀从飞禽继承而来 { …… // 定义麻雀自己特有的属性和行为 public override bool Fly() // 利用关键字 override 重载飞翔动作,实现自己的飞翔 { …… // 实现麻雀飞的动作 } } class 鹤 : 飞禽 // 鹤从飞禽继承而来 { …… // 定义鹤自己的特有的属性和行为 public override bool Fly() // 利用关键字 override 重载实现鹤的飞翔 {
d = new D(); // 实例化 d 对象,D 是 d 的实例类 a.Func(); // 执行 a.Func:1.先检查申明类 A 2.检查到是虚拟方法 3.转去检查实例 类 A,就为本身 4.执行实例类 A 中的方法 5.输出结果 Func In A b.Func(); // 执行 b.Func:1.先检查申明类 A 2.检查到是虚拟方法 3.转去检查实例 类 B,有重载的 4.执行实例类 B 中的方法 5.输出结果 Func In B c.Func(); // 执行 c.Func:1.先检查申明类 A 2.检查到是虚拟方法 3.转去检查实例 类 C,无重载的 4.转去检查类 C 的父类 B,有重载的 5.执行父类 B 中的 Func 方法 5.输 出结果 Func In B d.Func(); // 执行 d.Func:1.先检查申明类 A 2.检查到是虚拟方法 3.转去检查实例 类 D,无重载的(这个地方要注意了,虽然 D 里有实现 Func(),但没有使用 override 关键 字,所以不会被认为是重载) 4.转去检查类 D 的父类 A,就为本身 5.执行父类 A 中的 Func 方法 5.输出结果 Func In A D d1 = new D(); d1.Func(); // 执行 D 类里的 Func(),输出结果 Func In D Console.ReadLine(); } } } 3.
C++经典笔试题(附答案)

8:下列多重继承时的二义性问题如何解决?class A{ //类A的定义public:void print () {cout<<"Hello, this is A"<<endl;}};class B{ //类B的定义public:void print () {cout<<"Hello, this is B"<<endl;}};class C : public A, public B{ //类C由类A和类B共同派生而来public:void disp () {print ();}//编译器无法决定采用A类中定义的版本还是B类中的版本};解答:若两个基类中具有同名的数据成员或成员函数,应使用成员名限定来消除二义性,如:void disp () {A: : print () ; //加成员名限定A::}print()但更好的办法是在类C中也定义一个同名print函数,根据需要调用A::还是Bprint(),从而实现对基类同名函数的隐藏::9:下列公共基类导致的二义性如何解决?class A{ //公共基类public: //public 成员列表void print(){cout << "this is x in A: " <<endl;}class B: public A{};class C: public A{};class D : public B, public C{};void main(){D d;//声明一个D类对象dA* pa=(A*) &d; //上行转换产生二义性d.print () ; //print ()具有二义性,系统不知道是调用B类的还是C类的print ()函数}注意:把子类的指针或引用转换成基类指针或引用是上行转换,把基类指针或引用转换成子类指针或引用是下行转换。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
类只有2个变量,但是大小却是12,因为此类包含了虚函数!所以,另外4个字节其实就是虚函数表指针的大小!说白了,虚函数的原理就是准备一个虚函数表的指针,这个指针指向这个类所有的虚函数,然后用这个虚函数表指针访问所有的虚函数!并且,这个指向虚函数表的指针就保存在0043DFCC 这个地址处。
可以看下面的图来得到验证!
类ClassB继承自ClassA!解释如下:指向ClassA虚函数表指针保存在009C7834这个地址,但是指向ClassB虚函数表的指针保存在009C7924这个地址处。
但是由于ClassB没有修改虚函数,那么猜测4个对象应该指向同一个虚函数其实就是ClassAVirtualFunction()这个函数,至于是不是呢?继续看图!
对了吧?4个对象共同指向了00D01091这个地址,其实说白了这个但是就是虚函数的地址。
因为子类没有改写虚函数,所以子类指向的虚函数地址和父类指向的虚函数地址是一样的,如果我们改写虚函数的话,那么子类指向的虚函数地址就应该是变化的了!
可以看到,ClassB没有改写虚函数,但是ClassC重载了父类的虚函数。
所以
ClassB和ClassA指向了相同的虚函数地址。
但是ClassC由于重写了父类的虚函数,所以指向了新的函数地址!。