多态性

多态性是面向对象程序设计的关键技术之一。若程序设计语言不支持多态性,不能称为面向对象的语言。利用多态性技术,可以调用同一个函数名的函数,实现完全不同的功能。

多态性(polymorphism)多态性是考虑在不同层次的类中,以及在同一类中,同名的成员函数之间的关系问题。函数的重载,运算符的重载,属于编译时的多态性。以虚基类为基础的运行时的多态性是面向对象程序设计的标志性特征。体现了类推和比喻的思想方法。

虚函数

虚函数是一个类的成员函数,定义格式如下:
virtual 返回类型  函数名(参数表)

关键字virtual指明该成员函数为虚函数。virtual仅用于类定义中,如虚函数在类外定义,不可加virtual(但是类内部声明时要加virtual)。

当某一个类的一个类成员函数被定义为虚函数,则由该类派生出来的所有派生类中,该函数始终保持虚函数的特征。

最后一句话如何理解?

#include<iostream>
using namespace std; class Base
{
public:
virtual void show()
{
cout << "This is Base show" << endl;
}
}; class D :public Base
{
public:
void show()
{
cout << "This is D show" << endl;
}
}; class C :public D
{
public:
void show()
{
cout << "This is C show" << endl;
}
}; void main()
{
C c;
D *pd = &c;
pd->show();
}

倘若D的show不是虚函数,那么pd->show();访问的就是类D的show方法。但结果却是C的show方法,所以类D的show方法也是虚的。

在整个虚函数继承体系中,父类 子类的虚函数需要做到3同。函数名相同,函数返回值相同,函数参数相同(包括参数个数,参数类型)

当在派生类中重新定义虚函数(overriding a virtual function,亦译作覆盖)时,不必加关键字virtual。但重新定义时不仅要同名,而且它的参数表和返回类型全部与基类中的虚函数一样,否则链接时出错。

多态演示代码

#include<iostream>
using namespace std; class Animal
{
public:
void Eat()
{
cout << "Animal Eat" << endl;
}
void Sleep()
{
cout << "Animal Sleep" << endl;
}
}; class People :public Animal
{
public:
void Eat()
{
cout << "People Eat" << endl;
}
void Sleep()
{
cout << "People Sleep" << endl;
}
}; class Dog :public Animal
{
public:
void Eat()
{
cout << "Dog Eat" << endl;
}
void Sleep()
{
cout << "Dog Sleep" << endl;
}
}; class Pig :public Animal
{
public:
void Eat()
{
cout << "Pig Eat" << endl;
}
void Sleep()
{
cout << "Pig Sleep" << endl;
}
}; void fun(Animal *a)
{
a->Eat();
a->Sleep();
}
void main()
{
People p;
Dog dog;
Pig pig;
fun(&p);
fun(&dog);
fun(&pig);
}

由于赋值兼容规则,导致都是Animal的Eat和Sleep

#include<iostream>
using namespace std; class Animal
{
public:
virtual void Eat()
{
cout << "Animal Eat" << endl;
}
virtual void Sleep()
{
cout << "Animal Sleep" << endl;
}
}; class People :public Animal
{
public:
void Eat()
{
cout << "People Eat" << endl;
}
void Sleep()
{
cout << "People Sleep" << endl;
}
}; class Dog :public Animal
{
public:
void Eat()
{
cout << "Dog Eat" << endl;
}
void Sleep()
{
cout << "Dog Sleep" << endl;
}
}; class Pig :public Animal
{
public:
void Eat()
{
cout << "Pig Eat" << endl;
}
void Sleep()
{
cout << "Pig Sleep" << endl;
}
}; void fun(Animal *a)
{
a->Eat();
a->Sleep();
}
void main()
{
People p;
Dog dog;
Pig pig;
fun(&p);
fun(&dog);
fun(&pig);
}

虚函数注意事项

成员函数应尽可能地设置为虚函数,但必须注意以下几条:

1.派生类中定义虚函数必须与基类中的虚函数同名外,还必须同参数表,同返回类型。否则被认为是同名隐藏,而不是虚函数。如基类中返回基类指针,派生类中返回派生类指针是允许的,这是一个例外。

#include<iostream>
using namespace std; class Base
{
public:
virtual Base* Show()
{
cout << "This is Base Show()" << endl;
return this;
}
}; class Son :public Base
{
public:
virtual Son* Show()
{
cout << "This is Son Show()" << endl;
return this;
}
}; void main()
{
Son son;
Base *pb = &son;
pb->Show();
}

注:上述代码需要在VC6.0以上版本允许,VC6.0对C++语法支持不完善

