单例模式是应用最多的一种设计模式,它要求系统中每个类有且只能有一个实例对象。

主要优点:

1、提供了对唯一实例的受控访问。

2、由于在系统内存中只存在一个对象,因此可以节约系统资源,对于一些需要频繁创建和销毁的对象单例模式无疑可以提高系统的性能。

3、允许可变数目的实例。

主要缺点:

1、由于单利模式中没有抽象层,因此单例类的扩展有很大的困难。

2、单例类的职责过重,在一定程度上违背了“单一职责原则”。

3、滥用单例将带来一些负面问题,如为了节省资源将数据库连接池对象设计为的单例类,可能会导致共享连接池对象的程序过多而出现连接池溢出;如果实例化的对象长时间不被利用,系统会认为是垃圾而被回收,这将导致对象状态的丢失。

C++设计一个单例模式的方法如下:

1 构造函数声明为私有;   这样就保证了不能随意构造一个对象

2 将拷贝构造函数与等号运算符声明为私有,并不提供他们的实现; 即禁止对象被拷贝。

3 在类中声明一个静态的全局访问接口;

4 声明一个静态的私有实例化指针;

  1. class Singleton
  2. {
  3. public:
  4. //全局访问接口
  5. static Singleton *GetInstance()
  6. {
  7. if( instance_ == NULL )
  8. {
  9. instance_ = new Singleton;
  10. }
  11. return instance_;
  12. }
  13. ~Singleton()
  14. {
  15. cout << "~Singleton"<< endl;
  16. }
  17. private:
  18. Singleton(const Singleton& other);
  19. Singleton & operator=(const Singleton & other);
  20. Singleton()
  21. {
  22. cout << "Singleton"<<endl;
  23. }
  24. static Singleton *instance_; //引用性声明
  25. };
  26. Singleton * Singleton::instance_; //定义性声明
  27. int main(void)
  28. {
  29. Singleton *s1 = Singleton::GetInstance();
  30. Singleton *s2 = Singleton::GetInstance();  //s2的地址等于s1,即指向同一对象
  31. //Singleton s3(*s1); //既然是单例模式,是不允许被拷贝的。编译会出错
  32. return 0;
  33. }

上面就是单例类模式的C++实现,但是上述代码还有一个缺陷:单例类中申请的一些资源没有被释放,如instance_指向的空间没有被回收。一共有两种解决方式:

第一种解决方式:

  1. class Singleton
  2. {
  3. ...
  4. //提供一个回收接口,在应用中显示调用回收资源
  5. static void Free()
  6. {
  7. delete instance_;
  8. }
  9. ....
  10. };
  11. int main()
  12. {
  13. ...
  14. Singleton::Free(); //要显示调用进行资源回收
  15. }

这种方式虽然能实现功能,但是不太方便,每次都要手动回收资源,这是它的缺点。

第二种解决方式:

  1. class Singleton
  2. {
  3. ...
  4. public:
  5. class Garbo //资源回收机制
  6. {
  7. public:
  8. ~Garbo()
  9. {
  10. if( Singleton::instance_ != NULL )
  11. {
  12. delete instance_;
  13. }
  14. }
  15. };
  16. ...
  17. private:
  18. ...
  19. static Garbo garbo_;  //引用性声明
  20. };
  21. Singleton::Garbo Singleton::garbo_;//定义性声明

这种方式提供的处理方式显然要比第一种方式来的要便捷,因为它依靠内部提供的Garbo嵌套类来提供服务,当Singleton类生命周期结束时,Garbo的类对象garbo_也要销毁,它将调用析构函数,而在析构函数中又自动地释放了Singleton单例类申请的一些资源,这种实现就比较智能化。不需要手动释放资源。这是它的优势。

下面提供另一种实现C++单例类模式的方法:

  1. class Singleton
  2. {
  3. public:
  4. static Singleton& GetInstance()
  5. {
  6. static Singleton instance_;                   return instance_;
  7. }
  8. ~Singleton()
  9. {
  10. cout << "~Singleton"<<endl;
  11. }
  12. private:
  13. Singleton()
  14. {
  15. cout << "Singleton "<<endl;
  16. }
  17. Singleton(const Singleton &other);
  18. Singleton & operator=(const Singleton &other);
  19. };

