何为C++中的协变

C++中的协变

文章目录

前言

​ 在面向对象编程中,继承和多态是核心概念,它们允许程序员创建灵活的代码结构,可以轻松扩展和维护。C++中的协变是这些概念的一个高级应用,它允许派生类方法改变其返回类型,以返回更具体的派生类型。这种能力不仅增强了类型安全,还提供了更丰富的接口设计选项。


一、协变的概念

派生类重写基类虚函数时,与基类虚函数返回值类型不同 。即基类虚函数返回基类对象的指针或者引用,派生类虚函数返回派生类对象的指针或者引用时,称为协变。

二、构成协变的条件及案例

案例1:

父类、子类返回各自的指针或引用

​ 协变最常见的用途是在实现一个对象的克隆操作时。例如,一个基类定义了一个返回基类指针的clone方法,而在派生类中,我们希望clone返回派生类的指针,这样就可以保留对象的完整类型信息。

首先给出一段比较标准的协变示例代码:

cpp 复制代码
// 父类、子类返回各自的指针或引用
class Base 
{
public:
	virtual Base* clone() const { return new Base(*this); }
};

class Derived : public Base 
{
public:
	virtual Derived* clone() const override { return new Derived(*this); }
};

测试用例代码:

cpp 复制代码
void test1()
{
	Base b;
	Base* p1 = b.clone();

	Derived d;
	Derived* p2 = d.clone();
}

在上面的代码中,Derived类重写了Base类的clone方法,并改变了返回类型为Derived*

案例2:

父类、子类分别返回其他父子类的指针或引用

cpp 复制代码
// 首先任意给出一个父类及其派生类
class A {};
class B : public A{};

// 接着定义另外一对父子类,其中构成多态的函数返回值类型分别为已有的父子类型
class Person 
{
public:
	virtual A* f() 
	{ 
		cout << "A::f()" << endl;
		return new A;
	}
};

class Student : public Person 
{
public:
	virtual B* f() 
	{
		cout << "B::f()" << endl;
		return new B;
	}
};

注意:上面代码中的Person类和Student类中的 f() 函数重写构成多态的前提下,函数返回值不同的情况对应只能是父类中的 f() 函数返回值为父类,子类同理。

案例3:

类似的,在设计模式中也存在协变的案例,比如工厂模式:

首先给出一个产品基类和产品实体类:

cpp 复制代码
// 产品基类
class Product {
public:
    virtual void use() = 0;
};

// 具体产品A
class ConcreteProductA : public Product {
public:
    void use() override {
        std::cout << "Using ConcreteProductA" << std::endl;
    }
};

// 具体产品B
class ConcreteProductB : public Product {
public:
    void use() override {
        std::cout << "Using ConcreteProductB" << std::endl;
    }
};

下面是对应工厂类的多态案例:

(注意子类重写的函数返回值都是父类指针,但是指向的是子类对象)

cpp 复制代码
// 工厂基类
class Factory {
public:
    virtual Product* createProduct() = 0;
};

// 具体工厂A
class ConcreteFactoryA : public Factory {
public:
    Product* createProduct() override {
        return new ConcreteProductA();
    }
};

// 具体工厂B
class ConcreteFactoryB : public Factory {
public:
    Product* createProduct() override {
        return new ConcreteProductB();
    }
};

类似的,下面是工厂类协变的案例:

(注意子类重写的函数返回值都是子类指针,父子类对应函数返回值类型不同)

cpp 复制代码
class Factory {
public:
    virtual Product* createProduct() = 0;
    virtual ~Factory() {}
};

class ConcreteFactoryA : public Factory {
public:
    ConcreteProductA* createProduct() override {
        return new ConcreteProductA();
    }
};

class ConcreteFactoryB : public Factory {
public:
    ConcreteProductB* createProduct() override {
        return new ConcreteProductB();
    }
};

总结

​ 通过本文的探讨,我们了解到协变是C++中一个强大而有用的特性,它允许派生类在保持多态行为的同时,返回更具体的类型。这不仅使得代码更加直观,也为设计模式的实现提供了更大的灵活性。然而,使用协变时需要谨慎,确保不违反基本的继承原则和类型安全。

相关推荐
Spring-wind1 分钟前
【golang】为什么协程开销小于线程
java·开发语言·golang
Yang-Never6 分钟前
Open GL ES -> SurfaceView + 自定义EGL实现OpenGL渲染框架
android·开发语言·图像处理·android studio
wjs202414 分钟前
Perl 发送邮件
开发语言
大刘讲IT1 小时前
构建实时、融合的湖仓一体数据分析平台:基于 Delta Lake 与 Apache Iceberg
开发语言·python·sql·mysql·数据挖掘·数据分析·json
ChoSeitaku1 小时前
NO.82十六届蓝桥杯备战|动态规划-从记忆化搜索到动态规划|下楼梯|数字三角形(C++)
c++·蓝桥杯·动态规划
ChoSeitaku1 小时前
NO.73十六届蓝桥杯备战|搜索算法-剪枝与优化-记忆化搜索|数的划分|小猫爬山|斐波那契数|Function|天下第一|滑雪(C++)
c++·蓝桥杯·剪枝
龙虾在剥我的壳1 小时前
OpenCV——图像融合
c++·opencv·计算机视觉
冯韶晗1 小时前
Scala语言的区块链
开发语言·后端·golang
T - mars2 小时前
python爬虫:喜马拉雅登录案例
开发语言·python
末央&2 小时前
【C++】list底层封装和实现
c++·windows·list