您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > C/C++/C#

C++面向对象继承与多态

时间:2020-09-21 12:21:14  来源:  作者:

前言

这一篇开始正式展开讲多态,以及我们为什么要使用多态。

多态

什么是多态

引用百度百科的定义:

多态(Polymorphism)按字面的意思就是“多种状态”。在面向对象语言中,接口的多种不同的实现方式即为多态。引用Charlie Calverts对多态的描述——多态性是允许你将父对象设置成为一个或更多的他的子对象相等的技术,赋值之后,父对象就可以根据当前赋值给它的子对象的特性以不同的方式运作(摘自“Delphi4 编程技术内幕”)。简单的说,就是一句话:允许将子类类型的指针赋值给父类类型的指针。

我的理解是:子类可以通过父类的指针或者引用,调用子类重写父类的虚函数,以达到一个类型多种状态的效果。

这听起来好像没有什么,我可以直接通过子类的对象调用成员函数不就行了,为啥还要舍近求远将其赋值到一个父类指针再调用呢?起初学习的时候我也不懂为什么,直到后来我遇到了一个很典型的例子才恍然大悟,这个例子我会在下面讲到。

多态的条件

前面也零零散散地介绍了C++多态的条件,这里总结一下:

  • 需要有继承
  • 需要使用父类的指针或引用
  • 父类需要有虚函数,子类要重写父类的虚函数

需要上转型是JAVA多态的条件,C++主要是通过使用父类的指针或者引用来实现的,也可以认为是一种上转型吧。正是因为使用了父类的指针或者引用,才使得他能够调用子类的虚函数,而不是像上一篇的上转型导致的静态绑定,最终调用的是父类的虚函数。我们通过以下代码来回顾一下:

class base {
public:
    virtual void do_something() //有虚函数 {
        cout << "I'm base class" << endl;
    }};class derived : public base            //有继承
{
public:
    void do_something() //子类重写了父类的虚函数 {
        cout << "I'm derived class" << endl;
    }};void fun1(base &b) //父类的引用 {
    b.do_something();
}
void fun2(base *b) //父类的指针 {
    b->do_something();
}
void fun3(base b) {
    b.do_something();
}
int main() {
    derived d;
    fun1(d);    //I'm derived class
    fun2(&d);    //I'm derived class
    fun3(d);    //I'm base class
    return 0;
}

fun1()和fun2()实现的过程都是动态绑定的,即运行时才动态确定要调用哪个函数。那他究竟是怎么实现的?

动态绑定的原理

大家还记得虚类的对象是有一个vptr,多个同类对象的vptr指向同一个vtable。动态绑定就是通过这个vptr间接寻址来实现的。虽然子类对象被赋值到了父类的指针,但是对象的vptr是没有改变的,他指向的还是子类的vtable。 所以父类指针去调用某个虚函数的时候,就会去vtable里面找函数入口,那找到的自然是子类的函数入口。所以他不是在编译期间就确定的,而是在代码运行到那一行的时候才找到的函数入口。

那为什么只有指针或者引用才能达到这个效果呢?《深度探索C++对象模型》这本书对此有这样一个解释:

一个pointer或一个reference之所以支持多态,是因为它们并不引发内存任何与类型有关的内存委托操作。会受到改变的,只有它们所指向内存的大小和解释方式而已。

这样读起来有点拗口,简单讲就是指针或者引用的赋值并不会改变原对象内存里的内容,他只会改变对内存大小及内容的解释方式。举个简单的例子:我将int变量的地址赋值给了char型指针,char型指针才不管原来的变量是什么,他对外只解释一个字节的内容。

C++面向对象继承与多态

 

同理可知,子类对象的内存内容并没有发生改变,那么对象的vptr还是指向子类的vtable,所以调用的还是子类的的成员函数。而简单的上转型并不会有这样的效果,他会对内存进行重新分配。

