Singleton 模式

单例模式(Singleton Pattern)是 C++/Java等语言中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

这种模式实现了类只能创建唯一实例的功能。并且由该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。

单例模式有三种经典的设计方案:

  1. 延时加载,也称为懒汉模式 ,需要的时候才创建对象,也就是下面方法一用到的
  2. 双重锁模式,需要的时候才创建对象,线程安全,懒汉模式升级版,也就是下面方法二用到的
  3. 贪婪加载,也称为饿汉模式 , 程序执行前就创建好对象,也就是下面方法三用到的
  4. 测试,简单测试。

  解法一:只适用于单线程环境

  由于要求只能生成一个实例,因此我们必须把构造函数设为私有函数以禁止他人创建实例。我们可以定义一个静态的实例,在需要的时候创建该实例。下面定义类型Singletonl就是基于这个思路的实现;

class SingleTon
{
public:
static SingleTon* getInstance()
{
if (instance== NULL)
{
instance= new SingleTon(); //堆区申请对象
} return instance;
}
private:
SingleTon(){} //隐藏构造函数接口
SingleTon(const SingleTon&);//拷贝构造函数
static SingleTon* instance; //静态指针可以不依赖对象调用,保存唯一实例地址
};
SingleTon* SingleTon::instance= NULL;

  上述代码在Singleton的静态属性Instance中,只有在instance为null的时候才创建一个实例以避免重复创建。同时我们把构造函数定义为私有函数,这样就能确保只创建一个实例。

  注意:只适用于单线程环境,在多线程中,如果一个现成在刚完成if (instance== NULL)这段代码时,还没有来的及创建对象时,另一个线程也执行到这个阶段,如此一来,程序就会出现问题,也不满足我们单例模式的目的。

  加同步锁:if (instance== NULL)这段语句前后加锁,实现每次只能有一个线程访问

  通常在解决线程安全类问题常用的方法就是加锁,限制对临界区的访问。下面这段代码虽然解决线程安全问题,但程序的效率却低了不少。

class SingleTon
{
public:
static SingleTon* getInstance()
{
lock(); //加锁 注:并未具体实现此函数,这里仅仅做演示
if (instance== NULL)
{
instance= new SingleTon(); //堆区申请对象
}
unlock(); //解锁 return instance;
}
private:
SingleTon(){} //隐藏构造函数接口
SingleTon(const SingleTon&);//拷贝构造函数
static SingleTon* instance; //静态指针可以不依赖对象调用,保存唯一实例地址
};
SingleTon* SingleTon::instance= NULL;

  加锁解锁机制: 线程A在执行到lock()时,如果发现无法加锁,即有别的线程B正在使用(加锁)此临界资源,当A发现可以进行加锁操作时,即B已经执行完成并进行了解锁操作,此时A再加锁(防止别的线程执行),当A线程执行完遇到代码unlock()时进行解锁操作。

  但是,还有一个问题,当唯一的一个实例已经被创建出来后,后面的线程执行到此段代码时,无需再构建实例,但lock() unlock()会被反复执行,程序进行了许多不必要的开销,降低了效率。

难道真的鱼(线程安全)与熊掌(效率)不可兼得乎?

方法二:双重锁模式

  双重锁模式:在加锁前再判断一次,是否需要执行以下内容。虽然叫双重锁,可并不是真正的加了两重锁哦,只是利用了一点技巧而已,可见,有时候只要多想一点点总会有办法的嘛。

class SingleTon
{
public:
static SingleTon* getInstance()
{
if (instance== NULL) //对象已经生成,无需加锁、解锁操作
{
lock();
if (instance== NULL)
{
instance= new SingleTon();
}
unlock();
}
return instance;
}
private:
SingleTon(){}
SingleTon(const SingleTon&);
static SingleTon* instance;
};

方法三:提前创建实例

由于静态变量在代码段存放,在程序刚被加载时就已经存在,因此可以利用这一特点重新设计程序,直接构造实例并让静态的指针instance接收。

class SingleTon
{
public:
static SingleTon* getInstance()
{
return instance;
}
private:
SingleTon(){}
SingleTon(const SingleTon&);
static SingleTon* instance;
};
SingleTon* SingleTon::instance= new SingleTon();

测试:

Person类,属性有mname mage msex

class Person
{
public:
static Person* getInstance(char* name, int age, bool sex)
{
if (pm == NULL)
{
pm = new Person(name, age, sex);
}
return pm;
}
~Person()
{
delete[] mname;
mname = NULL;
}
private:
Person(char* name, int age, bool sex)
{
mname = new char[strlen(name) + 1]();
strcpy_s(mname, strlen(name) + 1, name);
mage = age;
msex = sex;
}
char* mname;
int mage;
bool msex;
static Person* pm;
};
Person* Person::pm = NULL; int main()
{
char name[] = "zhangsan";
Person* p1 = Person::getInstance(name, 25, true);
Person* p2 = Person::getInstance(name, 25, true);
Person* p3 = Person::getInstance(name, 25, true);
Person* p4 = Person::getInstance(name, 25, true);
return 0;
}

运行结果:



可以看到运行结果中,不论getInstance方法调用多少次,始终只有一个实例产生。

