C++设计模式---创建型模式

创建型模式分类: 

1. 简单工厂模式 

1.1 简单工厂模式理论

1.2 简单工厂模式案例

代码示例: 

#include<iostream>
using namespace std;

//抽象水果
class AbstractFruit
{
public:
	virtual void ShowName() = 0;

};

//苹果
class Apple :public AbstractFruit
{
	virtual void ShowName()
	{
		cout << "我是苹果!" << endl;
	} 
};

//香蕉
class Banana :public AbstractFruit
{
	virtual void ShowName()
	{
		cout << "我是香蕉!" << endl;
	}
};

//鸭梨
class Pear :public AbstractFruit
{
	virtual void ShowName()
	{
		cout << "我是鸭梨!" << endl;
	}
};

//水果工厂
class FruitFactor {
public:
	static AbstractFruit* CreateFruit(string flag) {
		if (flag == "apple") 
		{
			return new Apple;
		}
		else if (flag == "banana") 
		{
			return new Banana;
		}
		else if (flag == "pear") 
		{
			return new Pear;
		}
		else 
		{
			return NULL;
		}
	}
};

void test05() 
{
	FruitFactor* factory = new FruitFactor;
	AbstractFruit* fruit = factory->CreateFruit("apple");
	fruit->ShowName();
	delete fruit;

	fruit = factory->CreateFruit("banana");
	fruit->ShowName();
	delete fruit;

	fruit = factory->CreateFruit("pear");
	fruit->ShowName();
	delete fruit;

	fruit = NULL;
	delete factory;
	factory = NULL;
}

int main()
{
	test05();
	return 0;
}

1.3 简单工厂模式的优缺点

优点:

        不需要记住具体类名,甚至连具体参数都不用记忆。
        实现了对象创建和使用的分离。
        系统的可扩展性也就变得非常好,无需修改接口和原类。

缺点:

        增加系统中类的个数,复杂度和理解度增加。
        增加了系统的抽象性和理解难度。

        违反“开闭原则”,添加新产品需要修改工厂逻辑,工厂越来越复杂。

1.4 简单工厂模式适用场景 

1. 工厂类负责创建的对象比较少,由于创建的对象较少,不会造成工厂方法中的业务逻辑太过复杂。
2. 客户端只知道传入工厂类的参数,对于如何创建对象并不关心。

2. 工厂方法模式

2.1 工厂方法模式理论

 简单工厂模式+"开闭原则"=工厂方法模式

2.2 工厂方法模式案例

代码示例: 

#include <iostream>
using namespace std;

//抽象水果
class AbstractFruit 
{
public:
	virtual void ShowName() = 0;
};

//苹果
class Apple :public AbstractFruit 
{
public:
	virtual void ShowName() 
	{
		cout << "我是苹果!" << endl;
	}
};

//香蕉
class Banana :public AbstractFruit 
{
public:
	virtual void ShowName() 
	{
		cout << "我是香蕉!" << endl;
	}
};

//鸭梨
class Pear :public AbstractFruit 
{
public:
	virtual void ShowName() 
	{
		cout << "我是鸭梨!" << endl;
	}
};

//抽象工厂
class AbstractFruitFactory 
{
public:
	virtual AbstractFruit* CreateFruit() = 0;
};


//苹果工厂
class AppleFactory :public AbstractFruitFactory 
{
public:
	virtual AbstractFruit* CreateFruit() 
	{
		return new Apple;
	}
};

//香蕉工厂
class BananaFactory :public AbstractFruitFactory 
{
public:
	virtual AbstractFruit* CreateFruit() 
	{
		return new Banana;
	}
};

//鸭梨工厂
class PearFactory :public AbstractFruitFactory 
{
public:
	virtual AbstractFruit* CreateFruit() 
	{
		return new Pear;
	}
};


