单例模式(Singleton)

--本文内容部分引自《大话设计模式 Chapter21》

一.概念:保证一个类仅有一个实例,并提供一个访问它的全局访问点。

  通常我们可以让一个全局变量使一个对象被访问,但它不能阻止你实例化多个对象,一个最好的办法就是,让类自身负责保存它的唯一实例。这个类可以保证没有其他实例可以被创建,并且它可以提供一个访问该实例的方法。

二.结构

单例模式因为Singleton类封装它的唯一实例,这样它可以严格的控制客户怎样访问它及何时访问它,简单来说就是对唯一实例的受控访问。

单例类有状态,虽然实例唯一,却可以有子类来继承。

特点:

1、构造函数私有,防止外部实例化。这是第一点要求

2、唯一实例句柄声明为static,这就是所谓唯一

三.单线程模式:

     class CSingleton
{
public:
static CSingleton *GetInstance()
{
if (NULL != m_Instance)
{
m_Instace = new CSingleton();
}
} private:
CSingleton(); private:
static CSingleton * m_Instance;
};
//此处初始化
CSingleton* CSingleton::m_Instace = NULL;

这里相应有一些可以提升的地方,比如实际实例类型不定的情况下,这里可以改为使用模板,类型待编译时刻再定;仅仅私有构造,还有其他方式会导致实例化,如拷贝构造。

     template <typename T>
class CSingleton
{
public:
static T *GetInstance()
{
if (NULL != m_Instance)
{
m_Instace = new T;
}
} private:
CSingleton();
CSingleton(const T&);
void operator=(const T&); private:
static T * m_Instance;
}; //此处初始化
template <typename T>
T* CSingleton<T>::m_m_Instance= NULL;

四、多线程模式:

首先,提到多线程,那就马上要考虑同步的问题了,说到底就是锁,具体锁怎么实现这里不赘叙。

         static T *GetInstance()
{
m_mutex.Lock();
if (NULL != m_Instance)
{
m_Instance = new T;
}
m_mutex.UnLock();
}

还有需要考虑一个会遇到的多线程情况下遇到的问题,那就是上面写法的锁的位置,当句柄为NULL时,同时有两个线程调用到了GetInstance入口,拿到时间片的线程进入if里面new出对象,结束后另一个线程进入后,依旧会再new一次。这种可能是存在的。

为了应对这种低概率但又不能无视的情况,大神们给出了方案--双重锁定(Double-Check Locking).

GetInstance函数修改如下:

         static T *GetInstance()
{
if (NULL != m_Instance)
{
m_mutex.Lock();
if (NULL != m_Instance)
{
m_Instance = new T;
}
m_mutex.UnLock();
}
}

五、总结

1.以上所有代码实现都是所谓的懒汉式单例类,因为是在第一次被引用时才会将自己实例化。

相对有饿汉模式,就是在以上代码初始化的地方直接写成

     template <typename T>
T* CSingleton::m_Instance = new T;

这种静态初始化的方式称为饿汉式单例类

2.目前实际使用中,只在项目工程中的日志记录、配置文件操作、内存池等全局唯一实例上,使用起来还是很方便的。

