深入理解C++的静态绑定和动态绑定

拼搏现实的明天。 2022-06-13 09:29 364阅读 0赞

深入理解C++的静态绑定和动态绑定

理解静态绑定和动态绑定,需要先理解四个名词:

1、对象的静态类型:对象在声明时采用的类型,是在编译期确定的。

2、对象的动态类型:目前所指对象的类型,是在运行期决定的。

对象的动态类型可以更改,但是静态类型无法更改。例如:

  1. class A{...};
  2. class B:public A{...};
  3. class C:public A{...};
  4. C* pC = new C(); //pC的静态类型为它的声明类型C*,动态类型也是C*
  5. A* pA = new C(); //pA的静态类型为它的声明类型A*,动态类型为C*
  6. B* pB = new B();
  7. pB = pC; //pB的动态类型是可以更改的,现在它的动态类型为C*

3、静态绑定:在编译期,根据变量的静态类型(变量声明为基类还是派生类)来决定调用哪个函数,用基类声明的,就调用基类的方法,用派生类声明的就调用派生类的方法。

4、动态绑定:在运行期,根据变量实际指向的对象类型(该变量指向基类还是派生类)来决定调用哪个函数。

C++中非虚函数都是静态绑定的,虚函数是动态绑定的.

例如

  1. #include <iostream>
  2. using namespace std;
  3. class A
  4. {
  5. public:
  6. virtual void f1() //虚函数,采用动态绑定
  7. {
  8. cout<<"A::f1()"<<endl;
  9. }
  10. void f2() //非虚函数,采用静态绑定
  11. {
  12. cout<<"A::f2()"<<endl;
  13. }
  14. };
  15. class B:public A
  16. {
  17. public:
  18. virutal void f1()
  19. {
  20. cout<<"B::f1()"<<endl;
  21. }
  22. void f2() //这里重新定义了父类的non-virtual函数,是一个不好的设计,因为它会覆盖掉从父类继承而来的函数f1()
  23. {
  24. cout<<"B::f2"<<endl;
  25. }
  26. };
  27. int main()
  28. {
  29. B* pB = new B();
  30. A* pA = new B();
  31. pA->f1(); //调用B::f1()
  32. pA->f2(); //调用A::f2()
  33. pB->f1(); //调用B::f1()
  34. pB->f2(); //调用B::f2()
  35. return 0;
  36. }

由于函数f2()是non-virtual函数,所以采用静态绑定, 编译期会在编译期根据对象的静态类型选择函数 ,pA的静态类型为A*,所以pA->f2();调用A::f2()。函数f1()是virtual函数,采用的是动态绑定,虽然pA的静态类型为A*,但是pA的动态类型为B,所以pA->f1();调用B::f1()。

上面都是针对对象指针的情况,对于引用(reference)的情况同样适用.至于哪些是动态绑定,哪些是静态绑定,记住一句话:只有虚函数是动态绑定,其他都是静态绑定

特殊情况:当缺省参数和虚函数一起存在时,情况有些复杂,因为C++为了执行效率,对于虚函数的缺省参数,C++采用的是静态绑定。

例如:

  1. class A
  2. {
  3. public:
  4. virtual void printVal(int x = 5)
  5. {
  6. cout<<"x = "<<x<<endl;
  7. }
  8. };
  9. class B:public A
  10. {
  11. public:
  12. virtual void printVal(int x = 10)
  13. {
  14. cout<<"x = "<<x<<endl;
  15. }
  16. };
  17. B* pB = new B();
  18. A* pA = pB;
  19. pB->printVal();
  20. pA->printVal();

我们知道,pB->printVal();和pA->printVal();都将调用B::printVal(),但是它们的缺省参数是多少呢?由于缺省参数是静态绑定的,pB的静态类型为B*,所以pB->printVal()的缺省参数是10,而pA的静态类型为A*,所以pA->printVal()的缺省参数是5.

记住一句话:绝不重新定义继承而来的缺省参数值。具体可参考《Effective C++》这本书。

发表评论

表情:
评论列表 (有 0 条评论,364人围观)

还没有评论,来说两句吧...

相关阅读

    相关 动态静态详解

    动态绑定和静态绑定详解 弄清调用对象方法的执行过程十分重要.下面是调用过程的详细描述: 1.编译器查看对象的声明类型(注意和实际类型区分)和方法名。假设调用x.f(pa

    相关 C++动态静态

    为了支持c++的多态性,才用了动态绑定和静态绑定。理解他们的区别有助于更好的理解多态性,以及在编程的过程中避免犯错误。 需要理解四个名词: 1、对象的静态类型:对象在

    相关 动态静态

    为了支持c++的多态性,才用了动态绑定和静态绑定。理解他们的区别有助于更好的理解多态性,以及在编程的过程中避免犯错误,需要理解四个名词: 1、对象的静态类型:对象在声明时采