欢迎大家评论留言,互相学习。有不对的地方请帮忙指出,谢谢。

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

  1. c++设计模式系列----单例模式(Singleton模式

    单例模式是为了解决唯一对象实例问题而提出来的,许多时候整个系统只需要拥有一个全局对象,这样有利于我们协调系统整体的行为.比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单 ...

  2. Java设计模式(2)单态模式(Singleton模式)

    定义:Singleton模式主要作用是保证在Java应用程序中,一个类Class只有一个实例存在. 在很多操作中,比如建立目录 数据库连接都需要这样的单线程操作. 还有,singleton能够被状态化 ...

  3. Java中的GOF23(23中设计模式)--------- 单例模式(Singleton)

    Java中的GOF23(23中设计模式)--------- 单例模式(Singleton) 在Java这这门语言里面,它的优点在于它本身的可移植性上面,而要做到可移植的话,本身就需要一个中介作为翻译工 ...

  4. 设计模式学习之单例模式(Singleton,创建型模式)(4)

    假如程序中有一个Person类,我的需求就是需要在整个应用程序中只能new一个Person,而且这个Person实例在应用程序中进行共享,那么我们该如何实现呢? 第一步: 新建一个Person类,类中 ...

  5. 设计模式---对象性能模式之单例模式(Singleton)

    前提:“对象性能”模式  面向对象很好的解决了“抽象”的问题,但是必不可免地要付出一定的代价.对于通常情况来讲,面向对象的成本大都可以忽略不计.但是某些情况,面向对象所带来的成本必须谨慎处理. 前面是 ...

  6. 设计模式-单例模式(Singleton Pattren)(饿汉模式和懒汉模式)

    单例模式(Singleton Pattren):确保一个类在整个应用中只有一个实例,并提供一个全局访问点. 实现要点: 1. 私有化构造方法 2. 类的实例在类初始化的时候创建 3. 提供一个类方法, ...

  7. 第一式、单例模式-Singleton模式(创建型)

    一.简介 单例模式主要用的作用是用于保证程序运行中某个类只有一个实例,并提供一个全局入口点.单例模式(Singleton)为GOF阐述的标准24种设计模式中最简单的一个.但随着时间推移,GOF所阐述的 ...

  8. 创建型模式(一) 单例模式(Singleton)

    一.动机(Motivation) 在软件系统中,经常有这样一些特殊的类,必须保证它们在系统中只存在一个实例,才能确保它们的逻辑正确性.以及良好的效率. 如何绕过常规的构造器,提供一种机制来保证一个类只 ...

  9. 6、单例模式 Singleton模式 只有一个实例 创建型模式

    1.了解Singleton模式 程序在运行时,通常都会生成很多实例.例如,表示字符串的java . lang . string类的实例与字符串是- -对- -的关系,所以当有1000个字符串的时候,会 ...

随机推荐

  1. _IO_2_1_stdin_ 任意写及对 _IO_2_1_stdout_ 任意读的补充

    之前写过一篇 IO_FILE--leak 任意读,但是在学习的时候偷懒了,没有深入去看,这次碰到 winmt 师傅出的题,就傻眼了,故再写一篇博客来记录一下. 例题 ctfshow Incomplet ...

  2. Qt:QList、QStringList

    QList 0.说明 QList<T> 一个QList是存储相同类型一组数据的列表. QStringList是从QList<String>继承而来,并添加了一些好用的方法,如j ...

  3. 自己创建bmp图像

    随便找一张图片,右键选择打开方式为画图,再在画图中保存为bmp即可 如果要保存为png文件,也可以这样

  4. elementUI日期选择器 el-date-picker根据所选日期选择禁用

    我遇到这样一个场景,需要动态渲染时间表单,但是后端传过来的数据, 这个时候就不能预先找到想要限制的date了,因为连date本身,也是根据后端传来的数据生成的. 如图: 代码如下: //templat ...

  5. 流程控制、if、elif、else,whilie、break、continue的使用

    今日内容 流程控制理论 if判断 while循环 流程控制概念 流程控制就是控制事物的执行流程 执行流程的分类 顺序结构 从上往下依次执行,代码运行流程图如下 分支结构 根据某些条件判断做出不同的运行 ...

  6. Flutter ChartSpace:通过跨端 Canvas 实现图表库

    基于Flutter 的图形语法库,通过跨端 Canvas ,将基于 Javascript 的图形语法库 ChartSpace 扩展至 Flutter 端 作者:字节跳动终端技术--胡珀 背景 数据平台 ...

  7. SpringBoot连接Redis (Sentinel模式&Cluster模式)

    一.引入pom <dependency> <groupId>org.springframework.boot</groupId> <artifactId> ...

  8. GoLang设计模式21 - 装饰模式

    装饰器模式是一种结构型设计模式.通过装饰器模式可以为一个对象添加额外的功能而不需对其作出调整. 还是通过具体的案例来了解装饰器模式:假设我们开了一家披萨店,现在店里主营两款披萨: 素食狂披萨(Vegg ...

  9. thinkphp5 composer安装phpexcel插件及使用

    1: 首先composer加载phpexcel插件 composer require phpoffice/phpexcel 2: 页面引入 use PHPExcel_IOFactory; use PH ...

  10. php 23种设计模型 - 门面模式(外观模式)

    外观模式(Facade) 外观模式(Facade Pattern)隐藏系统的复杂性,并向客户端提供了一个客户端可以访问系统的接口.这种类型的设计模式属于结构型模式,它向现有的系统添加一个接口,来隐藏系 ...