C++中的动态类型与动态绑定、虚函数、运行时多态的实现

动态类型与静态类型

静态类型

是指不需要考虑表达式的执行期语义,仅分析程序文本而决定的表达式类型。静态类型仅依赖于包含表达式的程序文本的形式,而在程序运行时不会改变。通俗的讲,就是上下文无关,在编译时就可以确定其类型。

动态类型

是指由一个左值表达式表示的左值所引用的最终派生对象的类型。例:如果一个静态类型为“类 B ”的指针p 指向一个继承于 B的类 D 的对象,则表达式 *p 的动态类型为“D”。引用按照相似规则处理。一般地讲,基类的指针和基类引用有可能为动态类型,就是说在运行之前不能够确定其真实类型。通常我们说,“基类指针指向的对象的实际/真正类型”或“基类引用所引用的对象的实际/真正类型”,就是它们的动态类型。很显然,这个动态类型是 C++ 语言通过指针和引用实现运行时多态能力的核心概念。

动态绑定与静态绑定

静态绑定:编译时绑定,通过对象调用
动态绑定:运行时绑定,通过地址实现

只有采用“指针->函数()”或“引用变量.函数()”的方式调用C++类中的虚函数才会执行动态绑定。对于C++中的非虚函数,因为其不具备动态绑定的特征,所以不管采用什么样的方式调用,都不会执行动态绑定。
即所谓动态绑定,就是基类的指针或引用有可能指向不同的派生类对象,对于非虚函数,执行时实际调用该函数的对象类型即为该指针或引用的静态类型(基类类型);而对于虚函数,执行时实际调用该函数的对象类型为该指针或引用所指对象的实际类型。比如下面代码:

class Base {
 public:
   void func() {
      cout << "func() in Base." << endl;
   }
   virtual void test() {
     cout << "test() in Base." << endl;
  }
};
 
class Derived : public Base {
  void func() {
     cout << "func() in Derived." << endl;
  }
  virtual void test() {
     cout << "test() in Derived." << endl; 
  }
};
 
int main() {
  Base* b;
  b = new Derived();
  b->func();
  b->test();
}
输出结果

由运行结果可以看到,b是一个基类指针,它指向了一个派生类对象,基类Base里面有两个函数,其中test为虚函数,func为非虚函数。因此,对于test就表现为动态绑定,实际调用的是派生类对象中的test,而func为非虚函数,因此它表现为静态绑定,也就是说指针类型是什么,就会调用该类型相应的函数。

虚函数、动态绑定、运行时多态之间的关系

简单地说,虚函数是动态绑定的基础;动态绑定是实现运行时多态的基础。
要触发动态绑定,需满足两个条件:
(1) 只有虚函数才能进行动态绑定,非虚函数不进行动态绑定。
(2) 必须通过基类类型的引用或指针进行函数调用。
通过基类指针或基类引用做形参,当实参传入不同的派生类(或基类)的指针或引用,在函数内部触发动态绑定,从而在运行时实现多态的。

下面通过实际例子才展示运行时多态的实现方式:
如下代码是一个Base基类和它的三个派生类Derived1,Derived2,Derived3。


class Base {
 public:
 void Print() {
   cout << "Print() from Base." << endl;
 }
 
 virtual void Display() {
   cout << "Display() from Base." << endl;
 }
};
 
class Derived1 : public Base {
 public:
 
 void Print() {
   cout << "Print() from Derived1." << endl;
 }
 
 void Display() {
   cout << "Display() from Derived2." << endl;
 }
};
 
class Derived2 : public Base {
 public:
 void Print() {
   cout << "Print() from Derived2." << endl;
 }
 
 void Display() {
   cout << "Display() from Derived2." << endl;
 }
};
 
class Derived3 : public Base {
 public:
 void Print() {
   cout << "Print() from Derived3." << endl;
 }
 
 void Display() {
   cout << "Display() from Derived3." << endl;
 }
};

下面两个全局函数分别以基类指针和基类引用作形参来实现运行时多态:

//通过基类引用作形参实现多态
void Polymorphic1(Base& b) {
  b.Print();
  b.Display();
}
 
//通过基类指针作形参实现多态
void Polymorphic2(Base* b) {
  b->Print();
  b->Display();
}

下面是测试代码:

int main() {
  Base b;
  Derived1 d1;
  Derived2 d2;
  Derived3 d3;
 
  vector<Base> base_vec;
  base_vec.push_back(b);
  base_vec.push_back(d1);
  base_vec.push_back(d2);
  base_vec.push_back(d3);
 
  vector<Base*> base_ptr_vec;
  base_ptr_vec.push_back(&b);
  base_ptr_vec.push_back(&d1);
  base_ptr_vec.push_back(&d2);
  base_ptr_vec.push_back(&d3);
 
  cout << endl << "对通过基类引用作形参实现多态进行测试" << endl;
  //对通过基类引用作形参实现多态进行测试 (测试方式错误)
  for (int i = 0; i != base_vec.size(); ++i) {
    Polymorphic1(base_vec[i]);
  }
 
  cout << endl << "对通过基类指针作形参实现多态进行测试" << endl;
  //对通过基类指针作形参实现多态进行测试
  for (int i = 0; i != base_vec.size(); ++i) {
    Polymorphic2(base_ptr_vec[i]);
  }
 
  cout << endl << "对通过基类引用作形参实现多态进行测试" << endl;
  //对通过基类引用作形参实现多态进行测试 
    Polymorphic1(b);
    Polymorphic1(d1);
    Polymorphic1(d2);
    Polymorphic1(d3);
 
    return 0;
}

测试结果如下图:


测试结果

我们看到第一组想对通过基类引用作形参实现多态进行测试,需要将不同派生类的对象作实参传过去。然而,把派生类放到基类的vector中存储的过程中,派生类对象被自动转换为基类对象了,因而实际存储的均为基类对象,所以再从vector中取出对象元素做实参传递的时候,传递的均为基类对象,所以测试失败。

而下面两组我们分别把不同派生类的指针和对象作实参进行测试,结果显示均实现了运行时多态:即传入不同的对象,就会调用该对象相应的Display函数,因为在基类中,Display为虚函数,所以这里它实现了对象的动态绑定,从而实现了运行时多态;与之做对比的Print函数在基类中为非虚构函数,因此对Print函数不会进行动态绑定,而是静态绑定:即基类指针只能调用基类中的Print函数。

参考文献:

  1. http://blog.csdn.net/lynnboy/article/details/154894
  2. http://www.cnblogs.com/chris12/archive/2012/10/28/2744131.html
  3. http://blog.csdn.net/livelylittlefish/article/details/2171521

转载自https://blog.csdn.net/iicy266/article/details/11906509

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容