java如何實時動態獲取properties文件的內容
實時動態獲取properties文件的內容
用“ClassLoader.getResourceAsStream”讀取properties文件時會發現修改瞭.properties後,即使重新執行,讀入的仍為修改前的參數。
此問題的原因在於ClassLoader.getResourceAsStream讀入後,會將.properties保存在緩存中,重新執行時會從緩存中讀取,而不是再次讀取.properties文件。
動態讀取的代碼
import java.util.Properties; /** * 實時動態獲取properties文件的值 * @author Administrator * */ public class demo01 { /** * 根據配置變量實時獲取配置文件中的值 * @param key 配置名 * @param filePath 配置文件路徑名,例如:test.properties * @return 配置值 */ public static String getCurrentPropertiesValue(String key,String filePath){ String value=""; Properties p = new Properties(); try { //非實時動態獲取 //p.load(new InputStreamReader(this.class.getClassLoader().getResourceAsStream(filePath), "UTF-8")); //下面為動態獲取 String path = Thread.currentThread().getContextClassLoader().getResource("").getPath(); InputStream is = new FileInputStream(path +File.separator+ filePath); p.load(is); value=p.getProperty(key); } catch (UnsupportedEncodingException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } return value; } }
讀取配置文件 Properties VS ResourceBundle
在java開發中, 對於一些常用的配置性的信息, 通常會采用存放在屬性文件中, 因為修改配置文件無須重新編譯jar包. 對於屬性文件,通常可以使用Properties和ResourceBundle 兩個類來解析. 需要註意的是, 默認情況下java工程中的*.properties文件編碼格式是ISO-8859-1, Properties和ResourceBundle也是按照ISO-8859-1格式來解析屬性文件中字符串的. 所以對於解析包含中文的熟悉文件時,需要額外註意.
- ResourceBundle: 通常用於解析國際化資源屬性文件, 會根據本地環境自動選擇對應的國際化資源.
- Properties: 用來解析普通屬性文件
1. 常用API
1.1 Properties 常用API
Properties 繼承Hashtable<Object,Object>類.
方法簽名 | 方法描述 |
---|---|
public String getProperty(String key | 獲取屬性文件中的Key, 如果key不存在返回Null |
public String getProperty(String key, String defaultValue) | 獲取屬性文件中key對象的value, 如果key不存在則返回默認值defaultValue |
public Object get(String key) | 父類HashTable中的方法, 返回值類型為Object |
1.2 ResourceBundle 常用API
ResourceBundle 是一個接口, 默認使用PropertyResourceBundle來解析屬性文件.
方法簽名 | 方法描述 |
---|---|
public Locale getLocale() | 獲取本地國際化環境 |
public Enumeration getKeys() | 獲取屬性文件中所有key |
public final String getString(String key) | 獲取屬性文件中key對應的value, 返回值為String, 如果不存在, 則拋出異常 |
public final Object getObject(String key) | 獲取屬性文件中key對應的value, 返回值為Object, 如果不存在, 則拋出異常 |
2. Properties 解析屬性文件
默認使用ISO-8859-1 解析配置文件中的字符串, 所以會導致中文亂碼.
2.1 解析純英文配置文件
// 默認編碼(ISO-8859-1)讀取屬性文件, 中文亂碼 @Test public void test_properties_en() throws IOException{ // 屬性文件位置, 相對路徑為src/main/resources 或 src/test/resources, 不能添加classpath:/前綴 String propertyFileName = "jdbc.properties"; // 獲取字節流 InputStream is = getClass().getClassLoader().getResourceAsStream(propertyFileName); // 創建屬性文件, 並加載文件內容 Properties properties = new Properties(); properties.load(is); String username = properties.getProperty("jdbc.username"); String password = properties.getProperty("jdbc.password"); System.out.println("username:" + username + ", password:" + password); }
2.2 解析含中文配置文件
默認使用ISO-8859-1, 采用InputStreamReader轉換為UTF8字符流.
// 指定讀取文件編碼方式,支持讀取中文 @Test public void test_properties_zh() throws IOException{ // 屬性文件位置, 相對路徑為src/main/resources 或 src/test/resources, 不能添加classpath:/前綴 String propertyFileName = "jdbc.properties"; // 獲取字節流 InputStream is = getClass().getClassLoader().getResourceAsStream(propertyFileName); // 轉換為UTF-8格式字符流 InputStreamReader isr = new InputStreamReader(is, "UTF-8"); // 創建屬性文件, 並加載文件內容 Properties properties = new Properties(); properties.load(isr); String username = properties.getProperty("jdbc.username"); String password = properties.getProperty("jdbc.password"); System.out.println("username:" + username + ", password:" + password); }
3. ResourceBundle 解析屬性文件
3.1 解析純英文配置文件
@Test public void testRb_en() { // 資源配置文件,無須寫文件後綴名, 默認尋找properties文件 String bundleName = "jdbc"; // 設置本地默認環境為英文環境 Locale.setDefault(Locale.ENGLISH); // 指定加載 ResourceBundle rb = ResourceBundle.getBundle(bundleName); String username = rb.getString("jdbc.username"); String password = rb.getString("jdbc.password"); System.out.println("username:" + username + ", password:" + password); }
3.2 解析含中文配置文件
// 處理中文 @Test public void testRb_zh() { // 資源配置文件,無須寫文件後綴名, 默認尋找properties文件 String bundleName = "jdbc"; // 根據本地默認環境加載資源配置文件 ResourceBundle rb = ResourceBundle.getBundle(bundleName); String username = iso2Utf8(rb.getString("jdbc.username")); String password = iso2Utf8(rb.getString("jdbc.password")); System.out.println("username:" + username + ", password:" + password); } /** * @Description iso編碼格式字符串轉換為UTF8格式 * @param str iso 編碼字符串 * @return * @author zongf * @date 2019年1月8日-下午3:55:29 */ private String iso2Utf8(String str) { if(null == str) return null; try { return new String(str.getBytes("ISO-8859-1"), "UTF-8"); } catch (UnsupportedEncodingException e) { e.printStackTrace(); } return null; }
4. 屬性文件
筆者創建的是maven 工程, 使用的是junit 單元測試, 所以筆者的配置文件存放在 src/test/resources 目錄下.
jdbc.properties
jdbc.username=張三 jdbc.password=123456
jdbc_zh.properties
jdbc.username=張三 jdbc.password=123456
jdbc_en.properties
jdbc.username=zhangsan jdbc.password=123456
5. 實戰推薦
筆者認為, 一個設計良好的屬性配置類應該是一個常量類, 至少需要符合兩個設計原則:
屬性一旦設置不可動態修改, 即使在編譯環境也不能調用修改方法.
能夠直接通過類屬性進行訪問, 無須通過類對象訪問
自動裝配屬性, 而無須手動解析熟悉文件(spring中可借助自帶註解或自定義註解實現)
public class JdbcProperty { /** 用戶名 */ public static final String username; /** 用戶密碼 */ public static final String password; // 在Spring應用中,可借助註解或自定義註解進行自動裝配,筆者此處隻針對一般java應用 static { // 加載屬性文件 ResourceBundle resourceBundle = ResourceBundle.getBundle("jdbc"); // 初始化屬性 username = resourceBundle.getString("jdbc.username"); password = resourceBundle.getString("jdbc.password"); } }
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- 聊聊ResourceBundle和properties讀取配置文件的區別
- Java中關於文件路徑讀取問題的分析
- MyBatis使用註解開發和無主配置文件開發的情況
- Spring配置數據源的三種方式(小結)
- Java JDBC自定義封裝工具類的步驟和完整代碼