C++之多态使用小结

1、多态定义

1.1 多态概念     

        C++多态性(Polymorphism)是面向对象编程(OOP)的一个重要特性之一,它允许我们使用统一的接口来处理不同类型的对象。多态性使得程序更加灵活、可扩展并且易于维护。

       通俗来说,就是多种形态,具体点就是去完成某个行为,当不同的对象去完成时会产生出不同的状态

1.2 多态的构成条件

       多态是在不同继承关系的类对象,去调用同一函数,产生了不同的行为。对于同一行为买票(BuyTicket)比如,由于Student继承了Person。Person对象买票全价,Student对象买票半价。
  在继承中要构成多态还有两个条件:

  • 必须通过基类的指针或者引用调用虚函数。
  • 被调用的函数必须是虚函数,且派生类必须对基类的虚函数进行重写

2、多态的使用

2.1 虚函数

      构成多态的条件中提到了虚函数,所谓的虚函数,就是被virtual修饰的类成员函数。具体如下,函数BuyTicket()即为虚函数。其中只要在申明时添加virtual关键字,在具体实现虚函数方法时不需要添加。

class Person {
public:
	virtual void BuyTicket();
};
void Person::BuyTicket()
{
	cout << "买票-全价" << endl;
}

2.2 虚函数的重写

      虚函数的重写(覆盖):派生类中有一个跟基类完全相同的虚函数(即派生类虚函数与基类虚函数的返回值类型、函数名字、参数列表完全相同),称子类的虚函数重写了基类的虚函数。我们看如下例子派生类Student对虚函数BuyTicket() 进行了重写。

class Person {
public:
	virtual void BuyTicket();
};
void Person::BuyTicket()
{
	cout << "买票-全价" << endl;
}

class Student : public Person {
public:
	void BuyTicket();
};
void Student::BuyTicket()
{
	cout << "买票-半价" << endl;
}

3、多态练习

3.1 派生构造函数使用

      派生函数在使用基类中成员变量的同时,再添加成员变量,这种情况用法可参考如下案例。基类Person中包括成员变量:name、age、gender。派生类Student又添加成员变量:id、score。那么,在构造函数中,使用如下命令,其中name、age、gender为继承People,id、score为赋值。

Student::Student(string name, int age, string gender, string id, int score) : People(name, age, gender), id(id), score(score)
{}
#include <iostream>
#include<string>
using namespace std;
class People
{
public:
	People();
	People(string name, int age, string gender);
	~People();
	void PrintInfo();

public:
	string name;//姓名
	int age;//年龄
	string gender;//性别

};

People::People(string name, int age, string gender)
{
	this->name = name;
	this->age = age;
	this->gender = gender;
}
void People::PrintInfo()
{
	cout << "name :" << this->name << endl; 
	cout << "age :" << this->age << endl;
	cout << "gender :" << this->gender << endl;
}

People::People()
{
}

People::~People()
{
}

class Student :public People
{
public:
	Student();
	Student(string name, int age, string gender, string id, int score);
	~Student();
	void PrintInfo();

public:
	string id;
	int score;

};
Student::Student(string name, int age, string gender, string id, int score) : People(name, age, gender), id(id), score(score)
{}

void Student::PrintInfo()
{
	cout << "name :" << this->name << endl;
	cout << "age :" << this->age << endl;
	cout << "gender :" << this->gender << endl;
	cout << "id :" << this->id << endl;
	cout << "score :" << this->score << endl;
}

Student::Student()
{
}

Student::~Student()
{
}

3.2 案例一:

      下面介绍对这句话的理解:基类的指针引用或者调用调用虚函数

其可以这样理解:

(1)如果指针指向基类地址,那么只能访问基类的虚函数

(2)如果指针指向派生类地址,那么只能访问派生类中与基类对应的函数

#include <iostream>
#include<string>
using namespace std;
class Person
{
public:
	virtual void BuyTicket()
	{
		cout << "买票-全价" << endl;
	}
};
class Student : public Person
{
public:
	virtual void BuyTicket()
	{
		cout << "买票-半价" << endl;
	}
};
void Func(Person& p)
{
	p.BuyTicket();
}
int main()
{
	Person ps;
	Student st;
	Func(ps);
	Func(st);
	system("pause");
	return 0;
}

       上述代码构成了多态吗?首先是虚函数BuyTicket(),其次完成了重写。结果如下,原因是ps对应的是基类指针,因此只能访问基类的BuyTicket()函数;而st对应的是派生类指针,因此访问派生类的BuyTicket()函数。

3.3 案例二:

#include <iostream>
#include<string>
using namespace std;
class A
{
public:
	virtual void func(int val)
	{
		cout << "A->" << val << std::endl;
	}
};

class B : public A
{
public:
	virtual void func(int val)
	{
		cout << "B->" << val << std::endl;
	}
};

int main()
{
	A* p = new B;
	p->func(1);
	system("pause");
	return 0;
}

       上述的代码调用构成多态吗?首先是虚函数func(),其次完成了重写上述代码中,p指针是指向派生类B地址,因此,只能访问B对应的func()函数,是B->1

3.4 案例三:

#include <iostream>
#include<string>
using namespace std;
class A
{
public:
	virtual void func(int val = 1)
	{
		std::cout << "A->" << val << std::endl;
	}
	virtual void test()
	{
		func();
	}
};

class B : public A
{
public:
	void func(int val = 0)
	{
		cout << "B->" << val << std::endl;
	}
};

int main(int argc, char* argv[])
{
	B*p = new B;
	p->test();
	system("pause");
	return 0;
}

在 B 类的 func 函数中,有一个默认参数值为 0,这意味着当 test 函数被调用时,它会使用这个默认值。然而,test 函数中有一个显式的调用 func(),这意味着它会调用 B 类中的 func 实现,并传递参数 1

参考博客:

【1】【C++】多态(举例+详解,超级详细)_c++多态-CSDN博客

相关推荐

  1. C++的知识】

    2024-07-14 10:14:03       28 阅读
  2. C语言封装,继承,

    2024-07-14 10:14:03       28 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-07-14 10:14:03       67 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-07-14 10:14:03       72 阅读
  3. 在Django里面运行非项目文件

    2024-07-14 10:14:03       58 阅读
  4. Python语言-面向对象

    2024-07-14 10:14:03       69 阅读

热门阅读

  1. RESTful API的设计与实现

    2024-07-14 10:14:03       24 阅读
  2. 39.全连接层问题

    2024-07-14 10:14:03       22 阅读
  3. 力扣题解(分割回文串II)

    2024-07-14 10:14:03       22 阅读
  4. Linux C++ 054-设计模式之外观模式

    2024-07-14 10:14:03       26 阅读
  5. 大白话【卷积神经网络】工作原理

    2024-07-14 10:14:03       26 阅读
  6. [NOIP2005 普及组] 采药

    2024-07-14 10:14:03       24 阅读
  7. 【Git使用】管理代码

    2024-07-14 10:14:03       21 阅读
  8. 分区和分桶的区别

    2024-07-14 10:14:03       23 阅读
  9. vue vite自动化路由 无需手动配置

    2024-07-14 10:14:03       18 阅读
  10. C#学习

    2024-07-14 10:14:03       27 阅读