单例模式-C++的更多相关文章

  1. C++实现线程安全的单例模式

    在某些应用环境下面,一个类只允许有一个实例,这就是著名的单例模式.单例模式分为懒汉模式,跟饿汉模式两种. 首先给出饿汉模式的实现 template <class T> class sing ...

  2. 23种设计模式--单例模式-Singleton

    一.单例模式的介绍 单例模式简单说就是掌握系统的至高点,在程序中只实例化一次,这样就是单例模式,在系统比如说你是该系统的登录的第多少人,还有数据库的连接池等地方会使用,单例模式是最简单,最常用的模式之 ...

  3. angular2系列教程(十)两种启动方法、两个路由服务、引用类型和单例模式的妙用

    今天我们要讲的是ng2的路由系统. 例子

  4. java设计模式之--单例模式

    前言:最近看完<java多线程编程核心技术>一书后,对第六章的单例模式和多线程这章颇有兴趣,我知道我看完书还是记不住多少的,写篇博客记录自己所学的只是还是很有必要的,学习贵在坚持. 单例模 ...

  5. 设计模式C#合集--单例模式

    单例模式 代码: 第一种: private static Singleton singleton = null; private Singleton() { } public static Singl ...

  6. 设计模式之单例模式(Singleton)

    设计模式之单例模式(Singleton) 设计模式是前辈的一些经验总结之后的精髓,学习设计模式可以针对不同的问题给出更加优雅的解答 单例模式可分为俩种:懒汉模式和饿汉模式.俩种模式分别有不同的优势和缺 ...

  7. GOF23设计模式之单例模式

    ·核心作用: -保证一个类只有一个实例,并且提供一个访问该实例的全局访问点. ·常见应用场景: -Windows的Task Manager(任务管理器)就是很典型的单例模式 -Windows的Recy ...

  8. GJM : C#设计模式(1)——单例模式

    感谢您的阅读.喜欢的.有用的就请大哥大嫂们高抬贵手"推荐一下"吧!你的精神支持是博主强大的写作动力以及转载收藏动力.欢迎转载! 版权声明:本文原创发表于 [请点击连接前往] ,未经 ...

  9. PHP设计模式(四)单例模式(Singleton For PHP)

    今天讲单例设计模式,这种设计模式和工厂模式一样,用的非常非常多,同时单例模式比较容易的一种设计模式. 一.什么是单例设计模式 单例模式,也叫单子模式,是一种常用的软件设计模式.在应用这个模式时,单例对 ...

  10. java设计模式之单例模式(几种写法及比较)

    概念: Java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍三种:懒汉式单例.饿汉式单例.登记式单例. 单例模式有以下特点: 1.单例类只能有一个实例. 2.单例类必须自己创建 ...

随机推荐

  1. NHibernate简单使用介绍

    1.在数据库中新建表格,并插入记录,SQL如下: USE WFC_DB GO create table Students ( Id ,) not null, Name ), Age int, Scor ...

  2. POJ 1780 Code(有向图的欧拉通路)

    输入n(1<=n<=6),输出长度为10^n + n -1 的字符串答案. 其中,字符串以每n个为一组,使得所有组都互不相同,且输出的字符串要求字典序最小. 显然a[01...(n-1)] ...

  3. .NET 多线程

    多线程 在一个程序中,这些独立运行的程序片断叫作“线程”(Thread),利用它编程的概念就叫作“多线程处理”.多线程处理一个常见的例子就是用户界面.利用线程,用户可按下一个按钮,然后程序会立即作出响 ...

  4. md5 (c语言)

    /** * \file md5.h * * \brief MD5 message digest algorithm (hash function) * * Copyright (C) 2006-201 ...

  5. hibernate中一对多关系中的inverse,cascade属性

    举例说明: 一对多关系的两张表:boy.girl(一个男孩可以多个女朋友) boy表结构 Field   Type        ------  -----------  name    varcha ...

  6. October 22nd Week 43rd Saturday, 2016

    Beware the barrenness of a busy life. 当心忙碌的生活荒芜了人生. Recently I was busy in debugging an equipment, I ...

  7. 无废话ExtJs 入门教程二十一[继承:Extend]

    无废话ExtJs 入门教程二十一[继承:Extend] extjs技术交流,欢迎加群(201926085) 在开发中,我们在使用视图组件时,经常要设置宽度,高度,标题等属性.而这些属性可以通过“继承” ...

  8. UVALive5031 Graph and Queries(Treap)

    反向操作,先求出最终状态,再反向操作. 然后就是Treap 的合并,求第K大值. #include<cstdio> #include<iostream> #include< ...

  9. hdu分类 Dynamic Programming(这是一场漫长的旅途)

    下面是difficulty 1的题 1003   Max Sum 最长递增子序列.非常经典,最棒的解法是在线算法O(n)的复杂度. 贴的呢,是用dp做的代码. 先是一个高亮的dp递推式,然后找到最大处 ...

  10. MongoDB基本使用

    成功启动MongoDB后,再打开一个命令行窗口输入mongo,就可以进行数据库的一些操作. 输入help可以看到基本操作命令: show dbs:显示数据库列表 show collections:显示 ...