C++ Virtual详解

开发 后端
C++通过虚函数实现多态."无论发送消息的对象属于什么类,它们均发送具有同一形式的消息,对消息的处理方式可能随接手消息的对象而变"的处理方式被称为多态性。而虚函数是通过Virtual关键字来限定的。下面让我们一起来看。

Virtual是C++ OO(面向对象机制)机制中很重要的一个关键字。虚函数就是因为成员函数加了关键字virtual,可见它的重要性。

只要是学过C++的人都知道在类Base中加了Virtual关键字的函数就是虚拟函数(例如函数print),于是在Base的派生类Derived中就可以通过重写虚拟函数来实现对基类虚拟函数的覆盖。当基类Base的指针point指向派生类Derived的对象时,对point的print函数的调用实际上是调用了Derived的print函数而不是Base的print函数。这是面向对象中的多态性的体现。(关于虚拟机制是如何实现的,参见Inside the C++ Object Model ,Addison Wesley 1996)

  1. //---------------------------------------------------------  
  2. class Base  
  3. {  
  4. public:Base(){}  
  5. public:  
  6. virtual void print(){cout<<"Base";}  
  7. };  
  8. class Derived:public Base  
  9. {  
  10. public:Derived(){}  
  11. public:  
  12. void print(){cout<<"Derived";}  
  13. };  
  14. int main()  
  15. {  
  16. Base *point=new Derived();  
  17. point->print();  
  18. }   
  19. //---------------------------------------------------------  
  20. Output:  
  21. Derived  

这也许会使人联想到函数的重载,但稍加对比就会发现两者是完全不同的:

(1) 重载的几个函数必须在同一个类中; 

覆盖的函数必须在有继承关系的不同的类中

(2) 覆盖的几个函数必须函数名、参数、返回值都相同;

重载的函数必须函数名相同,参数不同。参数不同的目的就是为了在函数调用的时候编译器能够通过参数来判断程序是在调用的哪个函数。这也就很自然地解释了为什么函数不能通过返回值不同来重载,因为程序在调用函数时很有可能不关心返回值,编译器就无法从代码中看出程序在调用的是哪个函数了。

(3) 覆盖的函数前必须加关键字Virtual;

重载和Virtual没有任何瓜葛,加不加都不影响重载的运作。

关于C++的隐藏规则(引用自《高质量C++/C 编程指南》林锐 2001):

(1)如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无virtual

关键字,基类的函数将被隐藏(注意别与重载混淆)。

(2)如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有virtual
关键字。此时,基类的函数被隐藏(注意别与覆盖混淆)。

