C/C++编程笔记:C++入门知识丨多态性和虚函数
本篇要学习的内容和知识结构概览

多态性
编译时的多态性称为静态联编. 当调用重载函数时, 在编译期就确定下来调用哪个函数.
运行时的多态性称为动态联编. 在运行时才能确定调用哪个函数, 由虚函数来支持.
静态联编中的赋值兼容性及名字支配规律
派生一个类的原因并非总是为了添加新的成员或成员函数, 有时是为了重新定义基类的成员函数。
#define PI 3.14159
class Point {
double x;
double y;
public:
Point(double a, double b) {
x = a;
y = b;
}
double area() {
return ;
}
};
class Circle: public Point {
double radius;
public:
Circle(double a, double b, double r):Point(a, b) {
radius = r;
}
double area() {
return PI * radius * radius;
}
};
int main() {
Point a(1.5, 6.7);
Circle c(1.5, 6.7, 2.5);
cout << a.area() << endl; // 调用对象a的成员函数area()
cout << c.area() << endl; // 调用对象c的成员函数area()
Point * p = &c; // 对象c的地址为指向Point类型指针赋值
cout << p -> area() << endl; // 调用Point类的成员函数area()
Point & rc = c; // 对象c初始化Point类型的引用
cout << rc.area() << endl; // 调用Point类的成员函数area()
}
在派生类有同名函数的情况下
Point * pPoint; // 声明的基类指针只能指向基类
Circle * pCircle // 声明的派生类指针只能指向派生类
如果派生类没有基类的同名函数, 派生类的指针才根据继承原则调用基类的函数
虚函数
一旦定义了虚函数, 该基类的派生类中的同名函数也自动成为虚函数.
虚函数的定义
用关键字virtual来声明一个虚函数, 虚函数只能是类中的一个成员函数, 不能是静态成员.
像这样:
class Point {
double x;
double y;
public:
Point(double a, double b) {
x = a;
y = b;
}
// 用virtual关键字来定义一个虚函数
virtual double area() {
return ;
}
};
class Circle: public Point {
double radius;
public:
Circle(double a, double b, double r):Point(a, b) {
radius = r;
}
// 基类中area()函数为虚函数, 派生类中的同名函数也自动成为虚函数
double area() {
return PI * radius * radius;
}
};
虚函数实现多态性的条件
关键字virtual告诉编译器调用虚函数进行动态联编.
使用虚函数不一定产生多态性, 也不一定使用动态联编.
在调用中对虚函数使用成员名限定, 可以强制编译器对该函数使用静态联编.
产生运行多态性, 也就是动态联编有3个前提
(1)类之间的继承关系满足赋值兼容性规则
(2)改写了同名虚函数
(3)根据赋值兼容性规则使用指针(或引用)
像这样:
class Point {
double x;
double y;
public:
Point(double a, double b) {
x = a;
y = b;
}
// 用virtual关键字来定义一个虚函数
virtual double area() {
return ;
}
};
class Circle: public Point {
double radius;
public:
Circle(double a, double b, double r):Point(a, b) {
radius = r;
}
// 基类中area()函数为虚函数, 派生类中的同名函数也自动成为虚函数
double area() {
return PI * radius * radius;
}
};
void display(Point * p) {
cout << p -> area() << endl;
}
void display(Point & a) {
cout << a.area() << endl;
}
int main() {
Point a(1.5, 6.7);
Circle c(1.5, 6.7, 2.5);
Point * p = &c; // 对象c的地址为指向Point类型指针赋值
Point & rc = c; // 对象c初始化Point类型的引用
display(a); // 调用对象a的成员函数area()
display(p); // 根据运行时的多态性, p指向的c对象, 所以实际调用c对象的成员函数area()
display(rc); // 根据运行时的多态性, rc是对c对象的引用, 所以实际调用c对象的成员函数area()
/** 输出结果
0
19.6349
19.6349
*/
}
纯虚函数与抽象类
在基类中不给虚函数一个有意义的定义, 可以说明为纯虚函数, 将定义留给派生类去做
像这样:
class 类名 {
public:
virtual 函数类型 函数名(参数列表) = ;
};
抽象类:包含有纯虚函数的类称为抽象类. 一个抽象类至少有一个纯虚函数, 一个抽象类只能作为基类来派生新类, 不能说明抽象类的对象.
class Point {
double x;
double y;
public:
Point(double a, double b) {
x = a;
y = b;
}
// 用virtual关键字来定义一个虚函数
virtual double area() = ;
};
int main() {
// Point a(1.5, 6.7); // Point为抽象类, 不能实例化一个对象 error Variable type 'Point' is an abstract class
Point * p; // Point为抽象类, 可以声明一个指向抽象类对象的指针
}
注意
空虚函数定义 virtual void area() {}
纯虚函数定义 virtual void area() = 0;
纯虚函数的派生类仍是抽象类. 如果派生类中给出了基类所有纯虚函数的实现, 则该派生类不再是抽象类
类族
如果通过同一个基类派生一系列的类, 则将这些类总称为类族.
像这样:
#define PI 3.14159 // 抽象类 有一个纯虚函数 area()
class Shape {
public:
virtual double area() = ;
}; // 正方形 有一个连长数据成员
class Square: public Shape {
protected:
double h; public:
Square(double a) {
h = a;
}
double area() {
return h * h;
}
}; // 圆
class Circle: public Square {
public:
Circle(double a):Square(a) { }
double area() {
return h * h * PI;
}
}; // 三角形
class Triangle: public Square {
protected:
double w; public:
Triangle(double a, double b):Square(a) {
w = b;
}
double area() {
return w * h * 0.5;
}
}; // 矩形
class Rect: public Triangle {
public:
Rect(double a, double b):Triangle(a, b) { }
double area() {
return w * h;
}
}; int main() { Shape * s[];
s[] = new Square();
s[] = new Circle();
s[] = new Rect(, );
s[] = new Triangle(, );
s[] = new Square();
for (int i = ; i < ; i++) {
// 因为虚函数支持动态联编, 所以在运行时才确定每个元素的类型, 调用各自的成员函数
cout << "s[" << i << "] = " << s[i] -> area() << endl;
}
}
多重继承与虚函数
多重继承可以被视为多个单一继承的组合。
// 基类
class A {
public:
virtual void func() {
cout << "call A" << endl;
}
}; // 基类
class B {
public:
virtual void func() {
cout << "call B" << endl;
}
}; // 多重继承
class C: public A, public B {
public:
void func() {
cout << "call C" << endl;
}
}; int main() {
A * pa;
B * pb;
C * pc, c; pa = &c;
pb = &c;
pc = &c;
pa -> func(); // 动态联编, pa指向派生类对象c, 调用对象c的成员函数C::func();
pb -> func(); // 动态联编, pb指向派生类对象c, 调用对象c的成员函数C::func();
pc -> func(); // pc既是指向C类对象的指针, 又是指向的C类对象, 调用对象c的成员函数C::func();
}
总结
C++有两种多态性, 一种是编译时多态性, 也叫静态联编; 另一种是运行时多态性, 也叫动态联编. 这大大提高了我们解决问题的丰富性. 可能也是C++长久不衰的魅力所在吧! 我会继续深入学习C++, 继续挖掘语言的本质!
自学C/C++编程难度很大,不妨和一些志同道合的小伙伴一起学习成长!
C语言C++编程学习交流圈子,【点击进入】微信公众号:C语言编程学习基地
有一些源码和资料分享,欢迎转行也学习编程的伙伴,和大家一起交流成长会比自己琢磨更快哦!

