虚函数与默认实参

最近开始重新阅读C++ Primer,不愧是经典书籍,每次翻阅都有新的收获。

今日看书的过程中发现一个过去没有注意的细节,以一篇博文记录,也希望阅读本文的朋友能注意。

第十五章

15.2.4

5.虚函数与默认实参

像其他任何函数一样,虚函数也可以有默认实参。通常,如果有用在给定调用中的默认实参值,该值将在编译时确定。如果一个调用省略了默认值的实参,则所用的值由调用该函数的类型定义,与对象的动态类型无关。通过基类的引用或指针调用虚函数时,默认实参为在基类虚函数生命中定义的值,如果通过派生类的指针或者引用调用虚函数,则默认实参是在派生类的版本中声明的值。

在同一虚函数的基类版本和派生类版本中使用不同的默认实参几乎一定会引起麻烦。如果通过基类的引用或指针调用虚函数,但实际执行的是派生类中定义的版本,这时就可能会出现问题。在这种情况下,为虚函数的基类版本定义的默认实参将传给派生类定义的版本,而派生类版本是不同的默认实参定义的。

 

如果基类virtual函数中的默认实参和派生类中的默认实参不同,则一定会引起错误。原因在于这个值是在编译时确定,而且只与调用函数的类型有关,而和动态类型无关。也就是说,当动态绑定发生的时候,想要使用派生类中的默认实参,是使用的确实基类的!

以一个例子说明这个问题。

[cpp]  view plain copy
  1. //MyClass.h  
  2.    
  3. #pragma once  
  4. class MyClass  
  5. {  
  6. public:  
  7.        
  8.     MyClass(void);  
  9.     ~MyClass(void);  
  10.     virtual void f(int i=2);  
  11. };  
  12.    
  13.    
  14. //MyClass.cpp  
  15. #include "MyClass.h"  
  16. #include<iostream>  
  17. using namespace std;  
  18.    
  19. MyClass::MyClass(void)  
  20. {  
  21.    
  22. }  
  23.    
  24.    
  25. MyClass::~MyClass(void)  
  26. {  
  27. }  
  28. void MyClass::f(int i)  
  29. {  
  30.     cout<<"In MyClass "<<i<<endl;  
  31. }  
  32. //Derived.h  
  33. #pragma once  
  34. #include "myclass.h"  
  35. class Derived :  
  36.     public MyClass  
  37. {  
  38. public:  
  39.     Derived(void);  
  40.     ~Derived(void);  
  41.     void f(int i=3);  
  42. };  
  43.    
  44.    
  45. //Derived.cpp  
  46. #include "Derived.h"  
  47. #include<iostream>  
  48. using namespace std;  
  49.    
  50. Derived::Derived(void)  
  51. {  
  52. }  
  53.    
  54.    
  55. Derived::~Derived(void)  
  56. {  
  57. }  
  58. void Derived::f(int i)  
  59. {  
  60.     cout<<"In Derived "<<i<<endl;  
  61. }  
  62. //main.cpp  
  63. #include<iostream>  
  64. #include"MyClass.h"  
  65. #include"Derived.h"  
  66. using namespace std;  
  67.    
  68. int main()  
  69. {  
  70.     MyClass my;  
  71.     Derived de;  
  72.        
  73.     my.f();  
  74.     de.f();  
  75.    
  76.     MyClass *p=&my;  
  77.    
  78.     p->f();  
  79.     p=&de;  
  80.     p->f();  
  81. }  
运行结果为:
In MyClass 2
In Derived 3
In MyClass 2
In Derived 2

从运行结果可以清晰地看书,尽管实现了动态绑定,但是在p指向de以后,调用p->f()使用的参数确是基类中的默认实参。

还要注意的就是,默认实参可以在源文件或者头文件中进行指定,但是只能指定一次。通常应该在函数声明中指定默认实参,并且将该声明放在合适的头文件中。如果未放在头文件,而是放在函数定义(源文件)中,那么,要想使用这个默认实参,必须包含此cpp源文件。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值