多态典型用例之virtual
多态典型用例之virtual
参考:https://www.cnblogs.com/dormant/p/5223215.html
1.虚函数(virtual)
(1)在某基类中声明为 virtual 并在一个或多个派生类中被重新定义的成员函数。实现多态性,通过指向派生类的基类指针或引用,访问派生类中同名覆盖成员函数。
实例:
#include "pch.h"
#include <iostream>
using namespace std;
class A
{
public:
void print()
{
cout<<"class A"<<endl;
}
};
class B:public A
{
public:
void print()
{
cout<<"class B"<<endl;
}
};
int main()
{
A a;
B b;
a.print();
b.print();
return 0;
}
输出结果:

(2)通过class A和class B的print()这个接口,可看出两个class采用了不同的策略,但这并不是多态性行为(使用的是不同类型的指针),没有用到虚函数的功能。改main函数为如下:
int main()
{
A a;
B b;
A *P1=&a;
A *p2=&b;
p1->print();
p2->print();
return 0;
}
输出结果:

(3)p2明明指向的是class B的对象但却是调用的class A的print()函数,这不是我们所期望的结果,那么解决这个问题就需要用到虚函数。
class A
{
public:
virtual void print()
{
cout<<"class A"<<endl;
}
};
class B:public A
{
public:
void print()
{
cout<<"class B"<<endl;
}
};
输出结果:

现在,class A的成员函数print()已经成了虚函数 class B的print()也成了虚函数了。我们只需在把基类的成员函数设为virtual,其派生类的相应的函数也会自动变为虚函数。(对于在派生类的相应函数前是否需要用virtual关键字修饰,语法上可加可不加,不加的话编译器会自动加上,但为了阅读方便和规范性,建议加上)
总结:指向基类的指针在操作它的多态类对象时,会根据不同的类对象,调用其相应的函数,这个函数就是虚函数。
2.虚析构函数
当一个类有子类时,该类的析构函数必须是虚函数,原因:会有资源释放不完全的情况。
#include "pch.h"
#include <iostream>
using namespace std;
class Base
{
public:
~Base()
{
cout << "~Base()" << endl;
}
};
class Derived :public Base
{
public:
Derived()
{
p = new int(0);
}
~Derived()
{
cout << "~Derived()" << endl;
delete p;
}
private:
int *p;
};
void fun(Base *b)
{
delete b;
}
int main()
{
Base *b = new Derived();
fun(b);
return 0;
}
输出结果:

这里可以看到,对象销毁时只调用了父类的析构函数。如果这时子类的析构函数中有关于内存释放的操作,将会造成内存泄露。所以需要给父类的析构函数加上virtual。
class Base
{
public:
virtual ~Base()
{
cout << "~Base()" << endl;
}
};
输出结果:

3.纯虚函数(抽象函数)
纯虚函数是一种特殊的虚函数,在许多情况下,在基类中不能对虚函数给出有意义的实现,而把它声明为纯虚函数,它的实现留给该基类的派生类去做。这就是纯虚函数的作用。
纯虚函数就是没有函数体,同时在定义的时候,其函数名后面要加上“= 0”。
class base1
{
public:
virtual void display() const=0;
};
int main()
{
base1 b1;//错误
base2 b2;
derived d;
fun(&b1);
fun(&b2);
fun(&d);
return 0;
}
图:

base1是一个抽象类,不能实例化。
例1:
#include "pch.h"
#include <iostream>
using namespace std;
class base1
{
public:
virtual void display() const=0;
};
class base2 :public base1
{
public:
virtual void display() const;
};
void base2::display() const
{
cout << "base2::display()" << endl;
}
class derived :public base2
{
public:
virtual void display() const;
private:
};
void derived::display() const
{
cout << "derived::display()" << endl;
}
void fun(base1 *ptr)
{
ptr->display();
}
int main()
{
base2 base2;
derived derived;
fun(&base2);
fun(&derived);
return 0;
}
输出结果:

例2:
#include "pch.h"
#include<iostream>
using namespace std;
class Fish
{
public:
virtual void water() = 0;
virtual void eat() = 0;
};
class Shark : public Fish
{
public:
void water();
void eat();
};
void Shark::eat(){cout<<"Shark eat. "<<endl;}
void Shark::water(){cout<<"Shark water. "<<endl;}
void fun(Fish *f)
{
f->eat();
f->water();
}
void main()
{
Shark s;
Fish *f = &s;
fun(f);
}
输出结果:

a.定义纯虚函数时,不能定义纯虚函数的实现部分。即使是函数体为空也不可以,函数体为空就可以执行,只是什么也不做就返回。而纯虚函数不能调用。
(其实可以写纯虚函数的实现部分,编译器也可以通过,但是永远也无法调用。因为其为抽象类,不能产生自己的对象,而且子类中一定会重写纯虚函数,因此该类的虚表内函数一定会被替换掉,所以可以说永远也调用不到纯虚函数本身)
b."=0"表明程序将不定义该函数,函数声明是为派生类保留一个位置。“=0”的本质是将指向函数体的指针定为NULL。
c.在派生类中必须有重新定义的纯虚函数的函数体,这样的派生类才能用来定义对象。(如果不重写进行覆盖,程序会报错)
多态典型用例之virtual的更多相关文章
- Faas 典型场景——应用负载有显著的波峰波谷,典型用例-基于事件的数据处理
Serverless适用的两大场景 场景一:应用负载有显著的波峰波谷 Serverless化与否的评判标准并不是公司规模的大小,而是其业务背后的具体技术问题,比如业务波峰波谷明显,如何实现削峰填谷.一 ...
- 07——为多态基类声明为virtual析构函数
当基类确定被继承的时候,析构函数声明为virtual是必须的 当返回的派生类的指针或引用的时候,调用析构函数容易发生内存泄漏 当基类作为抽象类使用,声明pure virtual析构函数 析构函数的顺序 ...
- go语言多态接口样例
感觉比java玄幻啊~~~ package main import ( "fmt" ) type notifier interface{ notify() } type user ...
- log4j-slf4j 典型用例
一.maven 配置 <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-log4j ...
- C++:C++的两种多态形式
// // main.cpp // Test.cpp // // Created by mac on 15/8/11. // Copyright (c) 2015年. All rights reser ...
- 转:C++中多态是怎样实现的?
多态是一种不同的对象以单独的方式作用于相同消息的能力,这个概念是从自然语言中引进的.例如,动词“关闭”应用到不同的事务上其意思是不同的.关门,关闭银行账号或关闭一个程序的窗口都是不同的行为:其实际的意 ...
- 4.1 C++多态的概念及前提条件
参考:http://www.weixueyuan.net/view/6370.html 总结: 而多态的功能则是将函数名动态绑定到函数入口地址,这样的动态绑定过程称为运行期绑定. 而在运行期绑定的函数 ...
- C++多态,虚函数,虚函数表,纯虚函数
1.多态性 指相同对象收到不同消息或不同对象收到相同消息时产生不同的实现动作. C++支持两种多态性:编译时多态性,运行时多态性. a.编译时多态性:通过重载函数实现 ,模板(2次编译) ...
- Effective C++ Item 41 了解隐式接口和编译期多态
本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie 经验:class 和 templates 都支持接口和多态. 对 classes 而言接口是 ...
随机推荐
- SpringMVC拦截器与SpringBoot自定义拦截器
首先我们先回顾一下传统拦截器的写法: 第一步创建一个类实现HandlerInterceptor接口,重写接口的方法. 第二步在XML中进行如下配置,就可以实现自定义拦截器了 SpringBoot实现自 ...
- 作业:用pygame实现俄罗斯方块
用Pygame实现俄罗斯方块 参考资料后我安装了Pygame,参考了网上的代码实现了俄罗斯方块小游戏.我试着理解网上的代码的原理和含义,对这些代码的原理有了一个粗略地理解,代码通过参数,RGB值等来实 ...
- 【layui】日期选择一闪而过问题
添加 trigger: 'click',
- mybatis:updatebyexample与updateByExampleSelective
MyBatis,通常逆向工程工具生成接口和xml映射文件用于简单的单表操作. 有两个方法: updateByExample 和 updateByExampleSelective ,作用是对数据库进行 ...
- 【学习笔记】Docker基础
基本概念 Docker是什么? Docker是一种基于Golang开发的虚拟化技术,开发人员和系统管理员使用容器开发,部署和运行应用程序的平台. 使用Linux容器部署应用程序称为容器化. 容器不是新 ...
- Mysql系列(四) —— MySQL的Charset和Collation
本文转载自:再见乱码:5分钟读懂MySQL字符集设置 一.内容概述 在MySQL的使用过程中,了解字符集.字符序的概念,以及不同设置对数据存储.比较的影响非常重要.不少同学在日常工作中遇到的" ...
- SQL系列(十)—— 联结(join)
在数据库设计中,基本上都是根实体ER关系和范式设计多张表.这样更能设计出更合理.紧凑的表结构,提高有效数据存储,数据的管理维护更简便,也提高了库表的伸缩性. 但是事物总是平衡的,在很多场景中单张表的数 ...
- Prometheus监控学习笔记之容器监控Grafana模块
0x00 概述 Grafana 是一个开源的,可以用于大规模指标数据的可视化项目,甚至还能对指标进行报警.基于友好的 Apache License 2.0 开源协议,目前是prometheus监控展示 ...
- ELK学习笔记之Kibana权限控制和集群监控
详细请参考如下四篇博客,注意ELK6中移除了Xpack的默认账户和密码,需要手动设置 Kibana安全特性之权限控制 ELK 集群 Kibana 使用 X-Pack 权限控制,监控集群状态,警报,监视 ...
- ASP.NET Core MVC的Razor视图中,使用Html.Raw方法输出原生的html
我们在ASP.NET Core MVC项目中,有一个Razor视图文件Index.cshtml,如下: @{ Layout = null; } <!DOCTYPE html> <ht ...