C++多态学习笔记

news2024/11/29 9:44:00

C++多态学习笔记

    • 一、多态概述
    • 二、多态的作用
    • 三、多态发生的三个条件
    • 四、多态实现的原理
    • 五、接口的定义
    • 六、模板方法模式
    • 七、虚析构函数和纯虚析构函数
      • 7.1 虚析构函数
      • 7.2 纯虚析构函数
    • 八、重写重载重定义
    • 九、父类引用子类对象

一、多态概述

 同一个操作作用于不同的对象,可以有不同的解释,会产生不同的效果,这就是多态

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class People
{
public:
	// 虚函数
	virtual void Mypro()
	{
		cout << "父类的虚函数" << endl;  // 在编译阶段就确定了调用speak这个函数 
	}
};

class xishi :public People
{
public:
	// 重写父类虚函数
	virtual void Mypro()
	{
		cout << "西施的虚函数" << endl;
	}
};

class wangzhaojun :public People
{
public:
	// 重写父类虚函数
	virtual void Mypro()
	{
		cout << "王昭君的虚函数" << endl;
	}
};

class diaochan :public People
{
public:
	// 重写父类虚函数
	virtual void Mypro()
	{
		cout << "貂蝉的虚函数" << endl;
	}
};

// 同一个操作
void doLogin(People* pro)
{
	pro->Mypro();
}

void test()
{
	// 创建不同的对象 会产生不同的效果


	People* pro = NULL;
	pro = new xishi;
	doLogin(pro);// 继承 向上转换   指针范围缩小
	delete pro;


	pro = new wangzhaojun;
	doLogin(pro);// 继承向上转换 
	delete pro;
}


int main()
{
	//test02();
	test();
	return EXIT_SUCCESS;
}

二、多态的作用

  • 可以解决项目中的紧耦合问题,提供程序的可扩展性
  • 应用程序不必为每一个实现子类的功能调用编写代码

三、多态发生的三个条件

  • 有多继承
  • 重写父类的虚函数
  • 父类指针指向子类对象

四、多态实现的原理

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

  • 编译器只要发现类中有虚函数,编译器就会创建一张表,表中存放类中所有的虚函数的入口地址

  • 编译器会在类对象中安插一个虚函数表指针,虚函数表指针指向本类的虚函数表

  • 虚函数表指针是从父类继承下来的,虚函数表指针指向父类的虚函数表,编译器未来初始化从父类继承过来的虚函数表指针,编译器在我们的所有构造函数中添加了初始化虚函数指针的代码,让从父类继承过来的虚函数表指针指向子类自己的虚函数表

  • 当编译器发现子类重写了父类的虚函数,那么子类重写的函数就会覆盖掉虚函数表对应的父类的函数

  • 当我们通过父类指针去调用speak函数时,编译器会根据animal指针指向的内存空间中的虚函数表指针找到speak函数

在这里插入图片描述
在这里插入图片描述

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class People
{
public:
	// 虚函数
	virtual void Mypro()
	{
		cout << "父类的虚函数" << endl;  // 在编译阶段就确定了调用speak这个函数 
	}
};

class xishi :public People
{
public:
	// 重写父类虚函数
	virtual void Mypro()
	{
		cout << "西施的虚函数" << endl;
	}
};

class wangzhaojun :public People
{
public:
	// 重写父类虚函数
	virtual void Mypro()
	{
		cout << "王昭君的虚函数" << endl;
	}
};

class diaochan :public People
{
public:
	// 重写父类虚函数
	virtual void Mypro()
	{
		cout << "貂蝉的虚函数" << endl;
	}
};

// 同一个操作
void doLogin(People* pro)
{
	pro->Mypro();
}

void test()
{
	// 创建不同的对象 会产生不同的效果
	People* pro = NULL;// 父类指针
	pro = new xishi;// 父类指针 指向子类对象
	doLogin(pro);// 继承 向上转换   指针范围缩小
	delete pro;

	pro = new wangzhaojun;
	doLogin(pro);// 继承向上转换 
	delete pro;
}


