C++基础4:多态

多态

多态:不同类型对象调用相同接口完成不同的行为。

根据对象的实际类型不同,可以自动完成不同的行为,而仅仅通过一致的调用形式。

关系
没有封装就不能继承,没有继承就没有运行时的多态。

绑定

早绑定/overload 晚绑定/override
函数和运算符重载 继承与虚函数
  • 覆盖(重写override)与虚函数的使用

  • 覆盖成立的三个条件
    1.继承
    2.子类覆盖(重写)父类虚函数
    3.父类指针/引用指向子类

  • 多态产生的效果
    同样的调用语句实现不同的表现,框架的基石,设计模式的基础。

  • 虚函数定义规则:
      1.如果虚函数在基类与派生类中出现,仅仅是名字相同,而形式参数不同,或者是返回类型不同,有无const.那么即使加上了virtual关键字,也是不会覆盖。
      2.只有类的成员函数才能说明为虚函数,因为虚函数仅适合用与有继承关系的类对象,所以普通函数不能说明为虚函数。
      3.静态成员函数不能是虚函数,因为静态成员函数的特点是不受限制于某个对象。
      4.内联(inline)函数不能是虚函数,因为内联函数不能在运行中动态确定位置。即使虚函数在类的内部定义,但是在编译的时候系统仍然将它看做是非内联的。
      5.构造函数不能是虚函数,因为构造的时候,对象还是一片未定型的空间,只有构造完成后,对象才是具体类的实例。
      6.析构函数可以是虚函数,而且通常声明为虚函数。

子类以new方式实例化,指针赋值给父类指针,delete父类指针时,只调用父类的析构函数,不调用子类的析构函数。

class A{
  virtual void test(){}
};
class B:public A{
  void test(){}
};
class C:public B{
  void test(){}
};
class A{
  virtual void test(){}
};
class B:public A{
};
class C:public B{
  void test(){}
};

特例

class A{
  virtual A* test(){}
};
class B:public A{
  B* test(){}
};

构造函数和析构函数是否可以是虚函数?

  • 多态的实现原理分析
    当类中声明虚函数时,编译器会在类中生成一个虚函数表(基类和派生类中各自都会生成一个)

虚函数表
虚函数表是一个存储类成员函数指针的数据结构
虚函数表是由编译器自动生成和维护的
virtual函数会被编译器放入虚函数表中
存在虚函数时,每个对象当中都有一个指向虚函数表的指针(vptr指针)

  • 纯虚函数
    纯虚函数是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现方法。在基类中实现纯虚函数的方法是在函数原型后加=0
class 类名{
    virtual 返回值类型 函数(形参列表) = 0;
}
  • 抽象类
    包含纯虚函数的类称为抽象类。由于抽象类包含了没有定义的纯虚函数,所以不能定义抽象类的对象。

继承抽象类仍然可能是抽象类?
纯虚函数可以写函数体,但是不可能被调用?

在很多C++书籍上没有覆盖(overrride)这种称谓,而是直接称为多态(polymorphism)。例如:《C++Primer》、《设计模式》、《C++对象模型》、《深入浅出MFC》等。

维基百科polymorphism
Polymorphism can be distinguished by when the implementation is selected: statically (at compile time) or dynamically (at run time, typically via a virtual function).

小结

  • 重载覆盖的区别
No. 重载 覆盖
1 重载要求函数名相同,但是参数列表必须不同,返回值可以相同也可以不同。 覆盖要求函数名、参数列表、返回值必须相同。
2 在类中重载是同一个类中不同成员函数之间的关系。 在类中覆盖则是子类和基类之间不同成员函数之间的关系。
3 重载函数的调用是根据参数列表决定。 覆盖函数的调用是根据对象类型决定。
4 重载函数是在编译时确定调用一个函数。 覆盖函数是在执行时确定调用个函数。
  • 代码如何判断是否调用了覆盖函数

测验

  • 写出下列程序的执行结果,并分析结果。(如果程序编译有错,请分析原因,并写出解决方法)
#include <iostream>

using namespace std;

class A{
public:
    virtual void Func(){
        cout << "A::Func()" << endl;
    }
    void Func(int n){
        cout << "A::Func( " << n << " )" << endl;
    }
};

class B:public A {
public:
    using A::Func;
    void Func(){
        cout << "B::Func()" << endl;
    }
};

int main(){
    B b;
    b.Func();   
    b.Func(1);  

    A a(b);     
    a.Func();   
    a.Func(2);  

    A c;
    c = b;      
    c.Func();   
    c.Func(3);  

    A* pa = &b;  
    pa->Func();  
    pa->Func(4); 

    A& fa = b;   
    fa.Func();   
    fa.Func(5);  

    return 0;
}
  • 参考答案
#include <iostream>
using namespace std;

class A{
public:
    virtual void Func(){
        cout << "A::Func()" << endl;
    }
    void Func(int n){
        cout << "A::Func( " << n << " )" << endl;
    }
};

class B:public A {
public:
    // using A::Func;
    void Func(){
        cout << "B::Func()" << endl;
    }
};

int main(){
    B b;
    b.Func();   // B::Func()
    b.Func(1);  // 编译错误,同名隐藏

    A a(b);     // 赋值兼容+默认拷贝构造
    a.Func();   // A::Func()
    a.Func(2);  // 重载 A::Func(2)

    A c;
    c = b;      // 赋值兼容+赋值运算符重载
    c.Func();   // A::Func()
    c.Func(3);  // 重载 A::Func(3) 

    A* pa = &b;  // 赋值兼容
    pa->Func();  // 覆盖 B::Func()
    pa->Func(4); // 重载 A::Func(4) 

    A& fa = b;   // 赋值兼容
    fa.Func();   // 覆盖 B::Func()
    fa.Func(5);  // 重载 A::Func(5) 

    return 0;
}
  • 写出下列程序的执行结果,并分析结果。(如果程序编译有错,请分析原因,并写出解决方法)
#include <iostream>

using namespace std;

class Base{
    public:
    Base(){
        cout << "Base constuct" << endl;
    }
    ~Base(){
        cout << "Base destuct" << endl;
    }
};
class Member{
public:
    Member(){
        cout << "Member constuct" << endl;
    }
    ~Member(){
        cout << "Member destuct" << endl;
    }
};
class Derive:public Base{
public:
    Derive(){
        cout << "Derive constuct" << endl;
    }
    ~Derive(){
        cout << "Derive destuct" << endl;
    }
private:
    Member m;
};
int main(){
    Base* pD = new Derive;
    delete pD;
}
  • 写出下列程序的执行结果,并分析结果。(如果程序编译有错,请分析原因,并写出解决方法)
#include <iostream>
using namespace std;
class Base{
public:
  virtual void Func()const{ cout << "Base" << endl; }
};

class Derive : public Base {
public:
  void Func() const { cout << "Derive" << endl; }
};

int main() {
  Base* pB = new Derive;
  pB->Func();
  (*pB).Func();
}
  • 写出下列程序的执行结果,并分析结果。(如果程序编译有错,请分析原因,并写出解决方法)
#include <iostream>
using namespace std;

class A{
public:
    void Func(){ cout << "A::Func" << endl;}
};
class B:public A{
public:
    virtual void Func(){ cout << "B:Func" << endl;}
};
class C:public B{
public:
    virtual void Func(){ cout << "C:Func" << endl;}
};

int main(){
   A* a = new C;
   a->Func();
}
  • 写出下列程序的执行结果,并分析结果。(如果程序编译有错,请分析原因,并写出解决方法)
#include <iostream>
using namespace std;
class Base{
public:
  friend ostream& operator<<(ostream& os,const Base& b){ return os << "Base"; }
};

class Derive : public Base {
public:
  friend ostream &operator<<(ostream &os, const Derive &b) { return os << "Derive"; }
};

int main() {
  Base* pB = new Derive;
  cout << (*pB) << endl;
}

双重转发(Double Dispatch)

#include <iostream>
using namespace std;
class Base{
public:
  virtual ostream &Put(ostream &os) const { return os << "Base"; }
  friend ostream &operator<<(ostream &os,const Base &b) {return b.Put(os);}
};

class Derive : public Base {
public:
  ostream &Put(ostream &os) const { return os << "Derive"; }
  friend ostream &operator<<(ostream &os, const Derive &b) { return b.Put(os); }
};

int main() {
  Base* pB = new Derive;
  cout << (*pB) << endl;
}

写出下列执行结果

#include <iostream>
#include <vector>
using namespace std;

class Member{
public:
    Member(){cout << __func__ << endl;}
    ~Member(){cout << __func__ << endl;}
};
class Base{
public:
    Base(){cout << __func__ << endl;}
    ~Base(){cout << __func__ << endl;}
};
class Derive:public Base{
    Member m;
public:
    Derive(){cout << __func__ << endl;}
    ~Derive(){cout << __func__ << endl;}
};

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

推荐阅读更多精彩内容

  • 1.面向对象的程序设计思想是什么? 答:把数据结构和对数据结构进行操作的方法封装形成一个个的对象。 2.什么是类?...
    少帅yangjie阅读 4,994评论 0 14
  • 一个博客,这个博客记录了他读这本书的笔记,总结得不错。《深度探索C++对象模型》笔记汇总 1. C++对象模型与内...
    Mr希灵阅读 5,577评论 0 13
  • 早上,看到七七老师发的的禅绕画图案有些复杂,上班一天都在忙,想抽空画一下都没空,还加班到晚上八点才回,想着怕着能不...
    小伍的今天阅读 519评论 5 5
  • 昨天,随着威少最后时刻绝杀掘金的同时,西部的季后赛首轮对阵也正式揭晓。 分别为: 勇士(1)VS开拓者(8)快船V...
    篮球行为大赏阅读 275评论 0 3
  • 活在这珍贵的人间,太阳强烈,水波温柔。 一 去年暑假,和一个已经上了大学的学姐谈起她的大学生活。她说:“我爸希望我...
    洪予希1999阅读 214评论 0 1