另外说一下,只用C++有静态绑定这个概念,其他面向对象类的语言都是动态绑定。可以看出C语言的知识是很细致入微的。

为什么要使用多态

到此其实多态已经讲完了,铺垫了这么多前置知识,其实多态就这么一点点。我主要还是想讲讲为什么要使用多态,只有知道了为什么,才能使我们在设计代码的时候考虑得到如何运用这个知识点。我们用一个游戏的例子来说明为什么。

游戏的描述如下:

  • 游戏有一个英雄角色,角色属性有生命(hp)和攻击力(ack)
  • 英雄可以对怪物进行攻击,同时也会受到怪物的攻击
  • 怪物属性有生命(hp)和攻击力(ack)
  • 怪物可以对英雄进行攻击,也会受到英雄的攻击
  • 现阶段有三种怪物:狼人,僵尸,女巫

我们先来实现怪物类:

class wolf //狼人类 {
public:
    wolf(int hp, int ack)
    : hp(hp)    , ack(ack)    {}    bool damage(int dm) {
        if (this->hp <= 0) return false;
        this->hp -= dm;
        return false;
    }    bool attack(hero &hr) {
        return hr.damage(this->ack);
    }private:
    int hp;
    int ack;
};class zombie {
public:
    zombie(int hp, int ack)
    : hp(hp)    , ack(ack)    {}    bool damage(int dm) {
        if (this->hp <= 0) return false;
        this->hp -= dm;
        return false;
    }    bool attack(hero &hr) {
        return hr.damage(this->ack);
    }private:
    int hp;
    int ack;
};class witch {
public:
    witch(int hp, int ack)
    : hp(hp)    , ack(ack)    {}    bool damage(int dm) {
        if (this->hp <= 0) return false;
        this->hp -= dm;
        return false;
    }    bool attack(hero &hr) {
        return hr.damage(this->ack);
    }private:
    int hp;
    int ack;
};

然后我们来实现英雄类:

class hero {
public:
    hero(int hp, int ack)
    : hp(hp)    , ack(ack)    {}    bool damage(int dm) {
        if (this->hp <= 0) return false;
        this->hp -= dm;
    }    bool attack(wolf &wf) {
        return wf.damage(this->ack);
    }    bool attack(zombie &zb) {
        return zb.damage(this->ack);
    }    bool attack(witch &wt) {
        return wt.damage(this->ack);
    }private:
    int hp;
    int ack;
};

我们发现,同样逻辑的attack()函数,我们需要实现三次。如果后期游戏要增添新的怪物,我们还得继续写attack()函数。 这其实还是一种面向过程的思想,并不是说写几个类出来就是面向对象了。而且这也完全不符合我们程序猿的编程习惯,我们程序猿不喜欢重复的东西。欸,这个时候多态就能发挥他的作用了。

我们来定义怪物们的基类:

class monster {
public:
    virtual bool damage(int dm) = 0;
    virtual bool attack(hero &hr) = 0;
};

之前说了,我们并不关心这个基类的虚函数具体是怎么实现的,那么我们就可以将其声明为纯虚类。然后让怪物都继承这个基类,实现上面这两个函数就可以了。这样我们就可以将hero类改造成这样:

class hero {
public:
    hero(int hp, int ack)
    : hp(hp)    , ack(ack)    {}    bool damage(int dm) {
        if (this->hp <= 0) return false;
        this->hp -= dm;
    }    bool attack(monster &ms) //参数修改为monster类一定要用指针或者引用 {
        return ms.damage(this->ack);
    }
private:
    int hp;
    int ack;
};

这样代码是不是就简洁很多。而且根据多态的性质,不同的怪物会调用其各自的damage()函数。以后要是新增怪物,只要继承和实现虚基类就好了,hero类并不需要进行修改。这就体现了面向对象编程的优势了,这还只是其中之一。

同理,要是有多种英雄,我们同样可以抽象出一个英雄类的虚基类,然后派生出各式各样的英雄,怪物类也不需要重复写多个attack()函数。