2.只有类的成员函数才能说明为虚函数。这是因为虚函数仅适用于有继承关系的类对象。

3.静态成员函数,是所有同一类对象共有,不受限于某个对象,不能作为虚函数。

4.一个类对象的静态和动态类型是相对的,实现动态多态性时,必须使用基类类型的指针变量或引用,使该指针指向该基类的不同派生类的对象,并通过该指针指向虚函数,才能实现动态的多态性。

5.内联函数每个对象一个拷贝,无映射关系,不能作为虚函数。还有一个,全局函数也不能是虚函数

6.析构函数可定义为虚函数,构造函数不能定义虚函数,因为在调用构造函数时对象还没有完成实例化。在基类中及其派生类中都动态分配的内存空间时,必须把析构函数定义为虚函数,实现撤消对象时的多态性。

#include<iostream>
using namespace std; class Base
{
public:
virtual Base* Show()
{
cout << "This is Base Show()" << endl;
return this;
}
Base()
{
cout << "Create Base" << endl;
}
~Base()
{
cout << "Free Base" << endl;
}
}; class Son :public Base
{
public:
virtual Son* Show()
{
cout << "This is Son Show()" << endl;
return this;
}
Son()
{
cout << "Create Son" << endl;
}
~Son()
{
cout << "Free Son" << endl;
}
}; void main()
{
Base *pb = new Son;
delete pb;
}

构造的时候父类、子类都构造了。析构的时候只析构了父类,导致内存泄漏。改进代码,父类析构函数变成虚函数,代码如下

#include<iostream>
using namespace std; class Base
{
public:
virtual Base* Show()
{
cout << "This is Base Show()" << endl;
return this;
}
Base()
{
cout << "Create Base" << endl;
}
virtual ~Base()
{
cout << "Free Base" << endl;
}
}; class Son :public Base
{
public:
virtual Son* Show()
{
cout << "This is Son Show()" << endl;
return this;
}
Son()
{
cout << "Create Son" << endl;
}
~Son()
{
cout << "Free Son" << endl;
}
}; void main()
{
Base *pb = new Son;
delete pb;
}

父类指针在指向子类对象的时候,之所以能够指向是因为在子类里面隐藏包含一个父类。pb指向的是整个空间的起始位置,而这个起始位置也恰恰是父类的起始位置。由于是Base类型,在delete的时候,编译器会认为你要删除的只是Base那么大的内存,而不会释放子类对象内存,导致内存泄漏。一旦把父类的析构函数声明为析构函数之后,这种释放就会达到一个多态的释放。即先检查一下我是否是因为创建子类而创建的父类,如果是则先释放子类对象,然后释放父类对象,达到级联式删除。

7.虚函数执行速度要稍慢一些。为了实现多态性,每一个派生类中均要保存相应虚函数的入口地址表,函数的调用机制也是间接实现。所以多态性总是要付出一定代价,但通用性是一个更高的目标。

8.如果定义放在类外,virtual只能加在函数声明前面,不能(再)加在函数定义前面。正确的定义必须不包括virtual。

重载、隐藏、覆盖

重载是针对一个类内部情况而言的,作为重载的依据:参数列表不同(包括参数的个数不同、类型不同或顺序不同,仅仅参数名称不同是不可以的);函数返回值也不能作为重载的依据。

隐藏是针对继承体系而言,父类子类有相同名字的成员函数,这里只要求一同即函数名字。并且不能使用virtual关键字,子类方法会隐藏父类所有同名方法

覆盖也是针对继承体系而言,但是需要加关键字virtual

