最简单的设计模式学习:Singleton模式

开发 后端
本文介绍了学习设计模式中最简单的模式:Singleton模式。Singleton的用处自然是保证一个类只有一个唯一的实例。

学习设计模式,自然从最简单的模式入手,而最简单的模式便是Singleton。所以***篇就来所以说Singleton模式。看完GOF和Design patterns in Java的书,感觉Singleton虽然简单,但是想写出一个好的Singleton也不是一上来就能写出来的。

Singleton模式的用处自然是保证一个类只有一个***的实例。在建模中涉及到的只能有一个对象,例如Struts中的Action类就是一例。除此之外,Singleton还使得该对象只有一个全局访问点。这就是SIngleton的作用。

说得比较抽象,我们来看一个简单Singleton的C++和Java的代码:

C++ Singleton模式:

类定义:

  1. class Singleton    
  2. {  
  3. public:  
  4.     static Singleton * Instance();  
  5.     ~Singleton();  
  6.  
  7. private:  
  8.     Singleton();  
  9.  
  10.     static Singleton * instance;  
  11. }; 

方法实现:

  1. Singleton * Singleton::instance = 0;  
  2.  
  3. Singleton::Singleton()  
  4. {  
  5.  
  6. }  
  7.  
  8. Singleton::~Singleton()  
  9. {  
  10.  
  11. }  
  12.  
  13. Singleton * Singleton::Instance()  
  14. {  
  15.     if (instance == 0) {  
  16.         instance = new Singleton();  
  17.     }  
  18.  
  19.     return instance;  

Java Singleton模式:

  1. public class Singleton {  
  2.       
  3.     private static Singleton instance;  
  4.       
  5.     public static Singleton getInstance() {  
  6.         if (instance == null)  
  7.             instance = new Singleton();  
  8.           
  9.         return instance;  
  10.     }  
  11.       
  12.     /** *//** Creates a new instance of Singleton */  
  13.     private Singleton() {  
  14.     }  

通过上面的例子可以看出,Singleton的实现并不难,只要将构造函数访问域设为私有,然后添加一个静态引用和一个获得该应用的静态方法即可。其实在C++中定义一个全局静态变量也可以达到这个效果,但是像Java这样的语言就是能使用Singleton了。

上面的程序有一个问题,就是只能运行在单线程的环境下。为此我在C++上作了个实验。首先#include 。在SIngleton::Instance()函数中增加一个Sleep(1000),程序如下:

  1. Singleton * Singleton::Instance()  
  2. {  
  3.     if (instance == 0) {  
  4.         Sleep(1000);  
  5.         instance = new Singleton();  
  6.     }  
  7.  
  8.     return instance;  

然后在主函数中创建两个线程,程序如下:

  1. static Singleton * s1 = 0, * s2 = 0;  
  2.  
  3. DWORD WINAPI ThreadProc1(PVOID)  
  4. {  
  5.     s1 = Singleton::Instance();  
  6.  
  7.     return 0;  
  8. }  
  9.  
  10. DWORD WINAPI ThreadProc2(PVOID)  
  11. {  
  12.     s2 = Singleton::Instance();  
  13.  
  14.     return 0;  
  15. }  
  16.  
  17. int main(int argc, char* argv[])  
  18. {  
  19.     DWORD threadID1;  
  20.     DWORD threadID2;  
  21.  
  22.     CreateThread(NULL, 0, ThreadProc1, NULL, 0, &threadID1);  
  23.     CreateThread(NULL, 0, ThreadProc2, NULL, 0, &threadID2);  
  24.  
  25.     Sleep(10000);  
  26.  
  27.     std::cout << s1 << " " << s2;  
  28.  
  29.     return 0;  

这样修改后在运行程序,打印出来的s1和s2地址就不是同一个地址了。结果如下:
0372D68 00372E68Press any key to continue

可见当在多线程环境下使用这个Singleton就会出现创建不止一个实力的情况,所以我们需要给Singleton加锁。请看下面的代码。

C++ Singleton模式:

  1. class Singleton    
  2. {  
  3. public:  
  4.     static Singleton * Instance();  
  5.     virtual ~Singleton();  
  6.  
  7. private:  
  8.     Singleton();  
  9.  
  10.     static CMutex mutex;  
  11.     static Singleton * instance;  
  12. }; 
  1. Singleton * Singleton::instance = 0;  
  2. CMutex Singleton::mutex;  
  3.  
  4. Singleton::Singleton()  
  5. {  
  6.  
  7. }  
  8.  
  9. Singleton::~Singleton()  
  10. {  
  11.  
  12. }  
  13.  
  14. Singleton * Singleton::Instance()  
  15. {  
  16.     mutex.Lock();  
  17.  
  18.     if (instance == 0) {  
  19.         Sleep(1000);  
  20.         instance = new Singleton();  
  21.     }  
  22.  
  23.     mutex.Unlock();  
  24.  
  25.     return instance;  

此外需要#include < afxmt.h>,,并且在项目设置中要设置动态链接MFC库。

Java Singleton模式:

  1. public class Singleton {  
  2.       
  3.     private static Singleton instance;  
  4.     private static Object lock = Singleton.class;  
  5.       
  6.     public static Singleton getInstance() {  
  7.         synchronized (lock) {  
  8.             if (instance == null)  
  9.                 instance = new Singleton();  
  10.               
  11.             return instance;  
  12.         }  
  13.     }  
  14.       
  15.     /** *//** Creates a new instance of Singleton */  
  16.     private Singleton() {  
  17.     }  

运用加锁就可以解决在多线程环境下使用Singleton模式所带来的问题了。

【编辑推荐】

  1. 详解Java Singleton模式的好处
  2. Java设计模式与应用浅谈
  3. 设计模式在EJB中的应用
  4. 探讨JSF框架中使用的设计模式
  5. J2EE设计模式之Template
责任编辑:周立方 来源: 百度百科
相关推荐

2009-09-02 16:23:27

C# Singleto

2009-08-31 15:48:02

C# Singleto

2012-08-22 10:10:25

单态单态设计设计模式

2011-07-18 16:51:51

Cocoa 单态 模式

2009-07-08 17:25:05

Java Single

2021-03-06 22:50:58

设计模式抽象

2009-08-31 16:12:02

C#使用Singlet

2012-07-10 02:01:53

设计模式命令模式

2023-07-31 12:27:30

单例设计模式

2011-11-17 16:03:05

Java工厂模式Clojure

2009-08-25 18:04:30

C#实现Singlet

2015-11-03 09:43:43

avascript设计模式工厂模式

2021-04-19 21:25:48

设计模式到元

2022-03-15 09:00:00

机器学习软件开发MLOps

2024-04-18 00:22:10

设计模式接口代码

2021-02-01 10:01:58

设计模式 Java单例模式

2009-08-31 16:01:31

C#和设计模式

2010-07-21 15:30:40

SQL Server

2012-08-30 09:07:33

设计模式

2009-08-12 13:22:44

Singleton模式
点赞
收藏

51CTO技术栈公众号