Java - Properties和ResourceBundle类学习

一、前言

  在项目的开发过程中,为了统一配置的管理,我们经常需要将一些配置信息根据环境的不同,配置在不同的properties中,然后从里面进行读取。而Properties类作为最基础也是最经常使用的类,通过本文我们来学习一下它的使用,然后再顺便学习下其他几种读取properties文件的方式。

二、Properties和ResourceBundle类

  Properties表示一个持久的属性集,属性列表通过key-value的形式存在,并且key和value都是字符串。我们先来看一下它的继承结构:

1. Properties 继承结构
public
class Properties extends Hashtable<Object,Object> {  
    /**
     * A property list that contains default values for any keys not
     * found in this property list.
     *
     * @serial
     */
    protected Properties defaults;
     
    public Properties() {
        this(null);
    }

    public Properties(Properties defaults) {
        this.defaults = defaults;
    }
}

看到它的继承结构,就知道这个类已经存在好久了。该类继承自Hashtable,所以该类拥有Map的一切功能,所以Map的put或者putAll方法都可以使用。

不过由于Properties中的每个key及value都是字符串,所以官方强烈反对使用它们,因为这些方法允许key或者value不是字符串,如果在set或get操作的时候,不是字符串的话,则会提示异常,所以建议使用的则是诸如setProperty这些方法。

2. Properties常用方法

这里Map相关的方法就不介绍了,主要介绍下自定义的方法:

2.1 setProperty方法
public synchronized Object setProperty(String key, String value) {
    return put(key, value);
}

底层通过Hashtable的put方法来实现,该方法的目的就是强制对属性的key和value都使用字符串的形式;

2.2 getProperties方法
public String getProperty(String key)
public String getProperty(String key, String defaultValue) {
    String val = getProperty(key);
    return (val == null) ? defaultValue : val;
}

获取属性列表中属性的key对应的值,第二个重载方法表示如果获取不到值返回参数中提供的默认值。

2.3 load方法
public synchronized void load(Reader reader) throws IOException
public synchronized void load(InputStream inStream) throws IOException

load方法表示从输入流(字节流和字符流)中读取属性列表到Properties中,读取的时候按照行进行读取。而有关读取行及相关转义的说明,可以参考对应的API文档,上面有详细的说明。

2.4 loadFromXML方法
public synchronized void loadFromXML(InputStream in)
        throws IOException, InvalidPropertiesFormatException

从输入流中读取XML文件中的所有属性,注意XML文档必须有相应的DTD声明:

<!DOCTYPE properties SYSTEM "http://java.sun.com/dtd/properties.dtd">
2.5 store方法
public void store(Writer writer, String comments)
    throws IOException

public void store(OutputStream out, String comments)
    throws IOException

和load的功能相反,将Properties的属性列表写入到输出流,其中参数表示对属性列表的说明。

2.6 storeToXML方法
public void storeToXML(OutputStream os, String comment)
    throws IOException

public void storeToXML(OutputStream os, String comment, String encoding)
    throws IOException

将属性写入到xml,并可以指定的编码格式。

2.7 propertyNames和stringPropertyNames方法
public Enumeration<?> propertyNames()
public Set<String> stringPropertyNames()

两个方法都是返回Properties属性列表中所有key,前者返回所有枚举,后者返回类型是字符串,注意如果没有在主属性列表中找到同名的键,则在默认属性列表中进行查找。

2.8 list方法
public void list(PrintStream out)
public void list(PrintWriter out)

将属性列表输出到指定的输出流,这个方法对调试很有用。

3. ResourceBundle简介

  ResourceBundle没有继承什么类,是一个单个的抽象类,该类可以说是国际化版的Properties,简单说就是可以根据本地化或语言的不同读取不同的配置文件,但要注意的一点是使用ResourceBundle读取的时候,properties的命名是有一定规范的:

名称_语言代码_国家代码.properties
// 如果是默认的
自定义名.properties
// 例如
myres_en_US.properties
myres_zh_CN.properties
myres.properties

  当指定的Locale是CN的时候,如果myres_zh_CN.properties、myres.properties两个文件都存在,则优先会使用myres_zh_CN.properties,当myres_zh_CN.properties不存在时候,会使用默认的myres.properties。

4. ResourceBundle常用方法
4.1 getBundle方法