int main()
{
	//test02();
	test();
	return EXIT_SUCCESS;
}

将代码分成 抽象层 实现层 业务层

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

// 抽象层  很多函数不在这里实现 仅仅定义一下
class rule
{
public:
	virtual int getnum(int a, int b)
	{
		return 0;
	}
};

// 实现层  通过虚函数 实现不改变父类的基础上 扩展功能
// 所以实现层 可以有很多类
class plus_rule :public rule
{
public:
	virtual int getnum(int a, int b)
	{
		return a + b;// 重载父类的虚函数
	}
};

// 实现层代码
class miux_rule :public rule
{
public:
	virtual int getnum(int a, int b)
	{
		return a - b;// 重载父类的虚函数  做减法
	}
};


// 业务层  业务层调用抽象层的代码 通过抽象层的对象指针 (该指针指向实现层对象)
int doLogin(rule *cal)
{
	int a = 10;
	int b = 20;
	int ret = cal->getnum(a,b);// 函数的调用还是通过父类指针  父类指针指向子类对象
	return ret;
}

// 测试
void test()
{
	rule* r = NULL;
	r = new plus_rule;
	cout << doLogin(r) << endl;
	delete r;

	
	rule *m = new miux_rule;// 父类指针指向子类对象   向上转换
	cout << doLogin(m) << endl;// 调用不同实现层的代码
	delete m;

}

int main()
{
	//test02();
	test();
	return EXIT_SUCCESS;
}

开闭原则:对修改源代码关闭,对扩展新功能进行开发

一般情况下,都会将抽象层的被重载函数写成纯虚函数

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

// 抽象层  很多函数不在这里实现 仅仅定义一下
class rule
{
public:
	virtual int getnum(int a, int b) = 0;
};

// 实现层  通过虚函数 实现不改变父类的基础上 扩展功能
// 所以实现层 可以有很多类
class plus_rule :public rule
{
public:
	virtual int getnum(int a, int b)
	{
		return a + b;// 重载父类的虚函数
	}
};

// 实现层代码
class miux_rule :public rule
{
public:
	virtual int getnum(int a, int b)
	{
		return a - b;// 重载父类的虚函数  做减法
	}
};


// 业务层  业务层调用抽象层的代码 通过抽象层的对象指针 (该指针指向实现层对象)
int doLogin(rule *cal)
{
	int a = 10;
	int b = 20;
	int ret = cal->getnum(a,b);// 函数的调用还是通过父类指针  父类指针指向子类对象
	return ret;
}

// 测试
void test()
{
	rule* r = NULL;
	r = new plus_rule;
	cout << doLogin(r) << endl;
	delete r;

	
	rule *m = new miux_rule;// 父类指针指向子类对象   向上转换
	cout << doLogin(m) << endl;// 调用不同实现层的代码
	delete m;

}

int main()
{
	//test02();
	test();
	return EXIT_SUCCESS;
}

有纯虚函数的类叫做抽象类,不可以实例化对象

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

// 抽象层  很多函数不在这里实现 仅仅定义一下
class rule
{
public:
	virtual int getnum(int a, int b) = 0;
};

// 实现层  通过虚函数 实现不改变父类的基础上 扩展功能
// 所以实现层 可以有很多类
class plus_rule :public rule
{
public:
	virtual int getnum(int a, int b)
	{
		return a + b;// 重载父类的虚函数
	}
};

// 实现层代码
class miux_rule :public rule
{
public:
	virtual int getnum(int a, int b)
	{
		return a - b;// 重载父类的虚函数  做减法
	}
};


// 业务层  业务层调用抽象层的代码 通过抽象层的对象指针 (该指针指向实现层对象)
int doLogin(rule *cal)
{
	int a = 10;
	int b = 20;
	int ret = cal->getnum(a,b);// 函数的调用还是通过父类指针  父类指针指向子类对象
	return ret;
}

