virtual 函数会动态绑定,而virtual函数的缺省參数值是静态绑定的。

用一个base类型的指针p去指向一个derived类对象。通过p调用虚函数时,会动态绑定到实际所指对象中的函数;用一个derived类型的指针p2指向一个derived对象,由p2调用函数时,直接就是调用的derived中的函数。其參数值也是derived类中函数相应的參数值。

#include <iostream>
using namespace std;
class A
{
public:
enum Color {red,green,blue};
A()
{
cout<<"Construct A"<<endl;
}
virtual void dodraw()
{
draw();
}
private:
virtual void draw(Color color=blue) const
{
cout<<"base draw:"<<color<<endl;
}
};
class B:public A
{
public:
B()
{
cout<<"Construct B"<<endl;
}
virtual void dodraw()
{
draw();
} private:
virtual void draw(Color color=red) const
{
cout<<"derived draw:"<<color<<endl;
}
};
int main()
{
B b;
A *p=&b;
B *p2=&b;
p->dodraw();
p2->dodraw();
return 0;
}

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQveWlucWlhb2h1YQ==/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">

上面的程序中。在B类中都重写了A中的两个virtual函数。从而。通过p和p2调用dodraw函数,事实上都是调用的B中dodraw。而B中的dodraw函数调用B中draw函数,从而draw的实參就是B中的缺省实參:red

#include <iostream>
using namespace std;
class A
{
public:
enum Color {red,green,blue};
A()
{
cout<<"Construct A"<<endl;
}
virtual void dodraw()
{
draw();
}
private:
virtual void draw(Color color=blue) const
{
cout<<"base draw:"<<color<<endl;
}
};
class B:public A
{
public:
B()
{
cout<<"Construct B"<<endl;
} private:
virtual void draw(Color color=red) const
{
cout<<"derived draw:"<<color<<endl;
}
};
int main()
{
B b;
A *p=&b;
B *p2=&b;
p->dodraw();
p2->dodraw();
return 0;
}





上面的程序,B和A中dodraw函数都是来源于A中,由p和p2调用dodraw时。都是调用的A类中的dodraw函数,dodraw再调用draw函数时。会动态绑定到实际所指的对象B中的draw函数,这在执行期确定,但draw函数的缺省实參是静态绑定的,就是A类中的blue值,这在编译器就已经确定。所以,假设又一次定义继承而来的缺省參数值,会出现一种奇怪的现象:virtual函数用的derived类中的,而參数是用的base类中的。

effective C++中条款37:绝不又一次定义继承而来的缺省參数值的更多相关文章

  1. Effective C++ Item 37 绝不又一次定义继承而来的缺省參数值

    本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie 经验:绝对不要又一次而来的缺省參数值.由于缺省參数值都是静态绑定,而 virtual 函数 ...

  2. Effective C++:条款37:绝不又一次定义继承而来的缺省參数值

    因为又一次定义继承而来的non-virtual函数是不对的(见上一个条款),所以这个条款就将问题局限于:绝不又一次定义继承一个带有缺省參数值的virtual函数. (一) virtual函数是动态绑定 ...

  3. 《effective C++》:条款37——绝不重新定义继承而来的缺省参数值

    引子: 阿里的一道题: #include <IOSTREAM> using namespace std; class A{ public: ) { cout<<"a~ ...

  4. Effective C++ Item 36 绝不又一次定义继承而来的 non-virtual 函数

    本文为senlie原创.转载请保留此地址:http://blog.csdn.net/zhengsenlie 经验:绝对不要又一次定义继承而来的 non-virtual 函数 --> Item 7 ...

  5. 条款37:绝不重新定义继承而来的缺省参数值(Never redefine a function's inherited default parameter value)

    NOTE: 1.绝不重新定义一个继承而来的缺省参数值,因为缺省参数值都是静态绑定的,而virtual 函数-----你唯一应该覆盖的东西----却是动态绑定的.

  6. Effective C++ 50条款

    条款1:尽量用const和inline而不用#define 以const 行使常量折叠,用inline 代替常用操作的宏定义,而且库里面有很多常用函数可用.当然不能抛弃宏,宏还是很有用滴.偶最近才发现 ...

  7. 从effective C++中窥探C++11特性

    这几天在看effective C++3rd,这本书算是比较经典的一本入门C++的书了.虽然年代比较久远书中讲的好多模式已经被的新特性取代了,但是从这些旧的模式中可以了解到一些C++新特性设计的初衷,也 ...

  8. 读书笔记_Effective_C++_条款三十七:绝不重新定义继承而来的缺省参数值

    先看下面的例子: enum MyColor { RED, GREEN, BLUE, }; class Shape { public: ; }; class Rectangle: public Shap ...

  9. Effective C++:条款39:明智而审慎地使用private继承

    (一) (1)private继承意味着"依据某物实现出".仅仅有实现部分被继承.接口部分应略去: (2)它仅仅在软件"实现"层面上有意义,在软件"设计 ...

随机推荐

  1. CentOS系统php5.6安装ImageMagick处理webp格式图片

    1.先安装webp yum install libwebp 2.编译安装ImageMagick 之前有过yum安装的先卸载 yum remove ImageMagick 我使用的是老版本ImageMa ...

  2. 汉化 的 空指针 bug

    韩梦飞沙  韩亚飞  313134555@qq.com  yue31313  han_meng_fei_sha nulljava.lang.NullPointerException at com.an ...

  3. LOJ P3952 时间复杂度 noip 暴力 模拟

    https://www.luogu.org/problemnew/show/P3952 模拟,日常认识到自己zz. #include<iostream> #include<cstdi ...

  4. 保存全局Crash报告

    CrashHandler.java UncaughtException处理类,当程序发生Uncaught异常的时候,有该类来接管程序,并记录发送错误报告 package  com.amanda;imp ...

  5. ACM需要掌握算法

    数据结构 栈,队列,链表 哈希表,哈希数组 堆,优先队列 双端队列 可并堆 左偏堆 二叉查找树 Treap 伸展树 并查集 集合计数问题 二分图的识别 平衡二叉树 二叉排序树 线段树 一维线段树 二维 ...

  6. LNMP下FTP服务器Pureftpd的安装和使用

    LNMP一键安装包里的Pureftpd安装脚本是需要MySQL支持的,安装前需要已经安装好LNMP,这个自带一个PHP编写的用户管理界面. 安装 进入lnmp解压后的目录,执行:./pureftpd. ...

  7. Run native executable in Android App

    Run native executable in Android App Demo † Here's demo application called "Run Native Exe" ...

  8. 【转】教你用C#读写、删除、更新excel表格记录

    文章出处:http://blog.csdn.net/kuangshazi515/article/details/6585118 如下图所示,编一个程序,鼠标单击窗体视图区(右边)时,获取一对坐标(X, ...

  9. Gradle在Windows环境与Linux上配置有哪些不同?

    我的开发环境:Windows + Android Studio + Gradle 2.8 all + Jenkins 公司CI 服务器环境: Linux + Gradle 2.10 bin + Jen ...

  10. OpenCL 获取Program信息

    本程序生成一个OpenCL Program,然后获取Program的source,事实上它的source就是一个char[],能够打印出来. 然后我们把这些内容和原来文本的内容对照,看看是否是我们想要 ...