成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

深入淺出的分析 Properties

開發 后端
最近在看 java 集合源碼的時候,發現原來我們經常使用的 Properties 類既然繼承自 Hashtable!又漲見識了!

 最近在看 java 集合源碼的時候,發現原來我們經常使用的 Properties 類既然繼承自 Hashtable!又漲見識了!

[[281908]]

01. 摘要

在集合系列的第一章,咱們了解到,Map 的實現類有 HashMap、LinkedHashMap、TreeMap、IdentityHashMap、WeakHashMap、Hashtable、Properties 等等。

在上一章節中,咱們介紹到 Hashtable 的數據結構和算法實現,在 Java 中其實還有一個非常重要的類 Properties,它繼承自 Hashtable,主要用于讀取配置文件。

本文通過看 JDK 和一些網友的博客總結,主要從 Properties 的用法實例來做介紹,如果有理解不當之處,歡迎指正。

02. 簡介

Properties 類是 java 工具包中非常重要的一個類,比如在實際開發中,有些變量,我們可以直接硬寫入到自定義的 java 枚舉類中。

但是有些變量,在測試環境、預生產環境、生產環境,變量所需要取的值都不一樣,這個時候,我們可以通過使用 properties 文件來加載程序需要的配置信息,以達到一行代碼,多處環境都可以運行的效果!

最常見的比如 JDBC 數據源配置文件,properties文件以.properties作為后綴,文件內容以鍵=值格式書寫,左邊是變量名稱,右邊是變量值,用#做注釋,比如新建一個jdbc.properties文件,內容如下:

Properties 類是 properties 文件和程序的中間橋梁,不論是從 properties 文件讀取信息,還是寫入信息到 properties 文件,都要經由 Properties 類。

好了,嘮叨了這么多,咱們回到本文要介紹的主角Properties!