C++——多态性 与 虚函数的更多相关文章

  1. sdut 6-2 多态性与虚函数

    6-2 多态性与虚函数 nid=24#time" title="C.C++.go.haskell.lua.pascal Time Limit1000ms Memory Limit ...

  2. C++学习之路—多态性与虚函数(二)纯虚函数与抽象类

    (根据<C++程序设计>(谭浩强)整理,整理者:华科小涛,@http://www.cnblogs.com/hust-ghtao转载请注明) 1    纯虚函数 在前面的博客中已经提到:有时 ...

  3. C++学习之路—多态性与虚函数(一)利用虚函数实现动态多态性

    (根据<C++程序设计>(谭浩强)整理,整理者:华科小涛,@http://www.cnblogs.com/hust-ghtao转载请注明) 多态性是面向对象程序设计的一个重要特征.顾名思义 ...

  4. C++之多态性与虚函数

    面向对象程序设计中的多态性是指向不同的对象发送同一个消息,不同对象对应同一消息产生不同行为.在程序中消息就是调用函数,不同的行为就是指不同的实现方法,即执行不同的函数体.也可以这样说就是实现了&quo ...

  5. C++学习7-面向对象编程基础(多态性与虚函数、 IO文件流操作)

    多态 多态性是指对不同类的对象发出相同的消息将返回不同的行为,消息主要是指类的成员函数的调用,不同的行为是指不同的实现: 函数重载 函数重载是多态性的一种简单形式,它是指允许在相同的作用域内,相同的函 ...

  6. C++ 多态性和虚函数

    2017-06-27 19:17:52 C++面向对象编程的一个重要的特性就是多态性,而多态性的实现需要依赖虚函数的帮助. 一.多态的作用: 隐藏实现细节,使得代码能够模块化: 接口重用,实现“一个接 ...

  7. [转]C++之多态性与虚函数

    面向对象程序设计中的多态性是指向不同的对象发送同一个消息,不同对象对应同一消息产生不同行为.在程序中消息就是调用函数,不同的行为就是指不同的实现方法,即执行不同的函数体.也可以这样说就是实现了“一个接 ...

  8. 2013级C++第14周(春)项目——多态性、虚函数和抽象类

    课程首页在:http://blog.csdn.net/sxhelijian/article/details/11890759,内有完整教学方案及资源链接 第一部分 阅读程序1.阅读.改动和执行关于交通 ...

  9. c++特别要点:多态性与虚函数

    本来是准备在工厂模式中顺便整理.但粗略浏览了,内容还是很多,需要单独开一篇. 一.什么是多态性? 多态性可以概括为“一个接口,多种方法”. 多态与非多态的区别在于“成员函数调用地址的早绑定和晚绑定”. ...

  10. C++多态性:虚函数的调用原理

    多态性给我们带来了好处:多态使得我们可以通过基类的引用或指针来指明一个对象(包含其派生类的对象),当调用函数时可以自动判断调用的是哪个对象的函数. 一个函数说明为虚函数,表明在继承的类中重载这个函数时 ...

随机推荐

  1. TOmCAT HTTPS 单向验证 忽略证书

    https://www.cnblogs.com/haha12/p/4381663.html

  2. JS验证正数字,正则的一种正数规则1

    JS中有一个验证数字的方法,就是!isNAN.NAN是非数字,!在JS里表示不是的意思,所以这个!isNAN就是判断不是非数字,也就是是数字.验证某个字符串是否是数字格式是:!isNaN(字符串)经过 ...

  3. 【Leetcode_easy】1046. Last Stone Weight

    problem 1046. Last Stone Weight 参考 1. Leetcode_easy_1046. Last Stone Weight; 完

  4. Linux配置Docker镜像加速器

    Docker默认镜像为官方镜像,可以配置成国内加速器提高速度 登录阿里云控制台,搜索容器镜像服务获取到镜像加速服务地址 新建配置文件 /etc/docker/daemon.json 输入以下内容 { ...

  5. Python文件的读取写入操作

    一.打开文件.关闭文件操作 想要读取文件或是写入文件,第一步便是打开文件,最后一步便是关闭文件.这里介绍两种打开(关闭)文件的方式: 1.open()方法 f=open(file_name[,acce ...

  6. RabbitMQ官方教程四 Routing(GOLANG语言实现)

    在上一教程中,我们构建了一个简单的日志记录系统. 我们能够向许多消费者广播日志消息. 在本教程中,我们将向其中添加功能-我们将使仅订阅消息的子集成为可能. 例如,我们将只能将严重错误消息定向到日志文件 ...

  7. 使用 LVS 实现负载均衡原理及安装配置详解(课堂随笔)

    一.负载均衡LVS基本介绍 LB集群的架构和原理很简单,就是当用户的请求过来时,会直接分发到Director Server上,然后它把用户的请求根据设置好的调度算法,智能均衡地分发到后端真正服务器(r ...

  8. eNSP配置基本与高级访问控制列表

    首先我们进行基本的acl控制 拓扑图如下所示 首先我们对路由器进行基本ip配置 并在路由器上设置ospf协议 添加相邻的网段 在路由器上运行了ospf协议后 使用display ip route-ta ...

  9. Java中常用的设计模式代码与理解

    Java中常用的设计模式代码与理解 一.单例模式 1.饿汉式 (太饿了,类加载的时候就创建实例) /** * 饿汉式单例模式 */ public class HungrySingleInstance ...

  10. centos7 为使用su命令的用户添加pam授权认证

    # 查看用户所属哪个组 groups 用户名 #查看当前有哪些用户运行程序 ps -aux|awk '{print $1}'|sort -rn|uniq -c|sort -rn # 清理不再使用的用户 ...