单例模式是最简单的设计模式,就让我像玩简单的游戏一样写下去吧。

v1: 简单模式

和这个版本有过一面之缘,但不敢苟同。

class Singleton
{
private:
Singleton() {}
public:
static Singleton * getIns()
{
static Singleton * ins = new Singleton();
return ins;
}
};

问题:何时析构不明确;最重要的是调用多次getIns函数会产生多个static Singleton指针,指向每次都调用都new出来的实例。

v2: 一般模式

典型写法

class Singleton
{
private:
Singleton() {}
static Singleton * ins;
public:
static Singleton * getIns()
{
if(!ins) ins = new Singleton();
return ins;
}
}; static Singleton * Singleton::ins = NULL;

问题:仍然未考虑析构问题;对象可能被复制出多个副本。

Java中由于允许在调用构造函数之前先初始化变量,因此有这样一种写法:

public class Singleton
{
private Singleton() {}
public static Singleton ins = new Singleton();
public static Singleton * getIns()
{
return ins;
}
};

简洁明了,也是蛮OK啦,析构也省了,并且由于初始化这个语句是JVM做的,因此人工的同步也省了(不带这么欺负C++程序员的 = =)。

v3: 加强模式

加入私有的复制构造函数以防出现单例对象的副本;加入一个内部静态类,整个程序结束后,静态类随着其他静态变量消亡,此时调用析构函数将ins析构。

class Singleton
{
private:
Singleton() {}
Singleton(const Singleton & s) {}
Singleton & operator = (const Singleton & s) {} static Singleton * ins; public:
static Singleton * getIns()
{
if(!ins) ins = new Singleton();
return ins;
} class CGarbo // 内部类
{
public:
~CGarbo()
{
if(Singleton::ins)
delete Singleton::ins;
}
};
static CGarbo Garbo;
};
static Singleton * Singleton::ins = NULL;

问题:不是线程安全的。

v4: hard模式

    static Singleton * getIns()
{
pthread_mutex_lock( &mutex_lock );
if(!ins) ins = new Singleton();
pthread_mutex_unlock( &mutex_lock );
return ins;
}

问题:不管ins是不是空,都要加锁,代价高

v5: 再接再励模式

当ins不为空时,反正getIns函数不管读写,直接返回就好了,读写锁尽可交给使用者。

    static Singleton * getIns()
{
if(!ins)
{
pthread_mutex_lock( &mutex_lock );
if(!ins) ins = new Singleton();
pthread_mutex_unlock( &mutex_lock );
}
return ins;
}

自然,getIns加了锁,析构的时候也同样要加锁。继续学习,再接再励!

v6:虎牢关模式

实际上,只有在第一次创建单例的时候需要获得锁。既然可以用一个内部类的静态实例实现单例对象的自动释放,何不把该静态对象的初始化也交由该内部类来维护呢:

class Singleton
{
public:
Singleton() {}
Singleton(const Singleton &s) {}
Singleton & operator = (const Singleton & s) {} public:
static Singleton *getInstance()
{
return GC::pinstance;
} class GC
{
public:
static Singleton *pinstance;
GC()
{
pinstance = new Singleton();
}
~GC()
{
if( pinstance )
{
delete pinstance;
}
}
};
GC gc;
};
Singleton * Singleton::GC::pinstance = nullptr;

注意,此处将Singleton的构造函数改成了public。因为在系统中没有Singleton实例时,Singleton的static成员也是不会被创建的,因此需将构造函数改成public,在使用词单例类时,先声明一个Singleton对象。

如此,进程启动,在初始化阶段先初始化了不属于任何实例对象的Singleton的各个静态成员,顺便new好了我们所需的单例对象指针,这一过程不是线程来干的,因此这样的new操作是安全的。而在程序运行结束时,gc自动析构,将单例对象释放掉,这一过程也不是线程来干的,因此也是线程安全的,そして,至于pthread_mutex? WTF。而由于不管存在多少Singleton对象,每次getInstance必然获取到的是同一个instance。

v7:  终极模式

等等,所有“非你莫属”的类,都要改写成单例模式么?NONONO,能不能只写一个单例模式,所有需要限制实例个数的类都可复用之?这样,当系统中需要多个不同类的单例对象时,可以节约许多代码。当然可以做到,用模板就可以:

