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

主要优点:

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. cegui-0.8.2编译过程详解

    cegui 编译过程详解(cegui-0.8.2) cegui配置整了好长时间了,在一位大牛帮助下终于搞定了,网上的教程大多是老版本的,cegui-0.8.2版的配置寥寥无几,现在总结一下,献给正在纠 ...

  2. 记一次有趣的互联网事件及console.log~

    寂寞的中国互联网又一次瘫痪了. 说是顶级域的根挂了,不知道是黑客还是某个实习生干挂的. 反正到现在还没有人来解释这件事. 先普及一下,为什么顶级域的根挂了全中国都挂了. 那是因为dns解析的特点是递归 ...

  3. MVC视图与控制器分离简单描述

    一,控制器 CheckIndexAreaRegistration.cs public class CheckIndexAreaRegistration : AreaRegistration { pub ...

  4. Make Things Move -- Javascript html5版(二)实现最基本的Sprite类和stage管理对象

    现在这几篇写的都是比较基础的东西,有过相应开发经验的朋友可直接忽略啦. 计算机模拟的动画都是由很多静态的一连串影像(sprite)在一定帧率(fps)内逐帧播放出来的. 对于js来说,我们可以用提供的 ...

  5. springMVC3学习(十二)--文件上传优化CommonsMultipartResolver

    基于上一篇文件上传发现效率很慢,我们应该对它进行优化  使用springMVC对文件上传的解析器 来处理文件上传的时候需要在spring的applicationContext里面加上springMVC ...

  6. 映射请求到Servlet

    Servlet规范当中对映射请求的描述: 在收到客户端请求时,web 容器确定转发到哪一个Web应用.选择的Web应用必须具有最长的上下文路径匹配请求URL的开始.当映射到Servlet时,URL匹配 ...

  7. 关于前端JS模块加载器实现的一些细节

    最近工作需要,实现一个特定环境的模块加载方案,实现过程中有一些技术细节不解,便参考 了一些项目的api设计约定与实现,记录下来备忘. 本文不探讨为什么实现模块化,以及模块化相关的规范,直接考虑一些技术 ...

  8. wp加载本地HTML(附带图片,CSS,JS)

    wp加载本地HTML(附带图片,CSS,JS) Windows Phone:Load Local HTML with Img,Css,Js by 唐小崇 http://www.cnblogs.com/ ...

  9. 关于SqlDataAdapter的使用

    原文传送门 如果使用SqlDataAdapter来查询数据返回给DataSet或者DataTable时需要注意以下几点:1.如果SqlDataAdapter的SelectCommand的连接并没有打开 ...

  10. 第一个windows 小游戏 贪吃蛇

    最近用dx尝试做了一个小的贪吃蛇游戏,代码放到github上面:https://github.com/nightwolf-chen/MyFreakout 说一下自己实现的过程: 首先,我把蛇这个抽象成 ...