void test05() 
{
	AbstractFruitFactory* factory = NULL;
	AbstractFruit* fruit = NULL;

	//创建一个苹果工厂
	factory = new AppleFactory;
	fruit = factory->CreateFruit();
	fruit->ShowName();
	delete fruit;
	delete factory;

	//创建一个香蕉工厂
	factory = new BananaFactory;
	fruit = factory->CreateFruit();
	fruit->ShowName();
	delete fruit;
	delete factory;

	//创建一个鸭梨工厂
	factory = new PearFactory;
	fruit = factory->CreateFruit();
	fruit->ShowName();
	delete fruit;
	delete factory;
	fruit = NULL;
	factory = NULL;

}

int main() 
{
	test05();
	return 0;
}

2.3 工厂方法模式适用场景

1. 客户端不知道它所需要的对象的类。
2. 抽象工厂类通过其子类来指定创建哪个对象。

2.4 工厂方法模式的总结 

传统:如果想创建两个苹果,得具体实例化两个,然后再调用功能,以后再用还得记名字

Apple a;
a.func();
Aplle b;
b.func();

工厂方法模式:可以不用具体一个对象,然后调用

factory = new AppleFactory;
fruit = factory->CreateFruit();//造a
fruit->ShowName();
fruit = factory->CreateFruit();//造b
fruit->ShowName();

3. 抽象工厂模式 

3.1 抽象工厂模式理论

3.2 抽象工厂模式案例 

#include <iostream>
using namespace std;

//抽象苹果
class AbstractApple 
{
public:
	virtual void ShowName() = 0;
};

//中国苹果
class ChinaApple :public AbstractApple 
{
public:
	virtual void ShowName() 
	{
		cout << "中国苹果!" << endl;
	}
};

//美国苹果
class USAApple :public AbstractApple 
{
public:
	virtual void ShowName() 
	{
		cout << "美国苹果!" << endl;
	}
};

//日本苹果
class JapanApple :public AbstractApple 
{
public:
	virtual void ShowName() 
	{
		cout << "日本苹果!" << endl;
	}
};

//抽象香蕉
class AbstractBanana 
{
public:
	virtual void ShowName() = 0;
};

//中国香蕉
class ChinaBanana :public AbstractBanana 
{
public:
	virtual void ShowName() 
	{
		cout << "中国香蕉!" << endl;
	}
};

//美国香蕉
class USABanan :public AbstractBanana 
{
public:
	virtual void ShowName() 
	{
		cout << "美国香蕉!" << endl;
	}
};

//日本香蕉
class JapanBanan :public AbstractBanana 
{
public:
	virtual void ShowName() 
	{
		cout << "日本香蕉!" << endl;
	}
};

//抽象鸭梨
class AbstractPear 
{
public:
	virtual void ShowName() = 0;
};


//中国鸭梨
class ChinaPear :public AbstractPear 
{
public:
	virtual void ShowName() 
	{
		cout << "中国鸭梨!" << endl;
	}
};

//美国鸭梨
class USAPear :public AbstractPear 
{
public:
	virtual void ShowName() 
	{
		cout << "美国鸭梨!" << endl;
	}
};

//日本鸭梨
class JapanPear :public AbstractPear 
{
public:
	virtual void ShowName() 
	{
		cout << "日本鸭梨!" << endl;
	}
};


//抽象工厂 针对产品族
class AbstractFactory 
{
public:
	virtual AbstractApple* CreateApple() = 0;
	virtual AbstractBanana* CreateBanana() = 0;
	virtual AbstractPear* CreatePear() = 0;
};

//中国工厂
class ChinaFactory :public AbstractFactory 
{
public:
	virtual AbstractApple* CreateApple() 
	{
		return new ChinaApple;
	}
	virtual AbstractBanana* CreateBanana() 
	{
		return new ChinaBanana;
	}
	virtual AbstractPear* CreatePear() 
	{
		return new ChinaPear;
	}
};

//美国工厂
class USAFactory :public AbstractFactory 
{
public:
	virtual AbstractApple* CreateApple() 
	{
		return new USAApple;
	}
	virtual AbstractBanana* CreateBanana() 
	{
		return new USABanan;
	}
	virtual AbstractPear* CreatePear() 
	{
		return new  USAPear;
	}
};

