在做項目時,遇到需要創建DAO、Service等類的實例的時候,想到用工廠方法來運作,而簡單工廠方法又有明顯的缺點: ①由於工廠類集中了所有實例的創建邏輯,違反了高內聚責任分配原則,將全部創建邏輯集中到了一個工廠類中; ②它所能創建的類只能是事先考慮到的,如果需要添加新的類,則就需要改變工廠類了。 ...
在做項目時,遇到需要創建DAO、Service等類的實例的時候,想到用工廠方法來運作,而簡單工廠方法又有明顯的缺點:
①由於工廠類集中了所有實例的創建邏輯,違反了高內聚責任分配原則,將全部創建邏輯集中到了一個工廠類中;
②它所能創建的類只能是事先考慮到的,如果需要添加新的類,則就需要改變工廠類了。
③當系統中的具體產品類不斷增多時候,可能會出現要求工廠類根據不同條件創建不同實例的需求.這種對條件的判斷和對具體產品類型的判斷交錯在一起,很難避免模塊功能的蔓延,對系統的維護和擴展非常不利;
所以考慮使用配置文件+java反射機制來動態創建類的實例,具體思路如下:
通過讀取本地的.properties文件來獲取我們需要實例化的類,然後通過反射來生成對象,這樣當你把發佈出去的時候,使用者只用更改配置文件就可以讓工廠去實例化自己後來才寫的實現類。
使用該方法也就解決了上述問題中:當實體增多時,工廠類需要重新編寫的問題了。
先來介紹一下.properties文件
在Java中,使用一種以.properties為擴展名的文本文件作為資源文件,該類型的文件的內容格式為類似:
#註釋
key=value
#為註釋,ResourceBundle處理時會自動忽略,對於key=value,應用較多的是資料庫的配置文件,比如這樣:
#資料庫配置信息 DRIVER=com.mysql.jdbc.Driver URL=jdbc:mysql://localhost:3306/cns user=test password=test
而在這裡,需要記錄的就是 key( 你的實體類名)=value(該類的具體路徑),這裡的value用於反射中forName(String className)來載入該類。
java.util.ResourceBundle
這個類提供軟體國際化的捷徑。通過此類,可以使您所編寫的程式可以: 輕鬆地本地化或翻譯成不同的語言 一次處理多個語言環境 以後可以輕鬆地進行修改,支持更多的語言環境 說的簡單點,這個類的作用就是讀取資源屬性文件(properties),然後根據.properties文件的名稱信息(本地化信息),匹配當前系統的國別語言信息(也可以程式指定),然後獲取相應的properties文件的內容。 使用這個類,要註意的一點是,這個properties文件的名字是有規範的:一般的命名規範是: 自定義名_語言代碼_國別代碼.properties, 如果是預設的,直接寫為:自定義名.properties 比如: myres_en_US.propertiesmyres_zh_CN.properties myres.properties 當在中文操作系統下,如果myres_zh_CN.properties、myres.properties兩個文件都存在,則優先會使用myres_zh_CN.properties,當myres_zh_CN.properties不存在時候,會使用預設的myres.properties。 沒有提供語言和地區的資源文件是系統預設的資源文件。 資源文件都必須是ISO-8859-1編碼,因此,對於所有非西方語系的處理,都必須先將之轉換為Java Unicode Escape格式。轉換方法是通過JDK自帶的工具native2ascii. 好吧,上邊這部分是我粘過來的,目前我還用不到上述的國際化功能,所以我只關註於用它來讀取預設文件。
直接上代碼:
import java.util.ResourceBundle; public class BeanFactory { // 載入配置文件 private static ResourceBundle bundle; static { bundle = ResourceBundle.getBundle("instance"); } /** * 根據指定的key,讀取配置文件獲取類的全路徑; 創建對象 * */ public static <T> T getInstance(String key,Class<T> clazz) { String className = bundle.getString(key); try { return (T) Class.forName(className).newInstance(); } catch (Exception e) { throw new RuntimeException(e); } } }
其中
bundle = ResourceBundle.getBundle("instance");
instance是我的資源文件名稱,全名是instance.properties
然後再通過指定的key來獲取具體的類名
bundle.getString(key)
之後利用反射來創建類的實例。