template <typename T>
class Singleton
{
public:
Singleton() {}
Singleton(const Singleton &s) {}
Singleton & operator = (const Singleton & s) {} public:
static Singleton *getInstance()
{
return GC::pinstance;
} class GC
{
public:
static T *pinstance;
GC()
{
pinstance = new T();
}
~GC()
{
if( pinstance )
{
delete pinstance; pinstance = nullptr;
}
}
};
GC gc;
};
template <typename T>
T * Singleton<T>::GC::pinstance = nullptr;

当然很显然,使用了模板的话,就需要一些约定俗成的东西了,例如在实例化的时候需要一个不带参数的构造函数T()。在使用时,对于用Singleton<T>::getInstance函数返回得到的对象指针,就是非你莫属的同一个指针了:

int main()
{
Singleton<int> sgton;
int *p1 = Singleton<int>::getInstance();
int *p2 = Singleton<int>::getInstance();
int *p3 = Singleton<int>::getInstance();
printf("%x\n", p1);
cout <<(p1==p2)<<endl;
cout <<(p2==p3)<<endl;
return ;
}

以上测试代码的结果为0x........ 1 1.

ちょっと待って!!还不够!以上代码还是有问题的!注意此时我们要限制对象个数的不是Singleton对象了,而是T对象。因此设计目标是:不管申明多少个Singleton对象都是合法的,而不管申明了多少个Singleton对象,调用getInstance都只会得到一个T对象。

再看上面的代码,如果用户手贱声明了多个Singleton对象会怎么样呢?对于一个Singleton类的static对象,只要程序中的Singleton对象个数大于0,就会维护唯一的一个static变量。可是上面的static变量是个指针!!这就意味着每次构造Singleton对象时,都会调用GC构造函数中的new运算符,将一个新的T对象的地址赋值给pinstance!!那旧的那个呢?WHO CARES!!这样一来轻则导致内存泄漏,重则导致系统运行出错——如果先申明了一个Singleton对象,用着用着中间又声明出来一个Singleton对象,那么pinstance指向的内容就被偷换了!

说了这么多,为了防止发生这样的事情,只需要在new的时候和delete的时候稍稍注意一下即可,为了防止new两次,加上if语句;为了防止delete两次(当然在其他场合也一样),养成delete之后就将指针赋为空,以免指针在其他地方被误引用。

template <typename T>
class Singleton
{
public:
Singleton() {}
Singleton(const Singleton &s) {}
Singleton & operator = (const Singleton & s) {} public:
static T *getInstance()
{
return GC::pinstance;
} class GC
{
public:
static T *pinstance;
GC()
{
if( !pinstance ) pinstance = new T();
}
~GC()
{
if( pinstance )
{
delete pinstance; pinstance = nullptr;
}
}
};
static GC gc;
};
template <typename T>
T * Singleton<T>::GC::pinstance = nullptr;

这样一来,只要在起子线程之前声明一个Singleton对象,那么不管后面是main线程还是子线程声明了其他的Singleton对象,T对象都不会出现第二个。

反思:

1  为什么Java用内部类维护单例?Java这么做是为了在不写synchronized关键字(减少锁的代价)的情况下,采用延时加载的方式实现单例。怎么做到不用锁而做到线程安全呢?在Java中,内部类都只会加载一次,因此内部类的初始化过程有且仅有一次,而这个过程是JVM干的,不受用户多线程的影响。为什么是延时加载呢?只有getInstance()用到了内部类的时候,才会加载内部类。

2  为什么C++写法和Java延时加载且线程安全的单例写法不一样呢?C++是不会加载内部类的,只有当有对象需要实例化时,才调用其构造函数进行初始化,同时也初始化内部类的static变量。如果像Java一样只是定义一个内部类而不去实例化它,是不行的。因此按照上面的写法,使用该单例模板类必须先定义一个Singleton对象,这时候由于内部类作为Singleton的成员,会先被构造,“顺便地”就正确地new出了一个T对象,pinstance指针是static变量,一个类最多只有一个,所以不会有两个pinstance,那么只要再内部类的构造中防止new出两个T对象,就可以保证进程中有唯一的pinstance指向唯一的T。

to be continue

