單例模式(反射+封裝注冊(cè)表)
2009年11月08日 星期日 下午 9:09
使用反射 在例9的帶注冊(cè)表的單例類(lèi)中,使用反射來(lái)實(shí)例化一個(gè)特殊的類(lèi)的對(duì)象。與例8相對(duì)的是通過(guò)這種實(shí)現(xiàn),Singleton.getInstance()方法不需要在每個(gè)被實(shí)現(xiàn)的子類(lèi)中重寫(xiě)了。
例9 使用反射實(shí)例化單例類(lèi)
- import java.util.HashMap;
- import org.apache.log4j.Logger;
-
- public class Singleton {
- private static HashMap map = new HashMap();
- private static Logger logger = Logger.getRootLogger();
-
- protected Singleton() {
-
- }
- public static synchronized Singleton getInstance(String classname) {
- Singleton singleton = (Singleton)map.get(classname);
-
- if(singleton != null) {
- logger.info("got singleton from map: " + singleton);
- return singleton;
- }
- try {
- singleton = (Singleton)Class.forName(classname).newInstance();
- }
- catch(ClassNotFoundException cnf) {
- logger.fatal("Couldn't find class " + classname);
- }
- catch(InstantiationException ie) {
- logger.fatal("Couldn't instantiate an object of type " + classname);
- }
- catch(IllegalAccessException ia) {
- logger.fatal("Couldn't access class " + classname);
- }
- map.put(classname, singleton);
- logger.info("created singleton: " + singleton);
-
- return singleton;
- }
- }
關(guān)于單例類(lèi)的注冊(cè)表應(yīng)該說(shuō)明的是:它們應(yīng)該被封裝在它們自己的類(lèi)中以便最大限度的進(jìn)行復(fù)用。
封裝注冊(cè)表 例10列出了一個(gè)單例注冊(cè)表類(lèi)。
例10 一個(gè)SingletonRegistry類(lèi)
- import java.util.HashMap;
- import org.apache.log4j.Logger;
-
- public class SingletonRegistry {
- public static SingletonRegistry REGISTRY = new SingletonRegistry();
-
- private static HashMap map = new HashMap();
- private static Logger logger = Logger.getRootLogger();
-
- protected SingletonRegistry() {
-
- }
- public static synchronized Object getInstance(String classname) {
- Object singleton = map.get(classname);
-
- if(singleton != null) {
- return singleton;
- }
- try {
- singleton = Class.forName(classname).newInstance();
- logger.info("created singleton: " + singleton);
- }
- catch(ClassNotFoundException cnf) {
- logger.fatal("Couldn't find class " + classname);
- }
- catch(InstantiationException ie) {
- logger.fatal("Couldn't instantiate an object of type " +
- classname);
- }
- catch(IllegalAccessException ia) {
- logger.fatal("Couldn't access class " + classname);
- }
- map.put(classname, singleton);
- return singleton;
- }
- }
注意我是把SingletonRegistry類(lèi)作為一個(gè)單例模式實(shí)現(xiàn)的。我也通用化了這個(gè)注冊(cè)表以便它能存儲(chǔ)和取回任何類(lèi)型的對(duì)象。例11顯示了的Singleton類(lèi)使用了這個(gè)注冊(cè)表。
例11 使用了一個(gè)封裝的注冊(cè)表的Singleton類(lèi)
- import java.util.HashMap;
- import org.apache.log4j.Logger;
-
- public class Singleton {
-
- protected Singleton() {
-
- }
- public static Singleton getInstance() {
- return (Singleton)SingletonRegistry.REGISTRY.getInstance(classname);
- }
- }
上面的Singleton類(lèi)使用那個(gè)注冊(cè)表的唯一實(shí)例通過(guò)類(lèi)名取得單例對(duì)象。
現(xiàn)在我們已經(jīng)知道如何實(shí)現(xiàn)線程安全的單例類(lèi)和如何使用一個(gè)注冊(cè)表去在運(yùn)行期指定單例類(lèi)名,接著讓我們考查一下如何安排類(lèi)載入器和處理序列化。
Classloaders
在許多情況下,使用多個(gè)類(lèi)載入器是很普通的--包括servlet容器--所以不管你在實(shí)現(xiàn)你的單例類(lèi)時(shí)是多么小心你都最終可以得到多個(gè)單例類(lèi)的實(shí)例。如果你想要確保你的單例類(lèi)只被同一個(gè)的類(lèi)載入器裝入,那你就必須自己指定這個(gè)類(lèi)載入器;例如:
- private static Class getClass(String classname)
- throws ClassNotFoundException {
- ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
-
- if(classLoader == null)
- classLoader = Singleton.class.getClassLoader();
-
- return (classLoader.loadClass(classname));
- }
- }
本站僅提供存儲(chǔ)服務(wù),所有內(nèi)容均由用戶(hù)發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)
點(diǎn)擊舉報(bào)。