C++多态和虚函数快速入门教程

Posted C语言学习联盟

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++多态和虚函数快速入门教程相关的知识,希望对你有一定的参考价值。


基类的指针也可以指向派生类对象,请看下面的例子:

    
      
      
    
  1. #include <iostream>

  2. using namespace std;

  3. //基类People

  4. class People{

  5. public:

  6. People(char *name, int age);

  7. void display();

  8. protected:

  9. char *m_name;

  10. int m_age;

  11. };

  12. People::People(char *name, int age): m_name(name), m_age(age){}

  13. void People::display(){

  14. cout<<m_name<<"今年"<<m_age<<"岁了,是个无业游民。"<<endl;

  15. }

  16. //派生类Teacher

  17. class Teacher: public People{

  18. public:

  19. Teacher(char *name, int age, int salary);

  20. void display();

  21. private:

  22. int m_salary;

  23. };

  24. Teacher::Teacher(char *name, int age, int salary): People(name, age), m_salary(salary){}

  25. void Teacher::display(){

  26. cout<<m_name<<"今年"<<m_age<<"岁了,是一名教师,每月有"<<m_salary<<"元的收入。"<<endl;

  27. }

  28. int main(){

  29. People *p = new People("王志刚", 23);

  30. p -> display();

  31. p = new Teacher("赵宏佳", 45, 8200);

  32. p -> display();

  33. return 0;

  34. }

运行结果:
王志刚今年23岁了,是个无业游民。
赵宏佳今年45岁了,是个无业游民。

我们直观上认为,如果指针指向了派生类对象,那么就应该使用派生类的成员变量和成员函数,这符合人们的思维习惯。但是本例的运行结果却告诉我们,当基类指针 p 指向派生类 Teacher 的对象时,虽然使用了 Teacher 的成员变量,但是却没有使用它的成员函数,导致输出结果不伦不类(赵宏佳本来是一名老师,输出结果却显示人家是个无业游民),不符合我们的预期。

换句话说,通过基类指针只能访问派生类的成员变量,但是不能访问派生类的成员函数。

为了消除这种尴尬,让基类指针能够访问派生类的成员函数,C++ 增加了虚函数(Virtual Function)。使用虚函数非常简单,只需要在函数声明前面增加 virtual 关键字。

更改上面的代码,将 display() 声明为虚函数:

    
      
      
    
  1. #include <iostream>

  2. using namespace std;

  3. //基类People

  4. class People{

  5. public:

  6. People(char *name, int age);

  7. virtual void display();  //声明为虚函数

  8. protected:

  9. char *m_name;

  10. int m_age;

  11. };

  12. People::People(char *name, int age): m_name(name), m_age(age){}

  13. void People::display(){

  14. cout<<m_name<<"今年"<<m_age<<"岁了,是个无业游民。"<<endl;

  15. }

  16. //派生类Teacher

  17. class Teacher: public People{

  18. public:

  19. Teacher(char *name, int age, int salary);

  20. virtual void display();  //声明为虚函数

  21. private:

  22. int m_salary;

  23. };

  24. Teacher::Teacher(char *name, int age, int salary): People(name, age), m_salary(salary){}

  25. void Teacher::display(){

  26. cout<<m_name<<"今年"<<m_age<<"岁了,是一名教师,每月有"<<m_salary<<"元的收入。"<<endl;

  27. }

  28. int main(){

  29. People *p = new People("王志刚", 23);

  30. p -> display();

  31. p = new Teacher("赵宏佳", 45, 8200);

  32. p -> display();

  33. return 0;

  34. }

运行结果:
王志刚今年23岁了,是个无业游民。
赵宏佳今年45岁了,是一名教师,每月有8200元的收入。

和前面的例子相比,本例仅仅是在 display() 函数声明前加了一个
virtual关键字,将成员函数声明为了虚函数(Virtual Function),这样就可以通过 p 指针调用 Teacher 类的成员函数了,运行结果也证明了这一点(赵宏佳已经是一名老师了,不再是无业游民了)。

有了虚函数,基类指针指向基类对象时就使用基类的成员(包括成员函数和成员变量),指向派生类对象时就使用派生类的成员。换句话说,基类指针可以按照基类的方式来做事,也可以按照派生类的方式来做事,它有多种形态,或者说有多种表现方式,我们将这种现象称为多态(Polymorphism)

上面的代码中,同样是
p->display();这条语句,当 p 指向不同的对象时,它执行的操作是不一样的。同一条语句可以执行不同的操作,看起来有不同表现方式,这就是多态。

多态是面向对象编程的主要特征之一,C++中虚函数的唯一用处就是构成多态。

C++提供多态的目的是:可以通过基类指针对所有派生类(包括直接派生和间接派生)的成员变量和成员函数进行“全方位”的访问,尤其是成员函数。如果没有多态,我们只能访问成员变量。

通过指针调用普通的成员函数时会根据指针的类型(通过哪个类定义的指针)来判断调用哪个类的成员函数,通过分析可以发现,这种说法并不适用于虚函数,虚函数是根据指针的指向来调用的,指针指向哪个类的对象就调用哪个类的虚函数。

借助引用也可以实现多态

引用在本质上是通过指针的方式实现的,既然借助指针可以实现多态,那么我们就有理由推断:借助引用也可以实现多态。

修改上例中 main() 函数内部的代码,用引用取代指针:

    
      
      
    
  1. int main(){

  2. People p("王志刚", 23);

  3. Teacher t("赵宏佳", 45, 8200);

  4. People &rp = p;

  5. People &rt = t;

  6. rp.display();

  7. rt.display();

  8. return 0;

  9. }

运行结果:
王志刚今年23岁了,是个无业游民。
赵宏佳今年45岁了,是一名教师,每月有8200元的收入。

由于引用类似于常量,只能在定义的同时初始化,并且以后也要从一而终,不能再引用其他数据,所以本例中必须要定义两个引用变量,一个用来引用基类对象,一个用来引用派生类对象。从运行结果可以看出,当基类的引用指代基类对象时,调用的是基类的成员,而指代派生类对象时,调用的是派生类的成员。

不过引用不像指针灵活,指针可以随时改变指向,而引用只能指代固定的对象,在多态性方面缺乏表现力,所以以后我们再谈及多态时一般是说指针。本例的主要目的是让读者知道,除了指针,引用也可以实现多态。

多态的用途

通过上面的例子读者可能还未发现多态的用途,不过确实也是,多态在小项目中鲜有有用武之地。

接下来的例子中,我们假设你正在玩一款军事游戏,敌人突然发动了地面战争,于是你命令陆军、空军及其所有现役装备进入作战状态。具体的代码如下所示:

    
      
      
    
  1. #include <iostream>

  2. using namespace std;

  3. //军队

  4. class Troops{

  5. public:

  6. virtual void fight(){ cout<<"Strike back!"<<endl; }

  7. };

  8. //陆军

  9. class Army: public Troops{

  10. public:

  11. void fight(){ cout<<"--Army is fighting!"<<endl; }

  12. };

  13. //99A主战坦克

  14. class _99A: public Army{

  15. public:

  16. void fight(){ cout<<"----99A(Tank) is fighting!"<<endl; }

  17. };

  18. //武直10武装直升机

  19. class WZ_10: public Army{

  20. public:

  21. void fight(){ cout<<"----WZ-10(Helicopter) is fighting!"<<endl; }

  22. };

  23. //长剑10巡航导弹

  24. class CJ_10: public Army{

  25. public:

  26. void fight(){ cout<<"----CJ-10(Missile) is fighting!"<<endl; }

  27. };

  28. //空军

  29. class AirForce: public Troops{

  30. public:

  31. void fight(){ cout<<"--AirForce is fighting!"<<endl; }

  32. };

  33. //J-20隐形歼击机

  34. class J_20: public AirForce{

  35. public:

  36. void fight(){ cout<<"----J-20(Fighter Plane) is fighting!"<<endl; }

  37. };

  38. //CH5无人机

  39. class CH_5: public AirForce{

  40. public:

  41. void fight(){ cout<<"----CH-5(UAV) is fighting!"<<endl; }

  42. };

  43. //轰6K轰炸机

  44. class H_6K: public AirForce{

  45. public:

  46. void fight(){ cout<<"----H-6K(Bomber) is fighting!"<<endl; }

  47. };

  48. int main(){

  49. Troops *p = new Troops;

  50. p ->fight();

  51. //陆军

  52. p = new Army;

  53. p ->fight();

  54. p = new _99A;

  55. p -> fight();

  56. p = new WZ_10;

  57. p -> fight();

  58. p = new CJ_10;

  59. p -> fight();

  60. //空军

  61. p = new AirForce;

  62. p -> fight();

  63. p = new J_20;

  64. p -> fight();

  65. p = new CH_5;

  66. p -> fight();

  67. p = new H_6K;

  68. p -> fight();

  69. return 0;

  70. }

运行结果:
Strike back!
--Army is fighting!
----99A(Tank) is fighting!
----WZ-10(Helicopter) is fighting!
----CJ-10(Missile) is fighting!
--AirForce is fighting!
----J-20(Fighter Plane) is fighting!
----CH-5(UAV) is fighting!
----H-6K(Bomber) is fighting!

这个例子中的派生类比较多,如果不使用多态,那么就需要定义多个指针变量,很容易造成混乱;而有了多态,只需要一个指针变量 p 就可以调用所有派生类的虚函数。

从这个例子中也可以发现,对于具有复杂继承关系的大中型程序,多态可以增加其灵活性,让代码更具有表现力。


小编整理了一套C语言学习资料,需要的话可以私信@C语言学习联盟回复领取资料即可,欢迎大家关注,有时间会及时分享相关技术博文,你的关注和点赞对小编都很重要,谢谢各位动动发财的手指点点关注啦~