ResourceBundle提供了多个重载的静态getBundle方法,用于获取资源文件,这里我们不多介绍,后续看实例即可:

public static final ResourceBundle getBundle(String baseName)
public static final ResourceBundle getBundle(String baseName,
                                                 Locale locale)
public static ResourceBundle getBundle(String baseName, Locale locale,
                                           ClassLoader loader)
public static final ResourceBundle getBundle(String baseName,
                                                 Control control)
public static ResourceBundle getBundle(String baseName, Locale targetLocale,
                                           ClassLoader loader, Control control)
4.2 getObject,getString,getStringArray方法
  1. getString方法比较简单,就是根据key获取属性列表中key对应的value,key和value都是String;
  2. getStringArray方法,用于获取字符串数组,返回值是字符串数组;
  3. getObject方法,通用的获取方法,获取其他任何类型;
public final String getString(String key)
public final String[] getStringArray(String key)
public final Object getObject(String key)
4.3 keySet,getKeys,containsKey方法

这几个方法都比较简单,见名知义,其中getKeys表示返回key的枚举形式。

public Set<String> keySet()
public abstract Enumeration<String> getKeys();
public boolean containsKey(String key)
4.4 getBaseBundleName,getLocale方法

getBaseBundleName方法就是获取加载的资源文件的文件名的,getLocale获取本地化环境信息的。

public String getBaseBundleName()
public Locale getLocale()
4.5 clearCache方法

通过getBundle方法读取资源文件,获取ResourceBundle时是从缓存中获取的,如果已经缓存,工厂方法将多次返回相同的资源实例,而clearCache方法就是用于清除缓存的:

public static final void clearCache()
public static final void clearCache(ClassLoader loader)

这里参考自:51cto - java.util.ResourceBundle使用详解

4. 简单示例

接下来我们来简单看下这些方法的相关使用说明。

4.1 Properties 通过store方法写入对应的文件中

  首先我们调用setProperty方法会将key-value存于内存中,此时可以通过getProperty方法读取,propertyNames方法进行遍历,但是并没有将键值对持久化到属性文件中,故需要调用store方法持久化键值对到属性文件中。

Properties properties = new Properties();
try {
    OutputStream output = new FileOutputStream("cache.properties");
    properties.setProperty("redis.server.address", "127.0.0.1");
    properties.setProperty("redis.server.port", "6379");
    properties.setProperty("redis.server.password", "");
    properties.setProperty("redis.server.timeout", "2000");
    properties.store(output, "缓存文件配置测试");
} catch (IOException io) {
    io.printStackTrace();
} finally {
    ...
}

最终生成的cache.properties文件:

#缓存文件配置测试
#Sun Aug 19 12:27:05 CST 2018
redis.server.timeout=2000
redis.server.address=127.0.0.1
redis.server.password=
redis.server.port=6379
4.2 Properties使用load方法加载

同样,我们可以通过load方法将属性文件中的属性加载到Properties对象中,然后进行访问:

Properties properties = new Properties();
InputStream inputStream = null;
try {
    inputStream = new FileInputStream("cache.properties");
    properties.load(inputStream);
    for (String key : properties.stringPropertyNames()) {
        System.out.println(key + "=" + properties.getProperty(key));
    }
} catch (IOException io) {
    io.printStackTrace();
} finally {
    ...
}

最终结果:

redis.server.timeout=2000
redis.server.address=127.0.0.1
redis.server.password=
redis.server.port=6379

因为Properties其实是一个Map,所以Map的遍历方式也适用于Properties,也可也借助Properties的propertyNames方法来进行遍历:

Enumeration<?> e = properties.propertyNames();
while (e.hasMoreElements()) {
    String key = (String) e.nextElement();
    String value = properties.getProperty(key);
    System.out.println(key + "=" + value);
}
4.3 ResourceBundle简单实例

我们先定义三个资源文件,放到src的根目录下面:
myres.properties

aaa=good 
bbb=thanks

myres_en_US.properties

aaa=good 
bbb=thanks

myres_zh_CN.properties

aaa=好
bbb=谢谢

添加测试代码:

