vlambda博客
学习文章列表

第十七章 Caché 设计模式 单例模式

第十七章 Caché 设计模式 单例模式

定义

保证一个类仅有一个实例,并提供一个访问它的全局访问点。

我们让一个全局变量使得一个对象访问,但它不能防止你实例化多个对象。一个最好的办法就是,让类自身负责保存它的唯一实例。这个类可以保证没有其他实例可以被创建,并且它可以提供一个访问该实例的方法。

使用场景

  • 整个项目需要一个共享访问点或共享数据

  • 创一个对象需要耗费的资源过多,比如访问I/O或者数据库等资源

  • 工具类对象。

结构图

完整示例

双重锁定

 
   
   
 
  1. Class PHA.YX.Design.Singleton.Singleton Extends %RegisteredObject

  2. {


  3. Parameter instance As COSEXPRESSION = "##class(PHA.YX.Design.Singleton.Singleton).%New()";


  4. Method %OnNew() As %Status [ Private, ServerOnly = 1 ]

  5. {

  6. Quit $$$OK

  7. }


  8. ClassMethod GetInstance() As Singleton

  9. {

  10. # dim %instance as PHA.YX.Design.Singleton.Singleton

  11. i '$d(%instance) || (%instance = "") d

  12. .l +^INSTANCE("INSTANCE"):5 d

  13. ..i '$d(%instance) || (%instance = "") d

  14. ...s %instance = ##class(PHA.YX.Design.Singleton.Singleton).%New()

  15. l -^INSTANCE("INSTANCE")

  16. q %instance

  17. }


  18. Method Call()

  19. {

  20. w "实例方法",!

  21. }


  22. }

为什么里面还有判断一次?

因为当instanc为null并且同时有个线程调用GetInstance()方法时,他们都将可以通过第一重instance=null的判断,然后由于lock机制,这两个变成则只有一个进入,另外一个在外排队等待,必须要其中的一个进入并出来后另外一个才能进入。而此时如果没有了第二重的instance是否为null判断,则第一个线程创建了实例,而第二个线程还是可以继续再创建新的实例,这就没有达到单例的目的。

调用

 
   
   
 
  1. /// d ##class(PHA.YX.Design.Program).Singleton()

  2. ClassMethod Singleton()

  3. {

  4. #dim s1 as PHA.YX.Design.Singleton.Singleton

  5. s s1 = ##class(PHA.YX.Design.Singleton.Singleton).GetInstance()

  6. d s1.Call()

  7. #dim s2 as PHA.YX.Design.Singleton.Singleton

  8. s s2 = ##class(PHA.YX.Design.Singleton.Singleton).GetInstance()

  9. d s2.Call()

  10. if (s1 = s2) d

  11. .w "个对象是相同的实例"

  12. }

 
   
   
 
  1. DHC-APP>d ##class(PHA.YX.Design.Program).Singleton()

  2. 实例方法

  3. 实例方法

  4. 个对象是相同的实例

思考

其他语言,还有懒加载,饿汉模式。感兴趣的同学实现后可以发我一起参考下。