【设计模式】C++单例模式的几种写法——Java自动加载内部类对象,C++怎么破?的更多相关文章

  1. 设计模式之单例模式的几种写法——java

    对于设计模式的使用场景和好处,之前有介绍一篇,今天主要是单例模式的编写方式,直接看代码吧 单例模式之饿汉模式,不会懒加载.线程安全 /** * @Author wangtao * @Descripti ...

  2. Java设计模式之单例模式(七种写法)

    Java设计模式之单例模式(七种写法) 第一种,懒汉式,lazy初始化,线程不安全,多线程中无法工作: public class Singleton { private static Singleto ...

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

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

  4. Android设计模式之单例模式的七种写法

    一 单例模式介绍及它的使用场景 单例模式是应用最广的模式,也是我最先知道的一种设计模式.在深入了解单例模式之前.每当遇到如:getInstance()这样的创建实例的代码时,我都会把它当做一种单例模式 ...

  5. jquery文档加载几种写法,图片加载写法

    jquery文档加载写法: $(function(){ }) ; //个人最常使用方式 $(document).ready(function(){ }); //调用文档对象下的ready方法传入一个函 ...

  6. java单例模式的几种写法比较

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

  7. Java 单例模式的七种写法

    Java 单例模式的七种写法 第一种(懒汉,线程不安全) public class Singleton { private static Singleton instance; private Sin ...

  8. Java单例模式的6种写法

    在Java中,单例有很多种写法,面试时,手写代码环节,除了写算法题,有时候也会让手写单例模式,这里记录一下单例的几种写法和优缺点. 初级写法 懒汉式 饿汉式 双锁检验 内部类 枚举式 1.初级写法 p ...

  9. 单例模式:Java单例模式的几种写法及它们的优缺点

    总结下Java单例模式的几种写法: 1. 饿汉式 public class Singleton { private static Singleton instance = new Singleton( ...

随机推荐

  1. 51 Nod 1678 lyk与gcd

    1678 lyk与gcd 基准时间限制:2 秒 空间限制:131072 KB 分值: 80 难度:5级算法题 这天,lyk又和gcd杠上了.它拥有一个n个数的数列,它想实现两种操作. 1:将  ai  ...

  2. react dva 表单校验

    import React,{ Component } from 'react'; import { connect } from 'dva'; import { WhiteSpace,NavBar , ...

  3. ckeditor编辑的使用方法

    一.下载安装Ckeditor,并将其整合到项目中 1.什么是CKeditor?为什么要使用它? 我们在做门户网站或者公文系统时,客户经常要求在录入时能够更改字体样式.大小.颜色并具备插入图片的功能.而 ...

  4. hdu 1848 Fibonacci again and again 组合游戏 SG函数

    题目链接 题意 三堆石子,分别为\(m,n,p\)个,两人依次取石子,每次只能在一堆当中取,并且取的个数只能是斐波那契数.最后没石子可取的人为负.问先手会赢还是会输? 思路 直接按定义计算\(SG\) ...

  5. Bluedroid之GKI

    1. 概述 GKI以库libbt-brcm_gki.a的形式提供给Bluedroid使用 该层是一个适配层,适配了OS相关的进程.内存相关的管理,还可以用于线程间传递消息 主要通过变量gki_cb(结 ...

  6. Android Bluetooth抓包

    1. 前提 这里介绍一种在Android上捕获蓝牙数据包的方法 首先你要有一部Android手机:然后你的Android系统版本要在4.4及以上 我没有做过Android开发,不清楚开发者们是如何抓蓝 ...

  7. 19年的桌面KDE的风雨和陪伴,没有什么能够割舍

    概述 KDE是史上功能最强大的桌面环境之一:开源且可自由使用.19年前,1996年10月14日,德国程序员 Matthias Ettrich 开始了这个美观的桌面环境的开发.KDE 提供了用户界面以及 ...

  8. 一张图让你学会Python【转】

    转自:http://blog.csdn.net/qq_30845505/article/details/51588423 有编程基础的人一看就可以了解 Python 的用法了.真正的 30 分钟上手. ...

  9. macro expand error

    cat test_macro.c #define TEST_MACRO(b) chip->##b int main(void) { TEST_MACRO(yyy) return 0; } gcc ...

  10. (5)ASP.NET HttpResponse 类

    HttpResponse 类用来封装来自 ASP.NET 操作的 HTTP 响应信息 https://msdn.microsoft.com/zh-cn/library/system.web.httpr ...