public static void main(String[] args) {
    Locale locale1 = new Locale("zh", "CN");
    ResourceBundle resb1 = ResourceBundle.getBundle("cache", locale1);
    System.out.println(resb1.getString("aaa"));

    ResourceBundle resb2 = ResourceBundle.getBundle("cache", Locale.getDefault());
    System.out.println(resb1.getString("aaa"));

    Locale locale3 = new Locale("en", "US");
    ResourceBundle resb3 = ResourceBundle.getBundle("cache", locale3);
    System.out.println(resb3.getString("aaa"));
}

output:

好
好
good 

这里需要注意下,通过getBundle方法来获取的时候,参数不用加properties后缀,只需要文件名就可以了,并且默认访问的路径是src,如果文件保存在其他目录,记得修改到对应的目录。

5. 其他读取properties文件的方法
5.1 Properties其他获取InputStream的方法

在这里,其实读取properties都是通过Properties来实现的,不过不同的是获取InputStream流的方式,我们来看一下各种获取InputStream流的方式:

    1. 从File文件获取:
InputStream inputStream = new FileInputStream(new File ("cache.properties"));
    1. 根据ClassLoader的getResourceAsStream方法来获取。其中该方法又分为两种方式:
  1. Class.getResourceAsStream(String path) : path 不以’/'开头时默认是从此类所在的包下取资源,以’/'开头则是从ClassPath根下获取。其实只是通过path构造一个绝对路径,最终还是由ClassLoader获取资源;
  2. Class.getClassLoader.getResourceAsStream(String path) :默认则是从ClassPath根下获取,path不能以’/'开头,最终是由ClassLoader获取资源;
  3. ServletContext.getResourceAsStream(String path):默认从WebAPP根目录下取资源,Tomcat下path是否以’/'开头无所谓,当然这和具体的容器实现有关;

这里参考自:iteye - Java中getResourceAsStream的用法

InputStream inputStream = PropertiesTest.class.getResourceAsStream("cache.properties");
    1. 通过URL来获取:
URL url = new URL("path");
InputStream inputStream= url.openStream();
    1. 如果是Spring环境,还可以通过ResourceLoader的getResource得到Resource,然后通过Resource的getInputStream来得到流:
ResourceLoader resourceLoader = new DefaultResourceLoader();
Resource resource = resourceLoader.getResource("/config/cache.properties");
inputStream = resource.getInputStream();
5.2 ResourceBundle类相关方法

前面也已经简单介绍过,我们可以借助java.util.ResourceBundle的getBundle静态方法来获取资源实例:

Locale locale1 = new Locale("zh", "CN");
ResourceBundle resb1 = ResourceBundle.getBundle("cache", locale1);

另外,也可以借助实现类 PropertyResourceBundle 通过从输入流中进行读取:

inputStream = new FileInputStream("cache.properties");
ResourceBundle resource = new PropertyResourceBundle(inputStream);

不过如果有兴趣的话可以看下PropertyResourceBundle的构造方法,它其实是借助Properties和HashMap来实现的:

public PropertyResourceBundle (InputStream stream) throws IOException {
    Properties properties = new Properties();
    properties.load(stream);
    lookup = new HashMap(properties);
}

private Map<String,Object> lookup;

三、总结

  到这里,就基本介绍完了Properties和ResourceBundle类及如何读取properties文件,其实,介绍的都比较基础,需要注意的可能就两点吧:一是路径的问题,二是流的关闭和异常处理问题。

本文参考自:
Java Properties类使用详解
Java读取properties文件的思考 - this & getClass

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,402评论 6 499
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,377评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,483评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,165评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,176评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,146评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,032评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,896评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,311评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,536评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,696评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,413评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,008评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,815评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,698评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,592评论 2 353

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,651评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,072评论 25 707
  • 最近,我遇见了一位蜜汁欢乐的的哥,他有种本事,能把平淡的生活过成节日。跟他聊天很引人入胜,似乎是一个能量场,能把人...
    千语树阅读 299评论 0 1
  • 暮色四合,风寒星冷。尘世如潮,可否醉一场不再醒来……门前邻人种植的蔬菜,经了一场雨,长势很好,已经可以食用了。看天...
    程杜阅读 144评论 0 0
  • 《眼泪》 也许只有眼泪 曾经相信过爱情 但我现在已经欲哭无泪 《悲剧》 两个人的相处本来就是一场博弈 或许最后也会...
    幻梦邪魂阅读 330评论 0 2