// 测试
void test()
{
	rule* r = NULL;
	r = new plus_rule;
	cout << doLogin(r) << endl;
	delete r;

	
	rule *m = new miux_rule;// 父类指针指向子类对象   向上转换
	cout << doLogin(m) << endl;// 调用不同实现层的代码
	delete m;

}

int main()
{
	//test02();
	test();
	return EXIT_SUCCESS;
}

子类继承抽象类,子类必须实现抽象类的所有纯虚函数,不然子类也变为抽象类

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;


class Maker
{
public:
	virtual void func1() = 0;
	virtual void func2() = 0;
};

// 子类需要实现父类的所有虚函数
class Son :public Maker
{
	virtual void func1()
	{

	}
	virtual void func2()
	{

	}
};

void test()
{
	Son s;
}

int main()
{
	test();
	return EXIT_SUCCESS;
}

五、接口的定义

 所谓的接口,就是将内部实现细节封装起来,外部用户通过预留的接口可以使用接口的功能而不需要知晓内部具体的细节,C++中,通过类实现面对对象的编程,在基类中只给出纯虚函数的声明,然后在派生类中实现纯虚函数的具体定义的方式实现接口,不同派生类实现接口的方式也不相同

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Father
{
public:
	virtual void func1() = 0;// 接口的声明
	virtual void func2(int a) = 0;
	virtual void func3(int a,int b) = 0;
};

class Son :public Father
{
public:
	virtual void func1()
	{

	}
	virtual void func2(int a)
	{

	}
	virtual void func3(int a,int b)
	{

	}
};


int main()
{

	return EXIT_SUCCESS;
}

六、模板方法模式

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Drink
{
public:
	//zhu
	virtual void Boil() = 0;

	// 冲泡
	virtual void Brew() = 0;

	// 导入杯中
	virtual void PourInCup() = 0;

	// 加点辅料
	virtual void addSonm() = 0; 

	// 模板方法  抽象类将所有的虚基类方法归纳 然后   实现层实现每一个方法
	void func()
	{
		Boil();
		Brew();
		PourInCup();
		addSonm();
	}

};


class Coffee :public Drink
{
public:

	//zhu
	virtual void Boil()
	{
		cout << "煮点露水" << endl;
	}

	// 冲泡
	virtual void Brew()
	{
		cout << "拿铁" << endl;
	}

	// 导入杯中
	virtual void PourInCup()
	{
		cout << "导入盆中" << endl;
	}

	// 加点辅料
	virtual void addSonm()
	{
		cout << "加点糖" << endl;
	}
};

class Tea :public Drink
{
public:
	//zhu
	virtual void Boil()
	{
		cout << "煮点露水" << endl;
	}

	// 冲泡
	virtual void Brew()
	{
		cout << "茶叶" << endl;
	}

	// 导入杯中
	virtual void PourInCup()
	{
		cout << "导入盆中" << endl;
	}

	// 加点辅料
	virtual void addSonm()
	{
		cout << "加点糖" << endl;
	}

};


void test()
{
	// 父类指针指向子类对象
	Drink* d = NULL;
	d = new Coffee;
	d->func();
	delete d;

	d = new Tea;
	d->func();
	delete d;
	d = NULL;
}

int main()
{
	test();
	return EXIT_SUCCESS;
}

七、虚析构函数和纯虚析构函数

运行下面这段代码,因为静态联编:不会执行子类的析构函数

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Animal
{
public:
	Animal()
	{
		cout << "Animal 的构造函数" << endl;
	}

	~Animal()
	{
		cout << "Animal 的析构函数" << endl;
	}
};

class Son :public Animal
{
public:
	Son()
	{
		cout << "Son的构造函数" << endl;
		pName = new char[64];// 新建一个堆
		memset(pName,0,64);// 初始化0  全部字节
		strcpy(pName,"如花");// 拷贝字符串
	}

