一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服務器之家 - 編程語言 - JAVA教程 - Java 配置加載機制詳解及實例

Java 配置加載機制詳解及實例

2020-06-18 11:17lqh JAVA教程

這篇文章主要介紹了Java 配置加載機制詳解及實例的相關資料,需要的朋友可以參考下

前言

現(xiàn)如今幾乎大多數(shù)Java應用,例如我們耳熟能詳?shù)膖omcat, struts2, netty…等等數(shù)都數(shù)不過來的軟件,要滿足通用性,都會提供配置文件供使用者定制功能。

甚至有一些例如Netty這樣的網絡框架,幾乎完全就是由配置驅動,這樣的軟件我們也通常稱之為”微內核架構”的軟件。你把它配置成什么,它就是什么。

It is what you configure it to be.

最常見的配置文件格式是XML, Properties等等文件。

本文探討加載配置中最通用也是最常見的場景,那就是把一個配置文件映射成Java里的POJO對象.

并探討如何實現(xiàn)不同方式的加載,例如,有一些配置是從本地XML文件里面加載的,而有一些配置需要從本地Properties文件加載,

更有甚者,有一些配置需要通過網絡加載配置。

如何實現(xiàn)這樣一個配置加載機制,讓我們擁有這個機制后,不會讓加載配置的代碼散布得到處都是,并且可擴展,可管理。

 配置加載器

首先,我們需要一個配置加載器,而這個配置加載器是可以有多種不同的加載方式的,因此,我們用一個接口來描述它,如下所示:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 上午11:47:12
 * @version 1.0
 *
 */
public interface IConfigLoader<T> {
 
  /**
   * load the config typed by T
   *
   * @return
   * @throws ConfigException
   */
  public T load() throws ConfigException;
}

可是,為什么我們需要在這個接口上聲明泛型 <T> ?

很明顯,當我們要使用一個配置加載器時,你得告訴這個配置加載器你需要加載后得到什么結果。

例如,你希望加載配置后得到一個 AppleConfig 對象,那么你就可以這么去使用上述定義的接口:

IConfigLoader<AppleConfig> loader = new AppleConfigLoader<AppleConfig>();
AppleConfig config = loader.load();

于是你將配置文件里的信息轉化成了一個AppleConfig對象,并且你能得到這個AppleConfig對象實例。

到目前,貌似只要我們的 AppleConfigLoader 里面實現(xiàn)了怎么加載配置文件的具體勞動,我們就可以輕易加載配置了。

可以這么說,但是不是還沒有考慮到,配置可能通過不同的方式加載呢,比如通過Properties加載,通過dom方式加載,通過sax方式加載,或者通過某些第三方的開源庫來加載。

因此,除了 配置加載器 ,我們還需要另外一種角色,配置加載方式的提供者。暫且,我們就叫它IConfigProvider。

配置加載方式的提供者

配置加載方式的提供者可以提供一種加載方式給配置加載器,換言之,提供一個 對象 給配置加載器。

  1. 如果通過dom方式加載,那么 提供者 提供一個 Document 對象給 加載器 。
  2. 如果通過Properties方式加載,那么 提供者 提供一個 Properties 對象給 加載器
  3. 如果通過第三方類庫提供的方式加載,比如apache-commons-digester3(tomcat的配置加載),那么 提供者 提供一個 Digester 對象給 加載器

提供者的職責就是 提供 ,僅此而已,只提供配置加載器所需要的對象,但它本身并不參與配置加載的勞動。

我們用一個接口 IConfigProvider 來定義這個 提供者

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 上午11:54:28
 * @version 1.0
 *
 */
public interface IConfigProvider<T> {
 
  /**
   * provide a config source used for loading config
   *
   * @return
   * @throws ConfigException
   */
  public T provide() throws ConfigException;
}

這里為什么又會有 <T> 來聲明泛型呢?

如果需要一個提供者,那么至少得告訴這個提供者它該提供什么吧。

因此,一個提供者會提供什么,由這個來決定。

同時,到這里,我們可以先建造一個工廠,讓它來生產特定的提供者:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 上午11:56:28
 * @version 1.0
 *
 */
public class ConfigProviderFactory {
 
  private ConfigProviderFactory() {
    throw new UnsupportedOperationException("Unable to initialize a factory class : "
        + getClass().getSimpleName());
  }
 
  public static IConfigProvider<Document> createDocumentProvider(String filePath) {
    return new DocumentProvider(filePath);
  }
 
  public static IConfigProvider<Properties> createPropertiesProvider(String filePath) {
    return new PropertiesProvider(filePath);
  }
 
  public static IConfigProvider<Digester> createDigesterProvider(String filePath) {
      return new DigesterProvider(filePath);
  }
}

