Linux C++ 060-设计模式之单例模式
本节关键字:Linux、C++、设计模式、单例模式
相关库函数:
概念
单例模式是设计模式中最简单的形式之一。这一模式的目的是使得类的一个对象成为系统中的唯一实例。要实现这一点,可以从客户端对其进行实例化开始。因此需要用一种只允许生成对象类的唯一实例的机制,“阻止”所有想要生成对象的访问。使用工厂方法来限制实例化过程。这个方法应该是静态方法(类方法),因为让类的实例去生成另一个唯一实例毫无意义。
动机
对于系统中的某些类来说,只有一个实例很重要,例如,一个系统中可以存在多个打印任务,但是只能有一个正在工作的任务;一个系统只能有一个窗口管理器或文件系统;一个系统只能有一个计时工具或ID(序号)生成器。如在Windows中就只能打开一个任务管理器。如果不使用机制对窗口对象进行唯一化,将弹出多个窗口,如果这些窗口显示的内容完全一致,则是重复对象,浪费内存资源;如果这些窗口显示的内容不一致,则意味着在某一瞬间系统有多个状态,与实际不符,也会给用户带来误解,不知道哪一个才是真实的状态。因此有时确保系统中某个对象的唯一性即一个类只能有一个实例非常重要。
如何保证一个类只有一个实例并且这个实例易于被访问呢?定义一个全局变量可以确保对象随时都可以被访问,但不能防止我们实例化多个对象。一个更好的解决办法是让类自身负责保存它的唯一实例。这个类可以保证没有其他实例被创建,并且它可以提供一个访问该实例的方法。这就是单例模式的模式动机。
构建方式
懒汉方式。指全局的单例实例在第一次被使用时构建。
饿汉方式。指全局的单例实例在类装载时构建。
要点
显然单例模式的要点有三个:一是某个类只能有一个实例;二是它必须自行创建这个实例;三是它必须自行向整个系统提供这个实例。
从具体实现角度来说,就是以下三点:一是单例模式的类只提供私有的构造函数,二是类定义中含有一个该类的静态私有对象,三是该类提供了一个静态的公有的函数用于创建或获取它本身的静态私有对象。
在下面的对象图中,有一个"单例对象",而"客户甲"、“客户乙” 和"客户丙"是单例对象的三个客户对象。可以看到,所有的客户对象共享一个单例对象。而且从单例对象到自身的连接线可以看出,单例对象持有对自己的引用。
一些资源管理器常常设计成单例模式。
在计算机系统中,需要管理的资源包括软件外部资源,譬如每台计算机可以有若干个打印机,但只能有一个Printer Spooler, 以避免两个打印作业同时输出到打印机中。每台计算机可以有若干传真卡,但是只应该有一个软件负责管理传真卡,以避免出现两份传真作业同时传到传真卡中的情况。每台计算机可以有若干通信端口,系统应当集中管理这些通信端口,以避免一个通信端口同时被两个请求同时调用。
需要管理的资源包括软件内部资源,譬如,大多数的软件都有一个(甚至多个)属性(properties)文件存放系统配置。这样的系统应当由一个对象来管理一个属性文件。
需要管理的软件内部资源也包括譬如负责记录网站来访人数的部件,记录软件系统内部事件、出错信息的部件,或是对系统的表现进行检查的部件等。这些部件都必须集中管理,不可整出多头。
这些资源管理器构件必须只有一个实例,这是其一;它们必须自行初始化,这是其二;允许整个系统访问自己这是其三。因此,它们都满足单例模式的条件,是单例模式的应用。
优缺点
优点
1、实例控制:单例模式会阻止其他对象实例化其自己的单例对象的副本,从而确保所有对象都访问唯一实例。
2、灵活性:因为类控制了实例化过程,所以类可以灵活更改实例化过程。
缺点
1、开销:虽然数量很少,但如果每次对象请求引用时都要检查是否存在类的实例,将仍然需要一些开销。可以通过使用静态初始化解决此问题。
2、可能的开发混淆:使用单例对象(尤其在类库中定义的对象)时,开发人员必须记住自己不能使用new关键字实例化对象。因为可能无法访问库源代码,因此应用程序开发人员可能会意外发现自己无法直接实例化此类。
3、对象生存期:不能解决删除单个对象的问题。在提供内存管理的语言中(例如基于.NET Framework的语言),只有单例类能够导致实例被取消分配,因为它包含对该实例的私有引用。在某些语言中(如 C++),其他类可以删除对象实例,但这样会导致单例类中出现悬浮引用。
代码示例
// 说明:在初始化类中的静态成员(该类的唯一一个对象)时,创建对象。
// 总结:
// 类中的构造函数是private类型,该类的实例对象是静态指针类型
// 访问该类的全局访问点是类中public类型的静态函数,函数的返回值是该类的指针类型
// 在类的外部需要对类中定义的静态成员(该类的静态指针类型对象)进行初始化,初始化形式:类型名 类名 作用域运算符 变量名 等于 初始值;(懒汉模式:初始值为指向实例对象的指针)
// 客户端只能通过类的静态成员函数实例化一个对象
//
// 单例模式 - 饿汉单例模式
// 饿汉式单例模式,顾名思义就是程序启动时就实例了该现象,并没有推迟到第一次使用该对象再进行实例化;如果运行过程中没有使用到,该实例对象就被浪费掉了
// 饿汉单例模式中,单例对象定义成了一个static静态对象,它是在程序启动时,main函数运行之前就初始化好的,因此不存在线程安全问题,可以放心在多线程环境中使用
class CSingleton_1 {
public:
static CSingleton_1* getInstance() {
return &single;
}
private:
static CSingleton_1 single;
CSingleton_1() {}
~CSingleton_1() {}
CSingleton_1(const CSingleton_1&) {}
};
CSingleton_1 CSingleton_1::single;
// 单例模式 - 懒汉单例模式
// 懒汉单例模式,顾名思义,就是对象的实例化,延迟到第一次使用它的时候
// 懒汉单例模式中,getInstance()是个不可重入函数,也就是它在多线程环境中执行,会出现竞态条件问题,因此需要考虑加锁
class CSingleton_2 {
public:
static CSingleton_2* getInstance() {
// 获取互斥锁
if (single == NULL) {
single = new CSingleton_2();
}
return single;
}
private:
static CSingleton_2* single;
CSingleton_2() {}
~CSingleton_2() {}
CSingleton_2(const CSingleton_2&) {}
// 注意如果没有下面的释放资源类,在main函数中将手动释放资源
class CRelease {
public:
~CRelease() {
if (single)
delete single;
}
};
static CRelease release;
// 定义线程间的互斥锁
static pthread_mutex_t mutex;
};
CSingleton_2* CSingleton_2::single = NULL;
CSingleton_2::CRelease CSingleton_2::release;
// 互斥锁初始化
pthread_mutex_t CSingleton_2::mutex = PTHREAD_MUTEX_INITIALIZER;
标签:060,CSingleton,C++,对象,single,实例,模式,单例,设计模式
From: https://blog.csdn.net/qq_45157350/article/details/135869815