//日本工厂
class JapanFactory :public AbstractFactory 
{
public:
	virtual AbstractApple* CreateApple() 
	{
		return new JapanApple;
	}
	virtual AbstractBanana* CreateBanana() 
	{
		return new JapanBanan;
	}
	virtual AbstractPear* CreatePear() 
	{
		return new  JapanPear;
	}
};

void test06() 
{
	AbstractFactory* factory = NULL;
	AbstractApple* apple = NULL;
	AbstractBanana* banana = NULL;
	AbstractPear* pear = NULL;

	//中国工厂
	factory = new ChinaFactory;
	apple = factory->CreateApple();
	banana = factory->CreateBanana();
	pear = factory->CreatePear();

	apple->ShowName();
	banana->ShowName();
	pear->ShowName();
}

int main() 
{
	test06();
	return 0;
}

3.3 抽象工厂模式的优缺点

优点:
        拥有工厂方法模式的优点

        当一个产品族中的多个对象被设计成一起工作时,它能够保证客户端始终只使用同一个产品族中的对象。

        增加新的产品族很方便,无须修改已有系统,符合“开闭原则”。

缺点:
​        增加新的产品等级结构麻烦,需要对原有系统进行较大的修改,甚至需要修改抽象层代码,这显然会带来较大的不便,违背了“开闭原则”。

4. 单例模式 

4.1 单例模式理论

        单例模式是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例的特殊类。通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源。如果希望在系统中某个类的对象只能存在一个,单例模式是最好的解决方案。

实现单例步骤:

1. 构造函数私有化
2. 增加静态私有的当前类的指针变量
3. 提供静态对外接口,可以让用户获得单例对象 

4.2 单例模式案例 

1.懒汉式(需要的时候才会创建)

2.饿汉式(直接创建,理解为main执行前,就创建好了)

#include <iostream>
using namespace std;


class A 
{
private:
	A() 
    {
		a = new A;
	}
public:
	static A* getInstance() 
    {
		return a;
	}
private:
	static A* a;
};


A* A::a = NULL;

//实现单例步骤
//1.构造函数私有化
//2.增加静态私有的当前类的指针变量
//3.提供静态对外接口,可以让用户获得单例对象

//单例 分为懒汉式 饿汉式

//1.懒汉式(需要的时候才会创建)
class Singleton_lazy 
{
private:
	Singleton_lazy() 
    { 
        cout << "我是懒汉构造!" << endl; 
    }
public:
	static Singleton_lazy* getInstance() 
    {
		if (pSingleton == NULL)
        {
			pSingleton = new Singleton_lazy;
		}
		return pSingleton;
	}
private:
	static Singleton_lazy* pSingleton;
};

//类外初始化
Singleton_lazy* Singleton_lazy::pSingleton = NULL;

//2.饿汉式
class Singleton_hungry 
{
private:
	Singleton_hungry() 
    { 
        cout << "我是饿汉构造!" << endl; 
    }
public:
	static Singleton_hungry* getInstance() 
    {
		return pSingleton;
	}
private:
	static Singleton_hungry* pSingleton;
};

//类外初始化
Singleton_hungry* Singleton_hungry::pSingleton = new Singleton_hungry;

void test01() 
{
	Singleton_lazy* p1 = Singleton_lazy::getInstance();
	Singleton_lazy* p2 = Singleton_lazy::getInstance();
	if (p1 == p2) 
    {
		cout << "两个指针指向同一块内存空间,是单例!" << endl;
	}
	else 
    {
		cout << "不是单例模式!" << endl;
	}

	Singleton_hungry* p3 = Singleton_hungry::getInstance();
	Singleton_hungry* p4 = Singleton_hungry::getInstance();
	if (p3 == p4) 
    {
		cout << "两个指针指向同一块内存空间,是单例!" << endl;
	}
	else 
    {
		cout << "不是单例模式!" << endl;
	}
}


int main() 
{
	//A a; // 或A*a=new A;  这两个会调用构造函数,要让不能构造函数
	//A::getInstance();

	test01();
	cout << "main函数开始执行!" << endl;

	return 0;
}

4.3  单例对象释放(不考虑内存释放问题)

1. 单例对象从头到尾就一份内存,没必要释放

2. 如果非要提供释放,可以如下,运行完自动释放:

#include <iostream>
using namespace std;

//实现单例步骤
//1.构造函数私有化
//2.增加静态私有的当前类的指针变量
//3.提供静态对外接口,可以让用户获得单例对象

//单例 分为懒汉式 饿汉式

//1.懒汉式(需要的时候才会创建)
class Singleton_lazy 
{
private:
	Singleton_lazy() 
	{ 
		cout << "我是懒汉构造!" << endl; 
	}
public:
	static Singleton_lazy* getInstance() 
	{
		if (pSingleton == NULL) 
		{
			pSingleton = new Singleton_lazy;
		}
		return pSingleton;
	}
#if 0 
	//这样释放不行,万一谁手贱释放,就直接没了,这个权限不能给
	static void freeSpace() 
	{
		if (pSingleton != NULL) 
		{
			delete pSingleton;
		}
	}
#endif
	//如果非要写,可以加个这个,运行完自动释放
	class Garbo 
	{
		Garbo() 
		{
			if (pSingleton != NULL) 
			{
				delete pSingleton;
			}
		}
	};

private:
	static Singleton_lazy* pSingleton;
	static Garbo garbo;
};

//类外初始化
Singleton_lazy* Singleton_lazy::pSingleton = NULL;

//2.饿汉式
class Singleton_hungry 
{
private:
	Singleton_hungry() 
	{ 
		cout << "我是饿汉构造!" << endl; 
	}
public:
	static Singleton_hungry* getInstance() 
	{
		return pSingleton;
	}
private:
	static Singleton_hungry* pSingleton;
};

//类外初始化
Singleton_hungry* Singleton_hungry::pSingleton = new Singleton_hungry;

void test07() 
{
	Singleton_lazy* p1 = Singleton_lazy::getInstance();
	Singleton_lazy* p2 = Singleton_lazy::getInstance();
	if (p1 == p2) 
	{
		cout << "两个指针指向同一块内存空间,是单例!" << endl;
	}
	else 
	{
		cout << "不是单例模式!" << endl;
	}

	Singleton_hungry* p3 = Singleton_hungry::getInstance();
	Singleton_hungry* p4 = Singleton_hungry::getInstance();
	if (p3 == p4) 
	{
		cout << "两个指针指向同一块内存空间,是单例!" << endl;
	}
	else 
	{
		cout << "不是单例模式!" << endl;
	}
}

#if 0
void test07_1() 
{
	Singleton_lazy::freeSpace();
}
#endif

int main() 
{

	test07();
	cout << "main函数开始执行!" << endl;
	return 0;
}

4.4 单例模式遇到多线程 

        懒汉式碰到多线程是线程不安全的,比如下班两个都判别pSingleton ==NULL,都创建一个,就不是单例了。

        使用懒汉式需要自己加锁并且解锁,步骤繁琐,推荐使用饿汉式。

4.5 单例模式的优缺点 

优点:

        单例模式提供了对唯一实例的受控访问

        节约系统资源。由于在系统内存中只存在一个对象。

缺点:

        扩展略难。单例模式中没有抽象层。

        单例类的职责过重。

4.6 单例模式适用场景 

1. 系统只需要一个实例对象,如系统要求提供一个唯一的序列号生成器或资源管理器(像windows的资源管理器),或者需要考虑资源消耗太大而只允许创建一个对象。

2. 客户调用类的单个实例只允许使用一个公共访问点,除了该公共访问点,不能通过其他途径访问该实例。

  • 37
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值