板橋里人 http://www.jdon.com 2002/05/07
模式實(shí)戰書(shū)籍《Java實(shí)用系統開(kāi)發(fā)指南》
單態(tài)定義:
Singleton模式主要作用是保證在Java應用程序中,一個(gè)類(lèi)Class只有一個(gè)實(shí)例存在。
在很多操作中,比如建立目錄 數據庫連接都需要這樣的單線(xiàn)程操作。
還有, singleton能夠被狀態(tài)化; 這樣,多個(gè)單態(tài)類(lèi)在一起就可以作為一個(gè)狀態(tài)倉庫一樣向外提供服務(wù),比如,你要論壇中的帖子計數器,每次瀏覽一次需要計數,單態(tài)類(lèi)能否保持住這個(gè)計數,并且能synchronize的安全自動(dòng)加1,如果你要把這個(gè)數字永久保存到數據庫,你可以在不修改單態(tài)接口的情況下方便的做到。
另外方面,Singleton也能夠被無(wú)狀態(tài)化。提供工具性質(zhì)的功能,
Singleton模式就為我們提供了這樣實(shí)現的可能。使用Singleton的好處還在于可以節省內存,因為它限制了實(shí)例的個(gè)數,有利于Java垃圾回收(garbage collection)。
我們常??吹焦S(chǎng)模式中類(lèi)裝入器(class loader)中也用Singleton模式實(shí)現的,因為被裝入的類(lèi)實(shí)際也屬于資源。
如何使用?
一般Singleton模式通常有幾種形式:
| public class Singleton { private Singleton(){} //在自己內部定義自己一個(gè)實(shí)例,是不是很奇怪? private static Singleton instance = new Singleton(); //這里提供了一個(gè)供外部訪(fǎng)問(wèn)本class的靜態(tài)方法,可以直接訪(fǎng)問(wèn)
|
第二種形式:
| public class Singleton { private static Singleton instance = null; }
|
使用Singleton.getInstance()可以訪(fǎng)問(wèn)單態(tài)類(lèi)。
上面第二中形式是lazy initialization,也就是說(shuō)第一次調用時(shí)初始Singleton,以后就不用再生成了。
注意到lazy initialization形式中的synchronized,這個(gè)synchronized很重要,如果沒(méi)有synchronized,那么使用getInstance()是有可能得到多個(gè)Singleton實(shí)例。關(guān)于lazy initialization的Singleton有很多涉及double-checked locking (DCL)的討論,有興趣者進(jìn)一步研究。
一般認為第一種形式要更加安全些。
使用Singleton注意事項:
有時(shí)在某些情況下,使用Singleton并不能達到Singleton的目的,如有多個(gè)Singleton對象同時(shí)被不同的類(lèi)裝入器裝載;在EJB這樣的分布式系統中使用也要注意這種情況,因為EJB是跨服務(wù)器,跨JVM的。
我們以SUN公司的寵物店源碼(Pet Store 1.3.1)的ServiceLocator為例稍微分析一下:
在Pet Store中ServiceLocator有兩種,一個(gè)是EJB目錄下;一個(gè)是WEB目錄下,我們檢查這兩個(gè)ServiceLocator會(huì )發(fā)現內容差不多,都是提供EJB的查詢(xún)定位服務(wù),可是為什么要分開(kāi)呢?仔細研究對這兩種ServiceLocator才發(fā)現區別:在WEB中的ServiceLocator的采取Singleton模式,ServiceLocator屬于資源定位,理所當然應該使用Singleton模式。但是在EJB中,Singleton模式已經(jīng)失去作用,所以ServiceLocator才分成兩種,一種面向WEB服務(wù)的,一種是面向EJB服務(wù)的。
Singleton模式看起來(lái)簡(jiǎn)單,使用方法也很方便,但是真正用好,是非常不容易,需要對Java的類(lèi) 線(xiàn)程 內存等概念有相當的了解。
總之:如果你的應用基于容器,那么Singleton模式少用或者不用,可以使用相關(guān)替代技術(shù)。
聯(lián)系客服