可以開始實現(xiàn)具體配置加載器了?

還不行!

到這里,假設我們有一個配置文件,叫apple.xml。而且我們要通過DOM方式把這一份apple.xml加載后變成AppleConfig對象。

那么,首先我要通過提供者工廠給我制造一個能提供Document的提供者。然后拿到這個提供者,我就可以調用它的provide方法來獲得Document對象,有了document對象,那么我就可以開始來加載配置了。

可是,如果要加載BananaConfig、PearConfig…….呢,其步驟都是一樣的。因此我們還要有一個抽象類,來實現(xiàn)一些默認的共同行為。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 上午11:59:19
 * @version 1.0
 *
 */
public abstract class AbstractConfigLoader <T, U> implements IConfigLoader<T>{
 
  protected IConfigProvider<U> provider;
 
  protected AbstractConfigLoader(IConfigProvider<U> provider) {
    this.provider = provider;
  }
 
  /*
   * @see IConfigLoader#load()
   */
  @Override
  public T load() throws ConfigException {
    return load(getProvider().provide());
  }
 
  public abstract T load(U loaderSource) throws ConfigException;
 
  protected IConfigProvider<U> getProvider() {
    return this.provider;
  }
}

每個配置加載器都有一個帶參數(shù)構造器,接收一個Provider。

泛型指明了我要加載的是AppleConfig還是BananConfig,泛型 <U> 指明了要用什么加載方式加載,是Document呢,還是Properties,或者其他。

實戰(zhàn)運用實例

有一份菜市場配置文件market.xml,配置了菜市場的商品,里面有兩種商品,分別是蘋果和雞蛋。

?
1
2
3
4
5
6
7
8
9
<market>
  <apple>
    <color>red</color>
    <price>100</price>
  </apple>
  <egg>
    <weight>200</weight>
  </egg>
</market>

另外還有一份關于各個檔口老板名字的配置文件,owner.properties

port1=Steve Jobs
port2=Bill Gates
port3=Kobe Bryant

我們先定義好如下類:MarketConfig.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 下午11:03:37
 * @version 1.0
 *
 */
public class MarketConfig {
 
  private AppleConfig appleConfig;
  private EggConfig eggConfig;
  private OwnerConfig ownerConfig;
 
  public AppleConfig getAppleConfig() {
    return appleConfig;
  }
  public void setAppleConfig(AppleConfig appleConfig) {
    this.appleConfig = appleConfig;
  }
  public EggConfig getEggConfig() {
    return eggConfig;
  }
  public void setEggConfig(EggConfig eggConfig) {
    this.eggConfig = eggConfig;
  }
  public OwnerConfig getOwnerConfig() {
    return ownerConfig;
  }
  public void setOwnerConfig(OwnerConfig ownerConfig) {
    this.ownerConfig = ownerConfig;
  }
}

AppleConfig.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 下午11:03:45
 * @version 1.0
 *
 */
public class AppleConfig {
 
  private int price;
  private String color;
 
  public void setPrice(int price) {
    this.price = price;
  }
 
  public int getPrice() {
    return this.price;
  }
 
  public void setColor(String color) {
    this.color = color;
  }
 
  public String getColor() {
    return this.color;
  }
}

EggConfig.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 下午11:03:58
 * @version 1.0
 *
 */
public class EggConfig {
 
  private int weight;
 
  public void setWeight(int weight) {
    this.weight = weight;
  }
 
  public int getWeight() {
    return this.weight;
  }
}

OwnerConfig.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 下午11:04:06
 * @version 1.0
 *
 */
public class OwnerConfig {
 
  private Map<String, String> owner = new HashMap<String, String>();
 
  public void addOwner(String portName, String owner) {
    this.owner.put(portName, owner);
  }
 
  public String getOwnerByPortName(String portName) {
    return this.owner.get(portName);
  }
 
  public Map<String, String> getOwners() {
    return Collections.unmodifiableMap(this.owner);
  }
}

這個例子有兩種配置加載方式,分別是Dom和Properties加載方式。

所以我們的提供者建造工廠需要制造兩種提供者provider.

而且需要定義2個配置加載器,分別是:

OwnerConfigLoader

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 下午11:24:50
 * @version 1.0
 *
 */
public class OwnerConfigLoader extends AbstractConfigLoader<OwnerConfig, Properties>{
 
  /**
   * @param provider
   */
  protected OwnerConfigLoader(IConfigProvider<Properties> provider) {
    super(provider);
  }
 
