在了解静态绑定和动态绑定之前,先了解什么是对象的静态类型,什么是对象的动态类型。


  • 对象的静态类型:对象在声明时采用的类型。是在编译器决定的。
  • 对象的动态类型:目前所指对象的类型。是在运行期决定的。
动态类型可以更改,而静态类型不可更改。看一个示例
  1. class Base
  2. {
  3. public:
  4. void setData(int i=10)
  5. {
  6. cout <<" virtual int Base::setData()"<<endl;
  7. }
  8. virtual int getData()
  9. {
  10. cout <<" virtual int Base::getData()"<<endl;
  11. }
  12. private:
  13. int m_value;
  14. };
  15. class Derive: public Base
  16. {
  17. public:
  18. void setData(int i=20)
  19. {
  20. cout <<" virtual int Derive::setData()"<<endl;
  21. }
  22. virtual int getData()
  23. {
  24. cout <<" virtual int Derive::getData()"<<endl;
  25. }
  26. };
  27. int _tmain(int argc, _TCHAR* argv[])
  28. {
  29. Derive *pd = new Derive;//pd的静态类型为Derive,动态类型也为Derive
  30. Base *pb = pd; //pb的静态类型为Base,动态类型为Derive
  31. return 0;
  32. }
搞清楚了什么是对象的静态类型,什么是对象的动态类型。
下面来介绍下什么是静态绑定,什么是动态绑定。

  • 静态绑定:绑定的对象是静态类型。某特性依赖于对象的静态类型,发生在编译期。
  • 动态绑定:绑定的对象是动态类型。有特性依赖于对象的动态类型,发生在运行期。
只有虚函数才使用的是动态绑定,其他的全部是静态绑定。

我们用pb,pd分别调用非虚函数,
  1. Derive *pd = new Derive;
  2. Base *pb = pd;
  3. pb->setData();
  4. pd->setData();
输出:

因为setData是静态绑定的,也就是编译器会在编译期根据对象的静态类型来选择函数。    pb的静态类型是Base,pd的静态类型是Derive。

我们用pd,pb分别调用虚函数

因为getData是虚函数,所以是动态绑定的。动态类型都是Derive*。
分别调用了基类的函数和派生类的函数。注:这样的设计特别不好,派生类与基类之间发生了名称遮掩。只是为了静态绑定和动态绑定才这么写的。如果派生类和基类是is-a关系,任何情况下都不要继承一个non-virtual函数。

注:指针和引用的动态类型与静态类型可能会不一致,但是对象的静态类型与动态类型是一致的。Derive d. d.setData()  d.getData()调用的都是派生类的成员函数。

当虚函数有缺省参数的时候,情况变得有点复杂。因为缺省参数采用的是静态绑定。
  1. class Base
  2. {
  3. public:
  4. virtual void getData(int i=10)
  5. {
  6. cout <<" virtual int Base::getData()" << i <<endl;
  7. }
  8. };
  9. class Derive: public Base
  10. {
  11. public:
  12. virtual void getData(int i = 20)
  13. {
  14. cout <<" virtual int Derive::getData()" << i <<endl;
  15. }
  16. };
  17. int _tmain(int argc, _TCHAR* argv[])
  18. {
  19. Derive *pd = new Derive;
  20. Base *pb = pd;
  21. pb->getData();
  22. pd->getData();
  23. return 0;
  24. }
输出:

虽然调用的都是缺省参数,由于缺省参数采用的是静态绑定。因此才会得到这样的结果。为了防止这样的情况出现:effective c++专门指出了一条:绝不重新定义继承而来的缺省参数。为了避免这样的结果,有一种方法就是使用NVI(non-virtual interface) :在基类中声明一个共有的非虚函数并给出缺省参数(因为是静态绑定),在其中调用私有的虚函数,这样在调用派生类方法时。由于共有非虚函数采取的是静态绑定,且派生类肯定继承了非虚函数。因此调用该函数时,默认的参数即可一直。

C++ 静态绑定与动态绑定------绝不重新定义继承而来的缺省参数的更多相关文章

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

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

  2. [EffectiveC++]item37:绝不重新定义继承而来的缺省参数值

    绝不重新定义继承而来的缺省参数值 静态类型 动态类型

  3. c++ 切勿重新定义继承来的带缺省参数的函数

    切勿重新定义继承来的带缺省参数的函数.我们知道,继承来的函数是virtual 的,至于原因在上一节中已经说明了,即“切勿重新定义父类non-virtual函数”.所以确切的描述应该是“切勿重新定义继承 ...

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

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

  5. Effective C++ -----条款37:绝不重新定义继承而来的缺省参数值

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

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

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

  7. 【36】绝不重新定义继承而来的non-virtual方法

    1.绝不重新定义继承而来的non-virtual方法,为什么? 首先想想,non-virtual方法是干什么的?也就是说,它的使用场景.父类的non-virtual方法,其实就是告诉子类,继承实现,子 ...

  8. Effective C++ -----条款36:绝不重新定义继承而来的non-virtual函数

    绝对不要重新定义继承而来的non-virtual函数.

  9. 条款36:绝不重新定义继承而来的non-virtual函数(Never redefine an inherited non-virtual function)

    NOTE: 1.绝对不要重新定义继承而来的non-virtual函数.

随机推荐

  1. Django高级实战 开发企业级问答网站✍✍✍

    Django高级实战 开发企业级问答网站 1. 创建项目与app 创建项目 django-admin startproject firstsite 创建app python manage.py sta ...

  2. NoSQL SimpleDB

  3. Android开发 EditText按回车按键后出现 focus search returned a view that wasn't able to take focus! 错误

    问题描述 将EditText这个View成为了ListView或者RecyclerView的item时,在按输入法的回车/下一步/next时会出现的 focus search returned a v ...

  4. linux 解压 WinRAR 压缩文件

    1.Download rar for linux wget http://www.rarlab.com/rar/rarlinux-x64-5.5.b1.tar.gz 2.Configure rar t ...

  5. TSP+期望——lightoj1287记忆化搜索,好题!

    感觉是很经典的题 记忆化时因为不好直接通过E判断某个状态是否已经求过,所以再加一个vis打标记即可 /*E[S][u]表示从u出发当前状态是S的期望*/ #include<bits/stdc++ ...

  6. VS2010-MFC(对话框:模态对话框及其弹出过程)

    转自:http://www.jizhuomi.com/software/160.html 一.模态对话框和非模态对话框 Windows对话框分为两类:模态对话框和非模态对话框. 模态对话框是这样的对话 ...

  7. mvvmlight下passwordBox绑定的解决方法

    在做公司的项目中,用到MVVMLight框架,需要做登陆注册的功能模块,但是在wpf中passwordBox没有内置绑定功能,因为出于安全的考虑.但是,博主又非常想要契合mvvm思想,一定要passw ...

  8. Compile_Netgen_WITH_OpenCascade

    title: Compile_Netgen_WITH_OpenCascade date: 2016-04-23 21:14:42 tags: 结合OCCT编译Netgen date: 2016-04- ...

  9. BZOJ 3236 AHOI 2013 作业 莫队+树状数组

    BZOJ 3236 AHOI 2013 作业 内存限制:512 MiB 时间限制:10000 ms 标准输入输出     题目类型:传统 评测方式:文本比较 题目大意: 此时己是凌晨两点,刚刚做了Co ...

  10. 【codeforces 500D】New Year Santa Network

    [题目链接]:http://codeforces.com/problemset/problem/500/D [题意] 有n个节点构成一棵树; 让你随机地选取3个不同的点a,b,c; 然后计算dis(a ...