这种实现方式利用了static修改函数内部的变量,当第一次调用GetInstance函数时,系统将构造一个Singleton对象,在后续再次调用这个函数时,系统不再执行这个语句,也就不会再构造Singleton类对象了,而直接返回instance_本身。另外,当Singleton类对象生命周期结束时,对象会自动调用析构函数销毁,这里也就不存在申请资源之类的了,需要注意的是GetInstance()函数返回的是引用,这样就不会调用拷贝构造函数了,使用时也应该声明Singleton的引用,如下:

  1. int main()
  2. {
  3. Singleton &s1 = Singleton::GetInstance();
  4. Singleton &s2 = Singleton::GetInstance(); //s1与s2是同一对象的引用
  5. return 0;
  6. }

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

  1. C# 的static与单例模式

    C# 的static与单例模式 static是静态对象,在类被第一次使用,或者第一次被实例化时执行 /// <summary> /// 线程安全的单件模式 /// </summary ...

  2. static 与单例模式、auto_ptr与单例模式、const 用法小结、mutable修饰符

    一.static 与单例模式 单例模式也就是简单的一种设计模式,它需要: 保证一个类只有一个实例,并提供一个全局访问点 禁止拷贝  C++ Code  1 2 3 4 5 6 7 8 9 10 11 ...

  3. (1) 类构造块,this(),static,单例模式串讲

    类构造块 在类只用一对大括号包含的内容,构造所有的对象时都会执行的内容,如果某个类有好几个够赞函数,公共部分抽取出来,放到构造块中. clas Boy { ... { syso("哭...& ...

  4. 第六节 静态的(static)和单例模式

    main函数 主函数是一个特殊的函数,作为程序的入口,可以被jvm(虚拟器)调用 主函数的定义 public 表示该函数的访问权限是最大的. static 代表主函数随着类的加载就已经存在了. voi ...

  5. 单例模式(一)static、final和单例模式

    static 那天我朋友问了我个问题,static和单例模式有什么区别,所以我觉得static可以讲一下 他的问题是,把对象弄成static是不是就不变了 显然,这是还没弄清楚引用和对象的区别 其实存 ...

  6. java中static关键字的作用

    java中static关键字主要有两种作用: 第一:为某特定数据类型或对象分配单一的存储空间,而与创建对象的个数无关. 第二,实现某个方法或属性与类而不是对象关联在一起 简单来说,在Java语言中,s ...

  7. 【设计模式】单例设计模式的N中Java实现方法

    转载请注明出处:http://blog.csdn.net/ns_code/article/details/17359719 特点 单例模式的特点: 1.只能有一个实例: 2.必须自己创建自己的一个实例 ...

  8. 探测器 C++ Singleton(辛格尔顿)

    一.静态模式不是单一的情况下, 刚开始学习的人可能误, 误以为所有的成员变量和成员方法用于 static , 就是单例模式了: class Singleton { public:     /* sta ...

  9. java开发常用技术

    基础部分 1. 线程和进程的区别 线程三个基本状态:就绪.执行.阻塞 线程五个基本操作:创建.就绪.运行.阻塞.终止 进程四种形式:主从式.会话式.消息或邮箱机制.共享存储区方式 进程是具有一定功能的 ...

随机推荐

  1. 在html中使用javascript显示本地图片的

    <html> <head> <script type="text/javascript"> function getFullPath(obj){ ...

  2. WCF心跳判断服务端及客户端是否掉线并实现重连接

    WCF心跳判断服务端及客户端是否掉线并实现重连接 本篇文章将通过一个实例实现对WCF中针对服务端以及客户端是否掉线进行判断:若掉线时服务器或客户端又在线时将实现自动重连:将通过WCF的双工知识以及相应 ...

  3. WPF 自定义 MessageBox (相对完善版)

    WPF 自定义 MessageBox (相对完善版)     基于WPF的自定义 MessageBox. 众所周知WPF界面美观.大多数WPF元素都可以简单的修改其样式,从而达到程序的风格统一.可是当 ...

  4. DIV+CSS命名参考

    用了一段CSS布局设计网页,发现自己的命名有点混乱,完全按照自己的想法命名,虽然没什么影响,有不给别人看源文件,但是工作室有时候和团队合作完成项目的时候,就遇到麻烦了,要修改一个地方相当的费事.所以还 ...

  5. TOGAF架构内容框架之架构制品(下)

    TOGAF架构内容框架之架构制品(下) 4.2.31 数据生命周期图(Data Lifecycle Diagram) 数据生命周期图是在业务流程的约束之下对业务数据在其整个生命周期(从概念阶段到最终退 ...

  6. ACE 容器之三 ACE_Unbounded_Queue的使用

    以下代码演示了如何在ACE_Unbounded_Queue这个队列容器中存储具体的数据元素和数据元素的指针. // ACEqueue.cpp : Defines the entry point for ...

  7. JBPM之JPdl小叙

    JPdl:jbpm流程定义文件,下面是一个简单的流程定义文件: <?xml version="1.0" encoding="UTF-8"?> < ...

  8. Dom4j下载及使用Dom4j读写XML简介

    sitinspring(如坐春风)原创,转载请注明作者及出处. 要使用dom4j读写XML文档,需要先下载dom4j包,dom4j官方网站在 http://www.dom4j.org/目前最新dom4 ...

  9. android调用js

    1.android中利用webview调用网页上的js代码. Android 中可以通过webview来实现和js的交互,在程序中调用js代码,只需要将webview控件的支持js的属性设置为true ...

  10. jvm内存JVM学习笔记-引用(Reference)机制

    在写这篇文章之前,xxx已经写过了几篇关于改jvm内存主题的文章,想要了解的朋友可以去翻一下之前的文章 如果你还不了解JVM的基本概念和内存划分,请阅读JVM学习笔记-基础知识和JVM学习笔记-内存处 ...