	~Son()
	{
		cout << "Son的析构函数" << endl;
		if (pName != NULL)
		{
			delete[] pName;
			pName = NULL;
		}
	}
public:
	char* pName;
};

void test()
{
	Animal* animal = new Son;
	delete animal;
}

int main()
{
	test();
	return EXIT_SUCCESS;
}

也就是,上面这段代码,基类指针指向了派生类对象,先执行基类的构造函数,然后执行子类的构造函数,但是delete的时候,只能执行基类的析构函数,不可以执行子类的析构函数

7.1 虚析构函数

虚析构函数是为了解决基类指针指向派生类对象,并使用基类指针释放派生类对象

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Animal
{
public:
	Animal()
	{
		cout << "Animal 的构造函数" << endl;
	}

	// 虚析构函数可以调用子类的虚析构函数
	virtual ~Animal()
	{
		cout << "Animal 的析构函数" << endl;
	}
};

class Son :public Animal
{
public:
	Son()
	{
		cout << "Son的构造函数" << endl;
		pName = new char[64];// 新建一个堆
		memset(pName,0,64);// 初始化0  全部字节
		strcpy(pName,"如花");// 拷贝字符串
	}

	~Son()
	{
		cout << "Son的析构函数" << endl;
		if (pName != NULL)
		{
			delete[] pName;
			pName = NULL;
		}
	}
public:
	char* pName;
};

void test()
{
	Animal* animal = new Son;
	delete animal;
}

int main()
{
	test();
	return EXIT_SUCCESS;
}

7.2 纯虚析构函数

纯虚析构函数,有纯虚析构函数的类是抽象类,不可以实例化对象

纯虚析构函数需要在类外实现

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Animal
{
public:
	Animal()
	{
		cout << "Animal 的构造函数" << endl;
	}

	// 虚析构函数可以调用子类的虚析构函数
	virtual ~Animal() = 0;
};

// 在类外实现纯虚析构函数
Animal::~Animal()
{

}

class Son :public Animal
{
public:
	Son()
	{
		cout << "Son的构造函数" << endl;
		pName = new char[64];// 新建一个堆
		memset(pName,0,64);// 初始化0  全部字节
		strcpy(pName,"如花");// 拷贝字符串
	}

	~Son()
	{
		cout << "Son的析构函数" << endl;
		if (pName != NULL)
		{
			delete[] pName;
			pName = NULL;
		}
	}
public:
	char* pName;
};

void test()
{
	Animal* animal = new Son;
	delete animal;
}

int main()
{
	test();
	return EXIT_SUCCESS;
}

虚析构函数和纯虚析构函数的区别:

  • 有纯虚析构函数的类是抽象类,不能实例对象 而且要在类外实现
  • 虚析构函数不需要在类外实现

八、重写重载重定义

  • 重载:同一作用域的同名函数
  • 重定义(隐藏):有继承,子类(派生类)重新定义父类(基类)的同名成员(非Virtual函数)
  • 重写:有继承,子类(派生类)重写父类(基类)的virtual函数,函数返回值,函数名字,函数参数,必须和基类中的虚函数一致

九、父类引用子类对象

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
using namespace std;

class Animal
{
public:
	virtual void speak()
	{
		cout << "Animal 的构造函数" << endl;
	}
};

class Dog :public Animal
{
public:
	virtual void speak()
	{
		cout << "构造函数" << endl;
	}
};


void test()
{
	Animal &animal = Dog();
	//delete animal;
	animal.speak();
}

