C++闲谈04——设计模式
创始人
2024-03-17 01:39:01

C++闲谈04——设计模式

单例模式

饿汉式单例模式

#include
#includeusing namespace std;
mutex mtx;class SingleTon{
public:static SingleTon* GetInstance() {if (instance == nullptr) {lock_guard lk(mtx);  // 不是尖括号instance = new SingleTon();}return instance;}
private:SingleTon() {cout << "test" << endl;}static SingleTon* instance;
};SingleTon* SingleTon::instance = nullptr;int main() {SingleTon* test1 = SingleTon::GetInstance();SingleTon* test2 = SingleTon::GetInstance();system("pause");return 0;
}

优点:对象提前创建好了 使用的时候无需等待

缺点:对象提前创建 占用内存

懒汉式单例模式

#include
#includeusing namespace std;
mutex mtx;class SingleTon{
public:static SingleTon* GetInstance() {if (instance == nullptr) {lock_guard lk(mtx);  // 不是尖括号instance = new SingleTon();}return instance;}
private:SingleTon() {cout << "test" << endl;}static SingleTon* instance;
};SingleTon* SingleTon::instance = nullptr;int main() {SingleTon* test1 = SingleTon::GetInstance();SingleTon* test2 = SingleTon::GetInstance();system("pause");return 0;
}

优点:使用的时候才创建 不占内存

缺点:使用的时候需要创建 比较耗时

工厂模式

工厂模式

简单工厂模式

  • 工厂类
  • 抽象产品类
  • 具体产品类

image-20221203120811174

#include 
#include 
using namespace std;typedef enum ProductTypeTag
{Hair,Hisense,
}PRODUCTTYPE;//抽象产品类 TV(电视机类)
class TV
{
public:virtual void Show() = 0;virtual ~TV(){};//声明析构函数为虚函数,防止内存泄漏
};//具体产品类 HairTV(海尔电视类)
class HairTV : public TV
{
public:void Show(){cout<<"I'm HairTV "<
public:void Show(){cout<<"I'm HisenseTV"<
public:TV* CreateTV(PRODUCTTYPE type){switch (type){case Hair:return new HairTV();case Hisense:return new HisenseTV();default:return NULL;}}
};int main(int argc, char *argv[])
{// 创建工厂类对象TVFactory* myTVFactory = new  TVFactory();TV* hairTV = myTVFactory->CreateTV(Hair);if (hairTV != NULL)hairTV->Show();TV* hisenseTV = myTVFactory->CreateTV(Hisense);if (hisenseTV != NULL)hisenseTV->Show();delete  myTVFactory;myTVFactory = NULL;delete hairTV;hairTV = NULL;delete hisenseTV;hisenseTV = NULL;  return 0;
}

TVFactory 是工厂类,它是整个系统的核心,提供了静态工厂方法CreateTV(),该方法中包含一个字符串类型的参数,在内部业务逻辑中根据参数值得不同实例化不同的具体产品类,返回相依的对象。简单工厂模式的缺点是 如增加 生成TCL电视时,需要修改工厂类TVFactory。

小结:

在程序中,需要创建的对象很多,导致对象的new操作多且杂时,需要使用简单工厂模式。

工厂方法模式

简单工厂模式中最大的缺点是当有新产品要加入系统时,必须要修改工厂类,加入必要的处理逻辑,违背了“开闭原则”

工厂方法模式定义:在工厂模式中,工厂父类负责定义创建产品对象的公告接口,而工厂子类负责生成具体的产品对象。目的是将产品的实例化操作延迟到工厂子类中完成,通过工厂子类来确定究竟应该实例化哪一个具体产品类。

image-20221203145257797

工厂方法模式的意义是定义一个创建产品对象的工厂接口,将实际创建工作推迟到子类当中。核心工厂类不再负责产品的创建,这样核心类成为一个抽象工厂角色,仅负责具体工厂子类必须实现的接口,这样进一步抽象化的好处是使得工厂方法模式可以使系统在不修改具体工厂角色的情况下引进新的产品。

#include 
using namespace std;/*抽象产品类 TV(电视机类)*/ 
class TV
{
public:virtual void Show() = 0;virtual ~TV();//声明析构函数为虚函数,防止内存泄漏
};/*具体产品类 HairTV(海尔电视机类)*/ 
class HairTV : public TV
{
public:void Show(){cout<< "I'm HairTV"<
public:void Show(){cout<< "I'm HisenseTV"<
public:virtual TV *CreateTV() = 0;virtual ~TVFactory(){};//析构函数声明为虚函数,防止内存泄漏
};/*具体工厂类 HairTVFactory(海尔电视机工厂类)*/ 
class HairTVFactory : public TVFactory
{
public:TV *CreateTV(){return new HairTV ();}
};/*具体工厂类 HisenseTV(海信电视机工厂类)*/ 
class HisenseTVFactory : public TVFactory
{
public:TV *CreateTV(){return new HisenseTV ();}
};int main(int argc , char *argv [])
{TVFactory *hairTVFactory = new HairTVFactory();TV *hairTV = hairTVFactory->CreateTV();hairTV->Show();TVFactory *hisenseTVFactory = new HisenseTVFactory();TV *hisenseTV = hisenseTVFactory->CreateTV();hisenseTV->Show();if (hairTVFactory!= NULL){delete hairTVFactory;hairTVFactory = NULL;}if (hairTV != NULL){delete hairTV;hairTV = NULL;}if (hisenseTVFactory != NULL){delete hisenseTVFactory;hisenseTVFactory = NULL;}if (hisenseTV != NULL){delete hisenseTV;hisenseTV = NULL;}return 0;
}

抽象工厂模式

抽象工厂模式是工厂方法模式的泛化版,工厂模式是一种特殊的抽象工厂模式,在工厂模式中,每个具体工厂只能生产一种具体的产品,如海尔电视机厂只生产海尔电视机,而抽象工厂方法模式中,一个具体的工厂可以生产多个具体产品。

image-20221203145524563

优点:

抽象工厂模式将产品族的依赖与约束关系放到抽象工厂中,便于管理。
职责解耦,用户不需要关心一堆自己不关心的细节,由抽象工厂来负责组件的创建
切换产品族容易,只需要增加一个具体工厂实现,客户端选择另一个套餐就可以了
缺点:

抽象工厂模式类增加的速度很快,有一个产品族就需要增加一个具体工厂实现,比较繁琐
产品族难以扩展产品。当产品族中增加一个产品时,抽象工厂接口中需要增加一个函数,对应的所有具体工厂实现都需要修改,修改放大严重。
抽象工厂并未完全屏蔽创建细节,给出的都是组件。对于这种情况可以结合工厂模式或简单工厂模式一起使用

#include 
using namespace std;// 抽象产品类类 Television(电视机类)
class Television
{
public:virtual void Show() = 0;virtual ~Television(){};//析构函数声明为虚函数,防止内存泄漏
};//具体产品类 HaierTelevision(海尔电视机类)
class HaierTelevision : public Television
{
public:void Show(){cout << "I'm HaierTelevision" << endl;}
};
//具体产品类 TCLTelevision(TCL电视机类)
class TCLTelevision : public Television
{
public:void Show(){cout << "I'm TCLTelevision" << endl;}
};// 抽象产品类  AirCondition(空调类)
class AirCondition
{
public:virtual void Show() = 0;virtual ~AirCondition(){};//析构函数声明为虚函数,防止内存泄漏
};
//具体产品类 HairAirCondition(海尔空调类)
class HairAirCondition : public AirCondition
{
public:void Show(){cout << "I'm HairAirCondition" << endl;}
};
//具体产品类 TCLAirCondition(TCL空调类)
class TCLAirCondition : public AirCondition
{
public:void Show(){cout << "I'm TCLAirCondition" << endl;}
};// 抽象工厂类 EFactory(电器工厂类)
class EFactory
{
public:virtual Television* CreateTelevision() = 0;virtual AirCondition* CreateAirCondition() = 0;virtual ~EFactory(){};//析构函数声明为虚函数,防止内存泄漏
};
//具体工厂类 HairFactory(海尔工厂类)
class HairFactory : public EFactory
{
public:Television* CreateTelevision(){return new HaierTelevision();}AirCondition* CreateAirCondition(){return new HairAirCondition();}
};
//具体工厂类 TCLFactory(TCL工厂类) 
class TCLFactory : public EFactory
{
public:Television* CreateTelevision(){return new TCLTelevision();}AirCondition* CreateAirCondition(){return new TCLAirCondition();}
};int main(int argc, char *argv[])
{EFactory *hairFactory = new HairFactory ();/*实例化工厂抽象类*/Television *haierTelevision =hairFactory->CreateTelevision();/*实例化产品抽象类*/AirCondition *haierAirCondition = hairFactory->CreateAirCondition();haierTelevision->Show();haierAirCondition->Show();EFactory *tCLFactory = new TCLFactory ();Television *tCLTelevision = tCLFactory->CreateTelevision();AirCondition *tCLAirCondition = tCLFactory->CreateAirCondition();tCLTelevision->Show();tCLAirCondition->Show();if (hairFactory != NULL){delete hairFactory;hairFactory = NULL;}if (haierTelevision != NULL){delete haierTelevision;haierTelevision= NULL;}if (tCLAirCondition != NULL){delete tCLAirCondition;tCLAirCondition = NULL;}if (tCLFactory != NULL){delete tCLFactory;tCLFactory= NULL;}if (tCLTelevision != NULL){delete tCLTelevision;tCLTelevision = NULL;}if (tCLAirCondition != NULL){delete tCLAirCondition;tCLAirCondition = NULL;}
}

总结

大家应该已经发现了,其实抽象工厂模式如果只有一个组件的话,其实是退化到了工厂方法模式,也就是没有了产品族的概念,只剩一个产品了,因此简单工厂,工厂方法,抽象工厂这三者之间是有内在联系的,区别只是产品的复杂度。

抽象工厂的本质是选择产品族,因此大家可以根据这个特征来识别是否可以应用抽象工厂。
简单工厂:唯一工厂类,一个产品抽象类,工厂类的创建方法依据入参判断并创建具体产品对象。
工厂方法:多个工厂类,一个产品抽象类,利用多态创建不同的产品对象,避免了大量的if-else判断。
抽象工厂:多个工厂类,多个产品抽象类,产品子类分组,同一个工厂实现类创建同组中的不同产品,减少了工厂子类的数量。 增加了泛化能力

相关内容

热门资讯

埃菲尔铁塔在哪 中国仿建埃菲尔... 2019年4月26日,广西南宁市,街头惊现一座巨型山寨版埃菲尔铁塔,高约20米,白色塔身,造型逼真,...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...
脚上的穴位图 脚面经络图对应的... 人体穴位作用图解大全更清晰直观的标注了各个人体穴位的作用,包括头部穴位图、胸部穴位图、背部穴位图、胳...
应用未安装解决办法 平板应用未... ---IT小技术,每天Get一个小技能!一、前言描述苹果IPad2居然不能安装怎么办?与此IPad不...
世界上最漂亮的人 世界上最漂亮... 此前在某网上,选出了全球265万颜值姣好的女性。从这些数量庞大的女性群体中,人们投票选出了心目中最美...
猫咪吃了塑料袋怎么办 猫咪误食... 你知道吗?塑料袋放久了会长猫哦!要说猫咪对塑料袋的喜爱程度完完全全可以媲美纸箱家里只要一有塑料袋的响...
demo什么意思 demo版本... 618快到了,各位的小金库大概也在准备开闸放水了吧。没有小金库的,也该向老婆撒娇卖萌服个软了,一切只...
埃菲尔铁塔在哪 中国仿建埃菲尔... 2019年4月26日,广西南宁市,街头惊现一座巨型山寨版埃菲尔铁塔,高约20米,白色塔身,造型逼真,...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...