定义
保证一个类仅有一个实例,并提供一个访问它的全局访问点。
我们让一个全局变量使得一个对象访问,但它不能防止你实例化多个对象。一个最好的办法就是,让类自身负责保存它的唯一实例。这个类可以保证没有其他实例可以被创建,并且它可以提供一个访问该实例的方法。
使用场景
整个项目需要一个共享访问点或共享数据
创一个对象需要耗费的资源过多,比如访问I/O或者数据库等资源
工具类对象。
结构图
完整示例
双重锁定
Class PHA.YX.Design.Singleton.Singleton Extends %RegisteredObject
{
Parameter instance As COSEXPRESSION = "##class(PHA.YX.Design.Singleton.Singleton).%New()";
Method %OnNew() As %Status [ Private, ServerOnly = 1 ]
{
Quit $$$OK
}
ClassMethod GetInstance() As Singleton
{
# dim %instance as PHA.YX.Design.Singleton.Singleton
i '$d(%instance) || (%instance = "") d
.l +^INSTANCE("INSTANCE"):5 d
..i '$d(%instance) || (%instance = "") d
...s %instance =class(PHA.YX.Design.Singleton.Singleton).%New()
l -^INSTANCE("INSTANCE")
q %instance
}
Method Call()
{
w "实例方法",!
}
}
为什么里面还有判断一次?
因为当instanc为null并且同时有个线程调用GetInstance()方法时,他们都将可以通过第一重instance=null的判断,然后由于lock机制,这两个变成则只有一个进入,另外一个在外排队等待,必须要其中的一个进入并出来后另外一个才能进入。而此时如果没有了第二重的instance是否为null判断,则第一个线程创建了实例,而第二个线程还是可以继续再创建新的实例,这就没有达到单例的目的。
调用
/// dclass(PHA.YX.Design.Program).Singleton()
ClassMethod Singleton()
{
#dim s1 as PHA.YX.Design.Singleton.Singleton
s s1 =class(PHA.YX.Design.Singleton.Singleton).GetInstance()
d s1.Call()
#dim s2 as PHA.YX.Design.Singleton.Singleton
s s2 =class(PHA.YX.Design.Singleton.Singleton).GetInstance()
d s2.Call()
if (s1 = s2) d
.w "两个对象是相同的实例"
}
DHC-APP>dclass(PHA.YX.Design.Program).Singleton()
实例方法
实例方法
两个对象是相同的实例
思考
其他语言,还有懒加载,饿汉模式。感兴趣的同学实现后可以发我一起参考下。