这里,林锐博士好像犯了个错误。C++并没有隐藏规则,林锐博士所总结的隐藏规则是他错误地理解C++多态性所致。下面请看林锐博士给出的隐藏规则的例证:

  1. #include <iostream.h>  
  2. class Base  
  3. {  
  4. public:  
  5. virtual void f(float x){ cout << "Base::f(float) " << x << endl; }  
  6. void g(float x){ cout << "Base::g(float) " << x << endl; }  
  7. void h(float x){ cout << "Base::h(float) " << x << endl; }  
  8. };  
  9. class Derived : public Base  
  10. {  
  11. public:  
  12. virtual void f(float x){ cout << "Derived::f(float) " << x << endl; }  
  13. void g(int x){ cout << "Derived::g(int) " << x << endl; }  
  14. void h(float x){ cout << "Derived::h(float) " << x << endl; }  
  15. };  
  16. void main(void)  
  17. {  
  18. Derived d;  
  19. Base *pb = &d;  
  20. Derived *pd = &d;  
  21. // Good : behavior depends solely on type of the object  
  22. pb->f(3.14f); // Derived::f(float) 3.14  
  23. pd->f(3.14f); // Derived::f(float) 3.14  
  24. // Bad : behavior depends on type of the pointer  
  25. pb->g(3.14f); // Base::g(float) 3.14  
  26. pd->g(3.14f); // Derived::g(int) 3 (surprise!)  
  27. // Bad : behavior depends on type of the pointer  
  28. pb->h(3.14f); // Base::h(float) 3.14 (surprise!)  
  29. pd->h(3.14f); // Derived::h(float) 3.14  

林锐博士认为bp 和dp 指向同一地址,按理说运行结果应该是相同的,而事实上运行结果不同,所以他把原因归结为C++的隐藏规则,其实这一观点是错的。决定bp和dp调用函数运行结果的不是他们指向的地址,而是他们的指针类型。“只有在通过基类指针或引用间接指向派生类子类型时多态性才会起作用”(C++ Primer 3rd Edition)。pb是基类指针,pd是派生类指针,pd的所有函数调用都只是调用自己的函数,和多态性无关,所以pd的所有函数调用的结果都输出Derived::是完全正常的;pb的函数调用如果有virtual则根据多态性调用派生类的,如果没有virtual则是正常的静态函数调用,还是调用基类的,所以有virtual的f函数调用输出Derived::,其它两个没有virtual则还是输出Base::很正常啊,nothing surprise! 

所以并没有所谓的隐藏规则,虽然《高质量C++/C 编程指南》是本很不错的书,可大家不要迷信哦。记住“只有在通过基类指针或引用间接指向派生类子类型时多态性才会起作用”。

纯虚函数:

C++语言为我们提供了一种语法结构,通过它可以指明,一个虚拟函数只是提供了一个可被子类型改写的接口。但是,它本身并不能通过虚拟机制被调用。这就是纯虚拟函数(pure virtual function)。 纯虚拟函数的声明如下所示:

  1. class Query {  
  2. public:  
  3. // 声明纯虚拟函数  
  4. virtual ostream& print( ostream&=cout ) const = 0;  
  5. // ...  
  6. }; 

这里函数声明后面紧跟赋值0。

包含(或继承)一个或多个纯虚拟函数的类被编译器识别为抽象基类。试图创建一个抽象基类的独立类对象会导致编译时刻错误。(类似地通过虚拟机制调用纯虚拟函数也是错误的例如)

  1. // Query 声明了纯虚拟函数  
  2. // 所以, 程序员不能创建独立的 Query 类对象  
  3. // ok: NameQuery 中的 Query 子对象  
  4. Query *pq = new NameQuery( "Nostromo" );  
  5. // 错误: new 表达式分配 Query 对象  
  6. Query *pq2 = new Query; 

抽象基类只能作为子对象出现在后续的派生类中。

如果只知道virtual加在函数前,那对virtual只了解了一半,virtual还有一个重要用法是virtual public,就是虚拟继承。虚拟继承在C++ Primer中有详细的描述,下面稍作修改的阐释一下:

在缺省情况下C++中的继承是“按值组合”的一种特殊情况。当我们写

  1. class Bear : public ZooAnimal { ... }; 

每个Bear 类对象都含有其ZooAnimal 基类子对象的所有非静态数据成员以及在Bear中声明的非静态数据成员类似地当派生类自己也作为一个基类对象时如:

  1. class PolarBear : public Bear { ... }; 

则PolarBear 类对象含有在PolarBear 中声明的所有非静态数据成员以及其Bear 子对象的所有非静态数据成员和ZooAnimal 子对象的所有非静态数据成员。在单继承下这种由继承支持的特殊形式的按值组合提供了最有效的最紧凑的对象表示。在多继承下当一个基类在派生层次中出现多次时就会有问题最主要的实际例子是iostream 类层次结构。ostream 和istream 类都从抽象ios 基类派生而来,而iostream 类又是从ostream 和istream 派生

  1. class iostream :public istream, public ostream { ... }; 

缺省情况下,每个iostream 类对象含有两个ios 子对象:在istream 子对象中的实例以及在ostream 子对象中的实例。这为什么不好?从效率上而言,存储ios 子对象的两个复本,浪费了存储区,因为iostream 只需要一个实例。而且,ios 构造函数被调用了两次每个子对象一次。更严重的问题是由于两个实例引起的二义性。例如,任何未限定修饰地访问ios 的成员都将导致编译时刻错误:到底访问哪个实例?如果ostream 和istream 对其ios 子对象的初始化稍稍不同,会怎样呢?怎样通过iostream 类保证这一对ios 值的一致性?在缺省的按值组合机制下,真的没有好办法可以保证这一点。

C++语言的解决方案是,提供另一种可替代按“引用组合”的继承机制虚拟继承(virtual inheritance )在虚拟继承下只有一个共享的基类子对象被继承而无论该基类在派生层次
中出现多少次共享的基类子对象被称为虚拟基类。

通过用关键字virtual 修政一个基类的声明可以将它指定为被虚拟派生。例如,下列声明使得ZooAnimal 成为Bear 和Raccoon 的虚拟基类:

  1. // 关键字 public 和 virtual的顺序不重要  
  2. class Bear : public virtual ZooAnimal { ... };  
  3. class Raccoon : virtual public ZooAnimal { ... }; 

虚拟派生不是基类本身的一个显式特性,而是它与派生类的关系如前面所说明的,虚拟继承提供了“按引用组合”。也就是说,对于子对象及其非静态成员的访问是间接进行的。这使得在多继承情况下,把多个虚拟基类子对象组合成派生类中的一个共享实例,从而提供了必要的灵活性。同时,即使一个基类是虚拟的,我们仍然可以通过该基类类型的指针或引用,来操纵派生类的对象。

没有虚函数的C++不能面向对象。从商业的角度看,面向对象能使系统具有可扩展性和可适应性,但只有C++类的语法而没有面向对象的话,就不会减少维护成本,而实际上会增加成本。所以没有虚函数是万万不能的,而关键字virtual则是关键。

【编辑推荐】

  1. C++多态技术的实现和反思
  2. 实现C++虚函数时相关注意事宜
  3. 如何实现C++虚函数表中的虚函数
  4. C++虚函数示例解读


 

责任编辑:于铁 来源: CSDN
相关推荐

2021-12-21 15:31:10

C++语言指针

2010-02-02 11:16:28

C++异常

2023-11-09 23:56:21

2010-01-18 16:17:53

C++代码

2024-01-22 10:49:55

C++for循环

2011-03-30 17:20:18

C++引用

2024-01-18 10:27:30

C++引用函数

2024-03-14 11:54:37

C++数据类型

2010-02-01 16:13:15

C++继承

2011-07-15 01:10:13

C++内存分配

2010-02-02 15:59:32

C++赋值函数

2011-08-04 13:38:01

Objective-C C++

2010-02-01 14:53:42

C++属性

2010-02-02 17:08:22

2010-02-02 10:33:22

C++引用

2010-02-06 10:50:10

C++统计对象个数

2009-10-09 14:24:27

2010-01-15 14:10:42

C++单元测试

2010-02-02 09:43:27

C++存储区域

2010-01-28 09:54:27

C++程序设计
点赞
收藏

51CTO技术栈公众号