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


  • 对象的静态类型:对象在声明时采用的类型。是在编译器决定的。
  • 对象的动态类型:目前所指对象的类型。是在运行期决定的。
动态类型可以更改,而静态类型不可更改。看一个示例
  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. java笔试之参数解析(正则匹配)

    在命令行输入如下命令: xcopy /s c:\ d:\, 各个参数如下: 参数1:命令字xcopy 参数2:字符串/s 参数3:字符串c:\ 参数4: 字符串d:\ 请编写一个参数解析程序,实现将命 ...

  2. 2 _ 基本框架 _ 检测VMX环境

    VT 是先开为大,谁先开谁上层,谁上层 谁权限大. 1 判断是否支持 VMX intel 白皮书 第3卷 传入 参数eax =1, 返回值 ecx 的第5位 = 1 则 surpported VMX. ...

  3. 【颓废篇】人生苦短,我用python(一)

    谁渴望来一场华(ang)丽(zang)的python交易! 最近突然产生了系统学习python的想法. 其实自从上次luogu冬日绘板dalao们都在写脚本就有这种想法了. 最近被计算几何势力干翻的我 ...

  4. Apache下更改.htaccess文件名称

    有时候我们需要更改.htaccess的名称以解决一些问题 比如:Eclipse下是不显示点开头的文件的 所以我们可以使用  Apache的AccessFileName来更改此配置文件的名称 Acces ...

  5. 01.MyBatis快速入门

    1.下载jar包 Mybatis包+数据库驱动包 https://github.com/mybatis/mybatis-3/releases 2.新建Java工程并导入jar包 3.创建数据库与表 C ...

  6. Android开发 处理内存申请失败的报错(Failed to allocate a 38189038 byte allocation with 16777216 free bytes and 20MB until OOM)

    问题原因 当你在操作图片或者其他大量文件数据时会出现:Failed to allocate a 38189038 byte allocation with 16777216 free bytes an ...

  7. 看 《android权威编程指南》 的笔记

    Android 编译工具 确保ant已安装并正常运行,android sdk的tools/和platform-tools目录包含在可执行文件的搜索路径中 切换到项目目录并执行以下命令: android ...

  8. 廖雪峰Java15JDBC编程-3JDBC接口-1JDBC简介

    JDBC:Java DataBase Connectivity Java程序访问数据库的标准接口 使用Java程序访问数据库的时候,Java代码并不是直接通过TCP连接去访问数据库,而是通过JDBC接 ...

  9. DRF 序列化组件单增

    目录 自定义序列化(矬) Serializer类(方式繁琐) 底层序列化类 UserSerializer 视图序列化步骤 底层反序列化类 UserCreatSerializer 视图反序列化步骤 Mo ...

  10. JAVA 垃圾回收读书笔记

    对象已死 在JAVA代码运行中,会不停的创建对象,因为内存空间不是无限的,Java虚拟机必须不停的回收无用的数据空间.那么虚拟机是怎么判断对象空间是需要被回收的呢,也就是怎么样的数据算是垃圾数据呢? ...