有同学还是觉得怪物类的实现还是重复度太高了,这没有体现多态的优势啊。其实不然,前面说到每个子类都应该重写基类的虚函数,是因为不同的子类都应该有他的特别之处, 所以才叫派生嘛。如果子类和子类,或者子类和基类完全一样那就没有必要继承与派生了。

这里重复度高只是因为代码量小,我只是举了个小小的例子,其实在真正的游戏中不同怪物子类的attack()函数和damage()函数的内部细节应该是不一样。比如不同的怪物有不同的攻击特效,有不同的受击效果,有不同的技能冷却时间等等。这些细节都是通过子类去重写基类的虚函数,才得以体现的。

总结

到此为止,我所了解的继承与多态算是总结完毕了。会简单地封装几个类并不是面向对象编程,只有彻底理解了封装、继承与多态,面向对象编程才算是入了个门。只有理解了这些,我们才能开始学习设计模式,才能领悟到设计模式的精髓所在。学设计模式建议大家去看《大话设计模式》这本书,以后有时间我也会在我的博客里总结一些设计模式。



Tags:C++   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
一、编程语言1.根据熟悉的语言,谈谈两种语言的区别?主要浅谈下C/C++和PHP语言的区别:1)PHP弱类型语言,一种脚本语言,对数据的类型不要求过多,较多的应用于Web应用开发,现在好多互...【详细内容】
2021-12-15  Tags: C++  点击:(17)  评论:(0)  加入收藏
函数调用约定(Calling Convention),是一个重要的基础概念,用来规定调用者和被调用者是如何传递参数的,既调用者如何将参数按照什么样的规范传递给被调用者。在参数传递中,有两个很...【详细内容】
2021-11-30  Tags: C++  点击:(19)  评论:(0)  加入收藏
一、为什么需要使用内存池在C/C++中我们通常使用malloc,free或new,delete来动态分配内存。一方面,因为这些函数涉及到了系统调用,所以频繁的调用必然会导致程序性能的损耗;另一...【详细内容】
2021-11-17  Tags: C++  点击:(37)  评论:(0)  加入收藏
C++编程中,你是否有为 我到底该写个struct还是class 而苦恼过?如果你到现在还不知道该如何选择,那么请求继续阅读,下文或许能给你些建议。问题的产生C++语言继承了 C语言的 stru...【详细内容】
2021-10-18  Tags: C++  点击:(61)  评论:(0)  加入收藏
C++在C的面向过程概念的基础上提供了面向对象和模板(泛型编程)的语法功能。下面以一个简单实例(动态数组的简单封装,包括下标的值可以是任意正数值,并提供边界检查)来说明C++是如...【详细内容】
2021-10-18  Tags: C++  点击:(49)  评论:(0)  加入收藏
0 前言Hello,大家好,欢迎来到『自由技艺』的 C++ 系列专题。代码重用,尽可能避免冗余代码是程序员的一项必备技能,今天就来给大家介绍其中一种:函数装饰器。在设计模式中,与它对应...【详细内容】
2021-09-28  Tags: C++  点击:(75)  评论:(0)  加入收藏
今天我们就来聊一聊 C++ 中的异常机制吧。在学校期间,我们很少会用得上异常机制。然而,工作之后,很多时候却不得不引入异常机制。因为一般情况下,使用函数的返回值来确定函数的...【详细内容】
2021-09-26  Tags: C++  点击:(181)  评论:(0)  加入收藏
一、内存泄漏(memory leak)内存泄漏(memory leak)是指由于疏忽或错误造成了程序未能释放掉不再使用的内存的情况。内存泄漏并非指内存在物理上的消失,而是应用程序分配某段内存...【详细内容】
2021-09-03  Tags: C++  点击:(104)  评论:(0)  加入收藏
stack容器#include <iostream>using namespace std;#include <stack>//容器头文件void test(){stack<int>p;p.push(100);p.push(1000);p.push(100);while(!p.empty()){cout<...【详细内容】
2021-08-17  Tags: C++  点击:(81)  评论:(0)  加入收藏
stl 常用遍历算法(for_each transform)示例代码(结论在结尾!!!!)#include<iostream>using namespace std;#include"vector"#include"map"#include"string"#include"list"#in...【详细内容】
2021-08-13  Tags: C++  点击:(89)  评论:(0)  加入收藏
▌简易百科推荐
一、简介很多时候我们都需要用到一些验证的方法,有时候需要用正则表达式校验数据时,往往需要到网上找很久,结果找到的还不是很符合自己想要的。所以我把自己整理的校验帮助类分...【详细内容】
2021-12-27  中年农码工    Tags:C#   点击:(1)  评论:(0)  加入收藏
引言在学习C语言或者其他编程语言的时候,我们编写的一个程序代码,基本都是在屏幕上打印出 hello world ,开始步入编程世(深)界(坑)的。C 语言版本的 hello world 代码:#include <std...【详细内容】
2021-12-21  一起学嵌入式    Tags:C 语言   点击:(10)  评论:(0)  加入收藏
读取SQLite数据库,就是读取一个路径\\192.168.100.**\position\db.sqlite下的文件<startup useLegacyV2RuntimeActivationPolicy="true"> <supportedRuntime version="v4.0"/...【详细内容】
2021-12-16  今朝我的奋斗    Tags:c#   点击:(21)  评论:(0)  加入收藏
什么是shellshell是c语言编写的程序,它在用户和操作系统之间架起了一座桥梁,用户可以通过这个桥梁访问操作系统内核服务。 它既是一种命令语言,同时也是一种程序设计语言,你可以...【详细内容】
2021-12-16  梦回故里归来    Tags:shell脚本   点击:(16)  评论:(0)  加入收藏
一、编程语言1.根据熟悉的语言,谈谈两种语言的区别?主要浅谈下C/C++和PHP语言的区别:1)PHP弱类型语言,一种脚本语言,对数据的类型不要求过多,较多的应用于Web应用开发,现在好多互...【详细内容】
2021-12-15  linux上的码农    Tags:c/c++   点击:(17)  评论:(0)  加入收藏
1.字符串数组+初始化char s1[]="array"; //字符数组char s2[6]="array"; //数组长度=字符串长度+1,因为字符串末尾会自动添&lsquo;\0&lsquo;printf("%s,%c\n",s1,s2[2]);...【详细内容】
2021-12-08  灯-灯灯    Tags:C语言   点击:(46)  评论:(0)  加入收藏
函数调用约定(Calling Convention),是一个重要的基础概念,用来规定调用者和被调用者是如何传递参数的,既调用者如何将参数按照什么样的规范传递给被调用者。在参数传递中,有两个很...【详细内容】
2021-11-30  小智雅汇    Tags:函数   点击:(19)  评论:(0)  加入收藏
一、问题提出问题:把m个苹果放入n个盘子中,允许有的盘子为空,共有多少种方法?注:5,1,1和1 5 1属同一种方法m,n均小于10二、算法分析设f(m,n) 为m个苹果,n个盘子的放法数目,则先对...【详细内容】
2021-11-17  C语言编程    Tags:C语言   点击:(46)  评论:(0)  加入收藏
一、为什么需要使用内存池在C/C++中我们通常使用malloc,free或new,delete来动态分配内存。一方面,因为这些函数涉及到了系统调用,所以频繁的调用必然会导致程序性能的损耗;另一...【详细内容】
2021-11-17  深度Linux    Tags:C++   点击:(37)  评论:(0)  加入收藏
OpenCV(Open Source Computer Vision Library)是一个(开源免费)发行的跨平台计算机视觉库,可以运行在Linux、Windows、Android、ios等操作系统上,它轻量级而且高效---由一系列...【详细内容】
2021-11-11  zls315    Tags:C#   点击:(50)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条