int main()
{
	test();
	return EXIT_SUCCESS;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/3392.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

2014年848数据结构真题复习

求k频度K0; for&#xff08;i1;i<n;i&#xff09; 假如是1——8&#xff0c;执行了9次&#xff0c;8次有效&#xff0c;最后一次无效for&#xff08;ji;j<n;j&#xff09;k 我的理解&#xff1a;假设n为8我们看k频度实际上就是看内圈for的有效循环次数第一轮是1——8 八次…

基础算法 第七课——归并排序

文章目录导言归并排序的概念步骤说明逐步分析STEP1STEP2STEP3STEP4STEP5STEP6STEP0总结导言 这&#xff0c;是一篇现学现卖的文章。因为&#xff0c;我根本没学过归并排序。所以&#xff0c;这篇文章&#xff0c;绝对能让您学懂归并。如果不懂&#xff0c;那我就再学一遍&…

KVM Forum 2022应该关注的话题

1. QEMU 和 KVM 自动性能基准测试 QEMU & KVM Automated Performance Benchmarking SUSE - Dario Faggioli, SUSE SUSE正在开发一个框架&#xff0c;用于对虚拟化工作负载进行自动性能基准测试。它是围绕着MMTests&#xff08;已经在Linux内核社区使用了几年&#xff09;建…

2022-Java 后端工程师面试指南 -(SSM)

前言 种一棵树最好的时间是十年前&#xff0c;其次是现在 Tips 面试指南系列&#xff0c;很多情况下不会去深挖细节&#xff0c;是小六六以被面试者的角色去回顾知识的一种方式&#xff0c;所以我默认大部分的东西&#xff0c;作为面试官的你&#xff0c;肯定是懂的。 上面的…

Mybatis之foreach

文章目录一、foreach属性二、使用foreach批量删除(法一)1.接口2.mapper文件3.测试类4.运行结果三、使用foreach批量删除(法二)1.mapper文件四、使用foreach批量插入1.接口2.mapper文件3.测试类4.运行结果一、foreach属性 collection&#xff1a;指定数组或者集合 item&#xf…

FPGA时序约束01——基本概念

前言1. 越来越多的时序问题 随着FPGA时钟频率加快与其实现的逻辑功能越来越复杂&#xff0c;开发者遇到的问题很多时候不再是代码逻辑的问题&#xff0c;而是时序问题。一些开发者可能有这样的经历&#xff0c;一个模块在100MHz时钟运行没问题&#xff0c;而将时钟频率改为150…

【仿牛客网笔记】 Spring Boot进阶,开发社区核心功能-事务管理

添加评论中会用到事务管理。 解决的程度不同&#xff0c;层级不同。我们一般选择中间的级别。 选择时既能满足业务的需要&#xff0c;又能保证业务的安全性&#xff0c;在这样的前提下我们追求一个更高的性能。 第一类丢失更新 图中是没有事务隔离的情况 第二类丢失更新 脏…

需求工程方法的学习

作业要求&#xff1a;总结尽可能多的需求工程的方法和技术&#xff0c;要求归纳总结各种方法的适用场景、优缺点等。说明&#xff1a;其中需求工程包括需求获取、需求分析、规格说明、验证、管理等。只要是用于需求工程相关的技术和方法都可以算。 软件需求工程划分为需求开发…

Linux 中 man手册中函数后面括号数字释义

文章目录简介参考资料简介 Linux手册页项目记录了用户空间程序使用的Linux内核和C库接口。 用man手册查看系统命令&#xff0c;系统调用&#xff0c;glibc函数时&#xff0c;会发现其后面会有个括号&#xff0c;括号里面是一个数字&#xff0c;比如&#xff1a; access(2), …

一文了解Spring框架

目录 SpringBoot VS Servlet Spring是什么&#xff1f; loC&#xff1a;控制反转 DI 创建一个Spring项目 创建一个Spring IOC容器 注册Bean对象 获取Bean对象 注意事项&#xff1a; 类注解 为什么有这么多类注解&#xff1f; 注册与注入 方法注解 Bean Spr…

《R语言数据分析》2022-2023第一学期课程分析报告

1 (30分)基本操作题 1.1 (10分) 请写出下面问题的R代码 1.(2分)安装并加载gtools扩展包。 install.packages(“gtools”) library(gtools) 2.(2分)查看当前已经加载的所有包。 as.data.frame(installed.packages())$Package 3.(2分)查看gtools包的帮助网页。 ?gtools…

《清单革命》内容梳理随笔

《清单革命》内容梳理&随笔 起 书即是将四散的知识按照逻辑和网状联系编排起来。你应该这样去读&#xff0c;高屋建瓴、层次有秩、显得貌似自己有经验&#xff08;褒义&#xff09;的读&#xff0c;读出一些感想和方法论&#xff0c;无论是读出书里的还是书外的&#xff…

【MySQL高级】SQL优化

5. SQL优化 5.1 大批量插入数据 环境准备 &#xff1a; CREATE TABLE tb_user_2 (id int(11) NOT NULL AUTO_INCREMENT,username varchar(45) NOT NULL,password varchar(96) NOT NULL,name varchar(45) NOT NULL,birthday datetime DEFAULT NULL,sex char(1) DEFAULT NULL,…

【数据库】实验五 数据库综合查询|多表查询、聚集函数、orderby、groupby

文章目录参考文章本文在实验四的基础上增加了orderby、聚集函数、groupby、多表查询的知识点&#xff0c;相较于上一次实验的难度变大了&#xff0c;嵌套表达更多了&#xff0c;逐渐开始套娃…… 其实可以看成一个偏正短语来拆分&#xff0c;再写成SQL语句&#xff0c;比如查询…

微信小程序|基于小程序实现人脸数量检测

一、文章前言二、具体流程及准备三、开发步骤四、完整代码一、文章前言 此文主要通过小程序实现检测图片中的人脸数量并标记出位置信息。 当近视的小伙伴看不清远处的人时&#xff0c;用小程序一键识别就可以在手机上看清楚啦&#xff0c;是不是很实用呢。 典型应用场景&#x…

2022年还在做手动测试?是该好好反思了

为什么会写这篇文章呢&#xff1f;主要是前段时间有个朋友在QQ上和我交流&#xff0c;说他干了10年的手工测试了&#xff0c;现在还能不能转行。 说实话&#xff0c;当时我听完非常惊讶&#xff01;由此&#xff0c;我写了今天这篇文章。内容纯属个人观点&#xff0c;如果对你…

STM32CubeMX学习笔记(46)——USB接口使用(HID自定义设备)

一、USB简介 USB&#xff08;Universal Serial BUS&#xff09;通用串行总线&#xff0c;是一个外部总线标准&#xff0c;用于规范电脑与外部设备的连接和通讯。是应用在 PC 领域的接口技术。USB 接口支持设备的即插即用和热插拔功能。USB 是在 1994 年底由英特尔、康柏、IBM、…

浅刷牛客链表题,逐步深入链表,理解链表

作者&#xff1a;渴望力量的土狗 博客主页&#xff1a;渴望力量的土狗的博客主页 专栏&#xff1a;手把手带你刷牛客 工欲善其事必先利其器&#xff0c;给大家介绍一款超牛的斩获大厂offer利器——牛客网 点击免费注册和我一起刷题吧 目录 1、反转链表 2、删除链表的倒数第n个…

RocketMQ 消息重新投递 解析——图解、源码级解析

&#x1f34a; Java学习&#xff1a;Java从入门到精通总结 &#x1f34a; 深入浅出RocketMQ设计思想&#xff1a;深入浅出RocketMQ设计思想 &#x1f34a; 绝对不一样的职场干货&#xff1a;大厂最佳实践经验指南 &#x1f4c6; 最近更新&#xff1a;2022年11月4日 &#x…

35、Java——一个案例学会Dao+service层对数据表的增删改查

✅作者简介&#xff1a;热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;乐趣国学的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏&#xff1a;Java案例分…