從集合 Map 架構圖可以看出,Properties 繼承自 Hashtable,表示一個持久的 map 集合,屬性列表以 key-value 的形式存在,Properties 類定義如下:

  1. public class Properties extends Hashtable<Object,Object> { 
  2.     ...... 

Properties 除了繼承 Hashtable 中所定義的方法,Properties 也定義了以下幾個常用方法,如圖所示:

 

常用方法介紹

set 方法(添加修改元素)

set 方法是將指定的 key, value 對添加到 map 里,在添加元素的時候,調用了 Hashtable 的 put 方法,與 Hashtable 不同的是, key 和 value 都是字符串。

打開 Properties 的 setProperty 方法,源碼如下:

  1. public synchronized Object setProperty(String key, String value) { 
  2.     //調用父類 Hashtable 的 put 方法 
  3.     return put(key, value); 

方法測試如下:

  1. public static void main(String[] args) { 
  2.     Properties properties = new Properties(); 
  3.     properties.setProperty("name1","張三"); 
  4.     properties.setProperty("name2","張四"); 
  5.     properties.setProperty("name3","張五"); 
  6.     System.out.println(properties.toString()); 

輸出結果:

{name3=張五, name2=張四, name1=張三}

get 方法(搜索指定元素)

get 方法根據指定的 key 值返回對應的 value,第一步是從調用 Hashtable 的 get 方法,如果有返回值,直接返回;如果沒有返回值,但是初始化時傳入了defaults變量,從 defaults變量中,也就是 Properties 中,去搜索是否有對于的變量,如果有就返回元素值。

打開 Properties 的 getProperty 方法,源碼如下:

  1. public String getProperty(String key) { 
  2.     //調用父類 Hashtable 的 get 方法 
  3.     Object oval = super.get(key); 
  4.     String sval = (oval instanceof String) ? (String)oval : null
  5.      //進行變量非空判斷 
  6.     return ((sval == null) && (defaults != null)) ? defaults.getProperty(key) : sval; 

查看 defaults 這個變量,源碼如下:

  1. public class Properties extends Hashtable<Object,Object> { 
  2.     protected Properties defaults; 

這個變量在什么時候賦值呢,打開源碼如下:

  1. public Properties(Properties defaults) { 
  2.     this.defaults = defaults; 

可以發現,在 Properties 構造方法初始化階段,如果你給了一個自定義的 defaults ,當調用 Hashtable 的 get 方法沒有搜索到元素值的時候,并且 defaults 也不等于空,那么就會進一步在 defaults 里面進行搜索元素值。

方法測試如下:

  1. public static void main(String[] args) { 
  2.     Properties properties = new Properties(); 
  3.     properties.setProperty("name1","張三"); 
  4.     properties.setProperty("name2","張四"); 
  5.     properties.setProperty("name3","張五"); 
  6.     //將 properties 作為參數初始化到 newProperties 中 
  7.     Properties newProperties = new Properties(properties); 
  8.     newProperties.setProperty("name4","李三"); 
  9.     //查詢key中 name1 的值 
  10.     System.out.println("查詢結果:" + properties.getProperty("name1")); 

輸出結果:

通過key查詢結果:張三

load方法(加載配置文件)

load 方法,表示將 properties 文件以輸入流的形式加載文件,并且提取里面的鍵、值對,將鍵值對元素添加到 map 中去。

打開 Properties 的 load 方法,源碼如下:

  1. public synchronized void load(InputStream inStream) throws IOException { 
  2.     //讀取文件流 
  3.     load0(new LineReader(inStream)); 

load0 方法,源碼如下:

  1. private void load0 (LineReader lr) throws IOException { 
  2.     char[] convtBuf = new char[1024]; 
  3.     int limit; 
  4.     int keyLen; 
  5.     int valueStart; 
  6.     char c; 
  7.     boolean hasSep; 
  8.     boolean precedingBackslash; 
  9.  
  10.     //一行一行的讀取 
  11.     while ((limit = lr.readLine()) >= 0) { 
  12.         c = 0; 
  13.         keyLen = 0; 
  14.         valueStart = limit; 
  15.         hasSep = false
  16.  
  17.         precedingBackslash = false
  18.         //判斷key的長度 
  19.         while (keyLen < limit) { 
  20.             c = lr.lineBuf[keyLen]; 
  21.             if ((c == '=' ||  c == ':') && !precedingBackslash) { 
  22.                 valueStart = keyLen + 1; 
  23.                 hasSep = true
  24.                 break; 
  25.             } else if ((c == ' ' || c == '\t' ||  c == '\f') && !precedingBackslash) { 
  26.                 valueStart = keyLen + 1; 
  27.                 break; 
  28.             } 
  29.             if (c == '\\') { 
  30.                 precedingBackslash = !precedingBackslash; 
  31.             } else { 
  32.                 precedingBackslash = false
  33.             } 
  34.             keyLen++; 
  35.         } 
  36.         //獲取值的起始位置 
  37.         while (valueStart < limit) { 
  38.             c = lr.lineBuf[valueStart]; 
  39.             if (c != ' ' && c != '\t' &&  c != '\f') { 
  40.                 if (!hasSep && (c == '=' ||  c == ':')) { 
  41.                     hasSep = true
  42.                 } else { 
  43.                     break; 
  44.                 } 
  45.             } 
  46.             valueStart++; 
  47.         } 
  48.         //獲取文件中的鍵和值參數 
  49.         String key = loadConvert(lr.lineBuf, 0, keyLen, convtBuf); 
  50.         String value = loadConvert(lr.lineBuf, valueStart, limit - valueStart, convtBuf); 
  51.         //調用 Hashtable 的 put 方法,將鍵值加入 map 中 
  52.         put(key, value); 
  53.     } 

好了,我們來在src/recources目錄下,新建一個custom.properties配置文件,內容如下:

  1. #定義一個變量名稱和值 
  2. userName=李三 
  3. userPwd=123456 
  4. userAge=18 
  5. userGender=男 
  6. userEmail=123@123.com 

方法測試如下:

  1. public class TestProperties  { 
  2.  
  3.     public static void main(String[] args) throws Exception { 
  4.         //初始化 Properties 
  5.         Properties prop = new Properties(); 
  6.         //加載配置文件 
  7.         InputStream in = TestProperties .class.getClassLoader().getResourceAsStream("custom.properties"); 
  8.         //讀取配置文件,指定編碼格式,避免讀取中文亂碼 
  9.         prop.load(new InputStreamReader(in"UTF-8")); 
  10.         //將內容輸出到控制臺 
  11.         prop.list(System.out); 
  12.     } 

輸出結果:

userPwd=123456

userEmail=123@123.com

userAge=18

userName=李三

userGender=男

propertyNames方法(讀取全部信息)

propertyNames 方法,表示讀取 Properties 的全部信息,本質是創建一個新的 Hashtable 對象,然后將原 Hashtable 中的數據復制到新的 Hashtable 中,并將 map 中的 key 全部返回。

打開 Properties 的 propertyNames 方法,源碼如下:

  1. public Enumeration<?> propertyNames() { 
  2.     Hashtable<String,Object> h = new Hashtable<>(); 
  3.     //將原 map 添加到新的 Hashtable 中 
  4.     enumerate(h); 
  5.     //返回 Hashtable 中全部的 key 元素 
  6.     return h.keys(); 

enumerate 方法,源碼如下:

  1. private synchronized void enumerate(Hashtable<String,Object> h) { 
  2.     //判斷 Properties 中是否有初始化的配置文件 
  3.     if (defaults != null) { 
  4.         defaults.enumerate(h); 
  5.     } 
  6.     //將原 Hashtable 中的數據添加到新的 Hashtable 中 
  7.     for (Enumeration<?> e = keys() ; e.hasMoreElements() ;) { 
  8.         String key = (String)e.nextElement(); 
  9.         h.put(key, get(key)); 
  10.     } 

方法測試如下:

  1. public static void main(String[] args) throws Exception { 
  2.     //初始化 Properties 
  3.     Properties prop = new Properties(); 
  4.     //加載配置文件 
  5.     InputStream in = TestProperties.class.getClassLoader().getResourceAsStream("custom.properties"); 
  6.     //讀取配置文件,指定讀取編碼 UTF-8,防止內容亂碼 
  7.     prop.load(new InputStreamReader(in"UTF-8")); 
  8.     //獲取 Properties 中全部的 key 元素 
  9.     Enumeration enProp = prop.propertyNames(); 
  10.     while (enProp.hasMoreElements()){ 
  11.         String key = (String) enProp.nextElement(); 
  12.         String value = prop.getProperty(key); 
  13.         System.out.println(key + "=" + value); 
  14.     } 

輸出內容如下:

userPwd=123456

userEmail=123@123.com

userAge=18

userName=李三

userGender=男

總結

Properties 繼承自 Hashtable,大部分方法都復用于 Hashtable,比如,get、put、remove、clear 方法,**與 Hashtable 不同的是, Properties中的 key 和 value 都是字符串,**如果需要獲取 properties 中全部內容,可以先通過迭代器或者 propertyNames 方法獲取 map 中所有的 key 元素,然后遍歷獲取 key 和 value。

需要注意的是,Properties 中的 setProperty 、load 方法,都加了synchronized同步鎖,用來控制線程同步。

03. properties 文件的加載方式

在實際開發中,經常會遇到讀取配置文件路徑找不到,或者讀取文件內容亂碼的問題,下面簡單介紹一下,properties 文件的幾種常用的加載方式。

properties 加載文件的方式,大致可以分兩類,第一類是使用 java.util.Properties 的 load 方法來加載文件流;第二類是使用 java.util.ResourceBundle 類來獲取文件內容。

在src/recources目錄下,新建一個custom.properties配置文件,文件編碼格式為UTF-8,內容還是以剛剛那個測試為例,各個加載方式如下!

通過文件路徑來加載文件

這類方法加載文件,主要是調用 Properties 的 load 方法,獲取文件路徑,讀取文件以流的形式加載文件。

方法如下:

  1. Properties prop = new Properties(); 
  2. //獲取文件絕對路徑 
  3. String filePath = "/coding/java/src/resources/custom.properties"
  4. //加載配置文件 
  5. InputStream in = new FileInputStream(new File(filePath)); 
  6. //讀取配置文件 
  7. prop.load(new InputStreamReader(in"UTF-8")); 
  8. System.out.println("userName:"+prop.getProperty("userName")); 

輸出結果:

userName:李三 

通過當前類加載器的getResourceAsStream方法獲取

這類方法加載文件,也是調用 Properties 的 load 方法,不同的是,通過類加載器來獲取文件路徑,如果當前文件是在src/resources目錄下,那么直接傳入文件名就可以了。

方法如下:

  1. Properties prop = new Properties(); 
  2. //加載配置文件 
  3. InputStream in = TestProperties.class.getClassLoader().getResourceAsStream("custom.properties"); 
  4. //讀取配置文件 
  5. prop.load(new InputStreamReader(in"UTF-8")); 
  6. System.out.println("userName:"+prop.getProperty("userName")); 

輸出結果:

userName:李三 

使用ClassLoader類的getSystemResourceAsStream方法獲取

和上面類似,也是通過類加載器來獲取文件流,方法如下:

Properties prop = new Properties();//加載配置文件InputStream in = ClassLoader.getSystemResourceAsStream("custom.properties");//讀取配置文件prop.load(new InputStreamReader(in, "UTF-8"));System.out.println("userName:"+prop.getProperty("userName"));

輸出結果:

userName:李三

使用 ResourceBundle 類加載文件

ResourceBundle 類加載文件,與 Properties 有所不同,ResourceBundle 獲取 properties 文件不需要加.properties后綴名,只需要文件名即可。

ResourceBundle 是按照iso8859編碼格式來讀取原屬性文件,如果是讀取中文內容,需要進行轉碼處理。

方法如下:

  1. //加載custom配置文件,不需要加`.properties`后綴名 
  2. ResourceBundle resource = ResourceBundle.getBundle("custom"); 
  3. //轉碼處理,解決讀取中文內容亂碼問題 
  4. String value = new String(resource.getString("userName").getBytes("ISO-8859-1"),"UTF-8"); 
  5. System.out.println("userName:"+value); 

輸出結果:

userName:李三

04. 總結

從源碼上可以看出,Properties 繼承自 Hashtable,大部分方法都復用于 Hashtable,與 Hashtable 不同的是, Properties 中的 key 和 value 都是字符串。

實際開發中,Properties 主要用于讀取配置文件,尤其是在不同的環境下,變量值需要不一樣的情況,可以通過讀取配置文件來避免將變量值寫死在 java 的枚舉類中,以達到一行代碼,多處運行的目的!

在讀取 Properties 配置文件的時候,容易因文件路徑找不到報錯,可以參考 properties 文件加載的幾種方式,

責任編輯:華軒 來源: Java極客技術
相關推薦

2019-11-14 09:53:30

Set集合存儲

2021-03-16 08:54:35

AQSAbstractQueJava

2011-07-04 10:39:57

Web

2013-11-14 15:53:53

AndroidAudioAudioFlinge

2009-11-30 16:46:29

學習Linux

2022-12-02 09:13:28

SeataAT模式

2019-01-07 15:29:07

HadoopYarn架構調度器

2017-07-02 18:04:53

塊加密算法AES算法

2012-05-21 10:06:26

FrameworkCocoa

2021-07-20 15:20:02

FlatBuffers阿里云Java

2022-09-26 09:01:15

語言數據JavaScript

2015-08-06 14:02:31

數據分析

2018-03-15 09:13:43

MySQL存儲引擎

2023-03-20 09:48:23

ReactJSX

2009-12-25 15:49:43

Linux rescu

2012-02-21 13:55:45

JavaScript

2022-01-11 07:52:22

CSS 技巧代碼重構

2018-11-09 16:24:25

物聯網云計算云系統

2022-11-09 08:06:15

GreatSQLMGR模式

2022-10-31 09:00:24

Promise數組參數
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品久久久久久婷婷天堂 | 日韩精品一区在线 | 久久一区二区三区免费 | 91视频在线看 | 欧美一区二区三区视频 | 久久久久久99 | 日本a视频| 亚洲影视在线 | 男人天堂网av| 国产一区二区三区久久久久久久久 | 播放一级毛片 | 一级网站 | 91精品国产91久久久久久吃药 | 精品欧美一区二区精品久久 | 日日想夜夜操 | 波多野结衣中文视频 | 国产高清在线精品一区二区三区 | 日韩国产三区 | 精品一二三 | 精精国产xxxx视频在线播放7 | 亚洲精品久久久久久久久久久 | 成人在线视频一区二区三区 | 91精品在线看 | 国产在线观看网站 | 福利视频一区二区 | a级黄色毛片免费播放视频 国产精品视频在线观看 | 欧美成人精品在线观看 | 精品亚洲一区二区三区 | 91精品免费视频 | 久久久久国产一区二区三区 | 国产精品福利在线观看 | 欧美黄色精品 | 久久综合成人精品亚洲另类欧美 | 国产成人精品在线 | 亚洲一区不卡 | 欧美精品一区二区免费 | 午夜欧美 | 国产成人jvid在线播放 | 久久国内精品 | 亚洲91精品 | 国产高清精品一区二区三区 |