设计模式之原型模式Prototype的C++实现

1、原型模式提出

在软件功能设计中,经常面临着"某些结构复杂的对象"的创建工作,且创建的对象想拥有其他对象在某一刻的状态,则可以使用原型模型。原型模型是通过拷贝构造函数来创建对象,并且该对象拥有其他对象在某一刻的状态。

2、需求描述

设计产品A,B,这个产品有重量和单价,输出产品的价格。

3、原型模式代码实现

cpp 复制代码
#include <iostream>
namespace factModel{
class AbsProduct{
public:
    AbsProduct(int kg,int price):m_kg(kg),m_price(price){};
    virtual void salePrice()=0;
    virtual AbsProduct* clone()=0;
    virtual ~AbsProduct()=default;
protected:
    int m_kg;
    int m_price;
};
class ProductA:public AbsProduct
{
public:
    ProductA(int kg,int price):AbsProduct(kg,price){};
    ProductA(const ProductA& org):AbsProduct(org.m_kg,org.m_price){
        this->m_kg = org.m_kg;
        this->m_price = org.m_kg;
    };

    virtual AbsProduct* clone() override
    {
        return new ProductA(this->m_kg,this->m_price);
    };

    virtual void salePrice()override
    {
        std::cout << "ProductA Info "<< m_kg << "kg, price " << m_price << ", sale " << m_kg * m_price
                  <<"."<< std::endl;
    };
    ~ProductA()=default;
};

class ProductB:public AbsProduct
{
public:
    ProductB(int kg,int price):AbsProduct(kg,price){};

    ProductB(const ProductB& org):AbsProduct(org.m_kg,org.m_price){
        this->m_kg = org.m_kg;
        this->m_price = org.m_kg;
    };
    virtual AbsProduct* clone()override
    {
        return new ProductB(this->m_kg,this->m_price);
    };
    virtual void salePrice()override
    {
        std::cout << "ProductB Info "<< m_kg << "kg, price " << m_price << ", sale " << m_kg * m_price
                  <<"."<< std::endl;
    };
    ~ProductB()=default;
};

//有构造参数时,是需要的具体工厂的,因为设计模式一般将new隔离在其他的文件中。(否则使用时就需要AbsProduct *pA = new ProductA(2,8);)
class AbsFact{
public:
    AbsFact()=default;
    virtual ~AbsFact()=default;
    virtual AbsProduct* createProduct()=0;
};
class FactA:public AbsFact
{
public:
    FactA()=default;
    ~FactA()=default;

    virtual AbsProduct* createProduct()override
    {
        AbsProduct* tmp = new ProductA(2,8);
        return tmp;
    }
};

class Use
{
public:
    AbsProduct* product;
    void setFactory(AbsProduct* pf){product = pf;};    //或者在构造函数中添加初始化字段值
    void saleProducts()
    {
         product->salePrice();
    }
};
}

int main()
{
    //创建方式1:通过工厂方法模式隔离ProductA的new
    factModel::AbsFact* pFactA = new factModel::FactA();
    factModel::AbsProduct *pPdtA = pFactA->createProduct();
    factModel::AbsProduct* pCloneA = pPdtA->clone();

    factModel::Use objA;
    objA.setFactory(pCloneA);
    objA.saleProducts();

    //创建方式2:直接使用new ProductB的方式创建对象
    factModel::AbsProduct *pPdtB = new factModel::ProductB(3,12);
    factModel::AbsProduct* pCloneB = pPdtB->clone();

    factModel::Use objB;
    objB.setFactory(pCloneB);
    objB.saleProducts();


    delete pFactA;
    pFactA = nullptr;
    delete pPdtA;
    pPdtA = nullptr;
    delete pCloneA;
    pCloneA = nullptr;

    delete pPdtB;
    pPdtB = nullptr;
    delete pCloneB;
    pCloneB = nullptr;
    return 0;
}

运行结果如下:

上面代码main函数中的创建方式根据实际情况选择,如果想隔离new,则使用方式1。原型模式的优点是将创建对象(该新创建的对象具有其他对象的状态,但与其他对象的地址区域是相互独立的)变得比较简洁。

相关推荐
历程里程碑25 分钟前
Linux20 : IO
linux·c语言·开发语言·数据结构·c++·算法
郝学胜-神的一滴27 分钟前
深入浅出:使用Linux系统函数构建高性能TCP服务器
linux·服务器·开发语言·网络·c++·tcp/ip·程序人生
天若有情67328 分钟前
【自研实战】轻量级ASCII字符串加密算法:从设计到落地(防查岗神器版)
网络·c++·算法·安全·数据安全·加密
czy87874751 小时前
深入了解 C++ 中的 `std::bind` 函数
开发语言·c++
我在人间贩卖青春1 小时前
C++之继承的方式
c++·private·public·protected·继承方式
BD_Marathon2 小时前
设计模式——依赖倒转原则
java·开发语言·设计模式
BD_Marathon2 小时前
设计模式——里氏替换原则
java·设计模式·里氏替换原则
jmxwzy2 小时前
设计模式总结
设计模式
智者知已应修善业3 小时前
【洛谷P9975奶牛被病毒传染最少数量推导,导出多样例】2025-2-26
c语言·c++·经验分享·笔记·算法·推荐算法
Trouvaille ~3 小时前
【Linux】应用层协议设计实战(一):自定义协议与网络计算器
linux·运维·服务器·网络·c++·http·应用层协议