  /*
   * @see AbstractConfigLoader#load(java.lang.Object)
   */
  @Override
  public OwnerConfig load(Properties props) throws ConfigException {
    OwnerConfig ownerConfig = new OwnerConfig();
 
    /**
     * 利用props,設置ownerConfig的屬性值
     *
     * 此處代碼省略
     */
    return ownerConfig;
  }
}

然后是MarketConfigLoader

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
import org.w3c.dom.Document;
 
/**
 *
 *
 * @author Bean
 * @date 2016年1月21日 下午11:18:56
 * @version 1.0
 *
 */
public class MarketConfigLoader extends AbstractConfigLoader<MarketConfig, Document> {
 
  /**
   * @param provider
   */
  protected MarketConfigLoader(IConfigProvider<Document> provider) {
    super(provider);
  }
 
  /*
   * AbstractConfigLoader#load(java.lang.Object)
   */
  @Override
  public MarketConfig load(Document document) throws ConfigException {
 
    MarketConfig marketConfig = new MarketConfig();
    AppleConfig appleConfig = new AppleConfig();
    EggConfig eggConfig = new EggConfig();
    /**
     * 在這里處理document,然后就能得到
     * AppleConfig和EggConfg
     *
     * 此處代碼省略
     */
    marketConfig.setAppleConfig(appleConfig);
    marketConfig.setEggConfig(eggConfig);
 
    /**
     * 由于OwnerConfig是需要properties方式來加載,不是xml
     * 所以這里要新建一個OwnerConfigLoader,委托它來加載OwnerConfig
     */
 
    OwnerConfigLoader ownerConfigLoader = new OwnerConfigLoader(ConfigProviderFactory.createPropertiesProvider(YOUR_FILE_PATH));
    OwnerConfig ownerConfig = ownerConfigLoader.load();
 
    marketConfig.setOwnerConfig(ownerConfig);
 
    return marketConfig;
  }
}

然后,我們在應用層面如何獲取到MarketConfig呢

?
1
2
MarketConfigLoader marketConfigLoader = new MarketConfigLoader(ConfigProviderFactory.createDocumentProvider(YOUR_FILE_PATH));
MarketConfig marketConfig = marketConfigLoader.load();

也許有個地方會人奇怪,明明有四個配置類,為什么只有2個配置加載器呢。因為MarketConfig、EggConfig和AppleConfig,都是從同一個xml配置文件里面加載,所以只要一個Document對象,通過MarketConfigLoader就可以全部加載。

而OwnerConfig是不同的加載方式,所以需要另外一個加載器。

總結

本文提出的配置加載機制,并不能夠實際幫忙加載配置,這事應該留給DOM,SAX,以及其他一些開源庫如dom4j,Digester去做。但本文提出的配置加載機制能夠讓配置加載機制更靈活,容易擴展,并且能夠集成多種配置加載方式,融合到一個機制進來,發(fā)揮各自有點。

實際上,有些軟件經常需要同時從多種不同格式的配置文件里面加載配置,例如struts2,以及我最近在研究并被氣到吐血的某國產開源數(shù)據(jù)庫中間件軟件,如果沒有一套完整的配置加載機制,那么代碼會比較散亂,可維護性不高。容易使人吐血。

通過此文希望大家理解并掌握 Java 配置加載機制的知識,謝謝大家對本站的支持!

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 国产精品成人va在线观看 | 无限韩国视频免费播放 | 99久久综合九九亚洲 | 好 舒服 好 粗 好硬 好爽 | 欧美日韩国产一区二区三区欧 | 国产成年人视频 | 日本艳鉧动漫1~6在线观看 | 国产一区二区三区久久精品小说 | 亚洲视频在线免费看 | 草莓香蕉绿巨人丝瓜榴莲18 | 奇米网7777| 香蕉91xj.cc | 公妇乱淫在线播放免费观看 | gayxxx视频| 久久99re2热在线播放7 | 美女扒开奶罩让男人吃奶 | 国色天香社区视频免费观看3 | 毛片免费全部免费观看 | 国产精品毛片久久久久久久 | 成人啪啪漫画全文阅读 | 亚洲国产视频网站 | 国产在线视频自拍 | 国产区一二三四区2021 | 丰满大屁股美女一级毛片 | 亚洲久操 | 亚洲国产情侣一区二区三区 | 91大神第九部红酒气质女 | 亚洲天堂99| 青青草99热久久 | 国内精品久久久久久久久久久久 | 大象传媒1234区 | 日韩精品一区二区三区中文字幕 | japanesepooping脱粪 | 青青热久麻豆精品视频在线观看 | 国产大秀视频一区二区三区 | 无限在线观看视频大全免费高清 | 欧美一区二区三区免费观看视频 | 日本视频高清 | 欧美高清在线 | 国产成人一区二区三区在线视频 | 四虎在线永久视频观看 |