C/C++编程笔记:C++入门知识丨多态性和虚函数的更多相关文章
- C/C++编程笔记:C++入门知识丨认识C++的函数和对象
一. 本篇要学习的内容和知识结构概览 二. 知识点逐条分析 1. 混合型语言 C++源文件的文件扩展名为.cpp, 也就是c plus plus的简写, 在该文件里有且只能有一个名为main的主函数, ...
- C/C++编程笔记:C++入门知识丨从结构到类的演变
先来看看本节知识的结构图吧! 接下来我们就逐步来看一下所有的知识点: 结构的演化 C++中的类是从结构演变而来的, 所以我们可以称C++为”带类的C”. 结构发生质的演变 C++结构中可以定义函数, ...
- C/C++编程笔记:C++入门知识丨运算符重载
本篇要学习的内容和知识结构概览 运算符重载使用场景 常规赋值操作 我们现在有一个类 想要实现这种赋值操作 具体实现如下: 所以说呢,我们在使用运算符进行运算的时候, 实际上也是通过函数来实现运算的. ...
- C/C++编程笔记:C++入门知识丨继承和派生
本篇要学习的内容和知识结构概览 继承和派生的概念 派生 通过特殊化已有的类来建立新类的过程, 叫做”类的派生”, 原有的类叫做”基类”, 新建立的类叫做”派生类”. 从类的成员角度看, 派生类自动地将 ...
- C/C++编程笔记:C++入门知识丨类和对象
本篇要学习的内容和知识结构概览 类及其实例化 类的定义 将一组对象的共同特征抽象出来, 从而形成类的概念. 类包括数据成员和成员函数, 不能在类的声明中对数据成员进行初始化 声明类 形式为: clas ...
- C/C++编程笔记:C++入门知识丨函数和函数模板
本篇要学习的内容和知识结构概览 函数的参数及其传递方式 1. 函数参数传递方式 传值: 传变量值: 将实参内存中的内容拷贝一份给形参, 两者是不同的两块内存 传地址值: 将实参所对应的内存空间的地址值 ...
- C/C++编程笔记:C++入门知识丨认识C++面向过程编程的特点
一. 本篇要学习的内容和知识结构概览 二. 知识点逐条分析 1. 使用函数重载 C++允许为同一个函数定义几个版本, 从而使一个函数名具有多种功能, 这称之为函数重载. 像这样: 虽然函数名一样, 但 ...
- C/C++编程笔记:C语言基础printf()和scanf()函数,大学入门知识
在解释这两个函数之前,为了方便大家更容易理解,我们先来讲讲下面的这幅图. 这个图中的例子其实很简单,我们就是把题目通过输入设备(例如键盘鼠标)传输到计算机中,然后让计算机进行运算得出结果,再然后把 ...
- UnityShader学习笔记1 — — 入门知识整理
注:资料整理自<Unity Shader入门精要>一书 一.渲染流程概念阶段: 应用阶段:(1)准备好场景数据:(如摄像机位置,物体以及光源等) (2)粗粒度剔除(Culling): ...
随机推荐
- 网页开发中利用CSS以图换字的多中实现方法总汇
在h1标签中,新增span标签来保存标题内容,然后将其样式设置为display:none <style> h1 { width: 64px; height: 64px; backgroun ...
- 03.springboot 整合RabbitMQ
1.引入依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId> ...
- Scala 基础(十一):Scala 函数式编程(三)高级(一)偏函数、作为参数的函数、匿名函数、高阶函数
1 偏函数 1)在对符合某个条件,而不是所有情况进行逻辑操作时,使用偏函数是一个不错的选择 2)将包在大括号内的一组case语句封装为函数,我们称之为偏函数,它只对会作用于指定类型的参数或指定范围值的 ...
- java 面向对象(十四):面向对象的特征二:继承性 (三) 关键字:super以及子类对象实例化全过程
关键字:super 1.super 关键字可以理解为:父类的2.可以用来调用的结构:属性.方法.构造器3.super调用属性.方法:3.1 我们可以在子类的方法或构造器中.通过使用"supe ...
- java 基本语法(九) 数组(二) 一维数组
1.一维数组的声明与初始化 正确的方式: int num;//声明 num = 10;//初始化 int id = 1001;//声明 + 初始化 int[] ids;//声明 //1.1 静态初始化 ...
- mysql中DDL库和表的管理
#DDL /* 数据定义语言 库和表的管理 一.库的管理 创建.修改.删除 二.表的管理 创建.修改.删除 创建:create 修改:alter 删除:drop */ #一.库的管理 #1.库的创建 ...
- js 分享QQ、QQ空间、微信、微博
//分享QQ好友 function qq(title,url,pic) { var p = { url: 'http://test.qicheyitiao.com',/*获取URL,可加上来自分享到Q ...
- 2020年Dubbo30道高频面试题!还在为面试烦恼赶快来看看!
前言 Dubbo是一个分布式服务框架,致力于提供高性能和透明化的RPC远程服务调用方案,以及SOA服务治理方案.简单的说,dubbo就是个服务框架,如果没有分布式的需求,其实是不需要用的,只有在分布式 ...
- IDEA 2020.1.2 idea 2020.1.3下载 安装 一键破解
IDEA 2020.1.2 idea 2020.1.3下载 安装 破解 本项目只做个人学习研究之用,不得用于商业用途!若资金允许,请点击链接购买正版,谢谢合作!学生凭学生证可免费申请正版授权!创业公司 ...
- [jvm] -- 类加载过程篇
类加载过程 系统加载 Class 类型的文件主要三步 加载 通过全类名获取定义此类的二进制字节流 将字节流所代表的静态存储结构转换为方法区的运行时数据结构 在内存中生成一个代表该类的 Class对象, ...