天天看点

Design Pattern: Singleton 模式

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

singleton的英文意义是独身,也就是只有一个人,应用在物件导向语言上,通常翻译作单例:单一个实例(instance)。 

很多时候,您会需要singleton模式,例如印表机管理,您希望程式中只能有一个print spooler,以避免两个列印动作同时输入至印表机中;例如资料库管理,因为建立连接(connection)物件会耗用资源,您希望程式中只能有一个 连接物件,所有其它的程式都透过这个物件来连接资料库,以避免连接物件的重复开启造成资源的耗用;例如系统程式属性档的读取,您使用单一个物件来读取属性 内容,而程式的其它部份都向这个物件要求属性资料,而不是自行读取属性资料。 

以印表机设计为例,有的设计人员会采取全域变数的方式来建立实例,并在程式中随机取用这个实例,java虽然不支援全域变数,但透过将物件包装在一个类别之中,也有人会采用这样的写法:

无论全域变数或是以上的例子,都无法保证只产生唯一个实例,您也许会注意不犯这个错误,但与您共同工作的伙伴也许会直觉的使用建构方法来产生一个 printspooler实例。 

singleton模式可以保证一个类别只有一个实例,并提供一个访问(visit)这个实例的方法。 

一个singleton实作即为java中的java.lang.runtime类别,每个java程式执行时都有一个唯一的runtime物件,可以透过它提供的静态方法getruntime()方法来取得这个物件,例如:

runtime runtime = runtime.getruntime();

取得runtime物件之后,您可以透过它进行一些外部命令的执行、进行垃圾处理等等指令,您可以开启runtime.java类别,开头的几行是这样写的:

有几个实作上面结构的方法,可以在第一次需要实例时再建立物件,也就是采用所谓的lazy initialization:

上面的实作适用于单执行绪的程式,在多执行绪的程式下,以下的写法在多个执行绪的竞争资源下,将仍有可能产生两个以上的实例,例如下面的情况:

thread1: if(instance == null) // true 

thread2: if(instance == null) // true 

thread1: instance = new singleton(); // 产生一个实例 

thread2: instance = new singleton(); // 又产生一个实例 

thread1: return instance; // 回传一个实例 

thread2: return instance; // 又回传一个实例

在多执行绪的环境下,为了避免资源同时竞争而导致如上产生多个实例的情况,加上同步(synchronized)机制

不过这种简单的写法不适合用于像伺服器这种服务很多执行绪的程式上,同步机制会造成相当的效能低落,为了顾及singleton、lazy initialization与效能问题,因而有了double-check locking的模式:

java中runtime类别的作法就简单多了,它舍弃了lazy initialization,如果您的实例初始化不是很久的话,可以用这种方式: