Properties
是 java
的配置文件。
在编写应用程序的时候,经常需要读写配置文件。例如,用户的设置:
1 | #上次最后打开的文件: |
配置文件的特点是,它的Key-Value
一般都是String-String
类型的,因此我们完全可以用Map<String, String>
来表示它。
因为配置文件非常常用,所以Java集合库
提供了一个Properties
来表示一组“配置”。由于历史遗留原因,Properties
内部本质上是一个Hashtable
,但我们只需要用到Properties
自身关于读写配置的接口。
读取配置文件
用Properties
读取配置文件非常简单。Java
默认配置文件以.properties
为扩展名,每行以key=value
表示,以#
课开头的是注释。以下是一个典型的配置文件:
1 | # setting.properties |
可见,用Properties
读取配置文件,一共有三步:
- 创建
Properties
实例; - 调用
load()
读取文件; - 调用
getProperty()
获取配置。
调用getProperty()
获取配置时,如果key
不存在,将返回null
。我们还可以提供一个默认值,这样,当key
不存在的时候,就返回默认值。
也可以从classpath
读取.properties
文件,因为load(InputStream)
方法接收一个InputStream
实例,表示一个字节流,它不一定是文件流,也可以是从jar包中读取的资源流:
1 | Properties props = new Properties(); |
试试从内存读取一个字节流:
1 | import java.io.*; |
如果有多个.properties
文件,可以反复调用load()
读取,后读取的key-value
会覆盖已读取的key-value
:
1 | Properties props = new Properties(); |
上面的代码演示了Properties
的一个常用用法:可以把默认配置文件放到classpath
中,然后,根据机器的环境编写另一个配置文件,覆盖某些默认的配置。
Properties
设计的目的是存储String
类型的key-value
,但Properties
实际上是从Hashtable
派生的,它的设计实际上是有问题的,但是为了保持兼容性,现在已经没法修改了。除了getProperty()
和setProperty()
方法外,还有从Hashtable
继承下来的get()
和put()
方法,这些方法的参数签名是Object
,我们在使用Properties
的时候,不要去调用这些从Hashtable
继承下来的方法。
写入配置文件
如果通过setProperty()
修改了Properties
实例,可以把配置写入文件,以便下次启动时获得最新配置。写入配置文件使用store()
方法:
1 | Properties props = new Properties(); |
编码
早期版本的Java
规定.properties
文件编码是ASCII编码
(ISO8859-1
),如果涉及到中文就必须用name=\u4e2d\u6587
来表示,非常别扭。从JDK9
开始,Java
的.properties
文件可以使用UTF-8
编码了。
不过,需要注意的是,由于load(InputStream)
默认总是以ASCII编码
读取字节流,所以会导致读到乱码。我们需要用另一个重载方法load(Reader)
读取:
1 | Properties props = new Properties(); |
就可以正常读取中文。InputStream
和Reader
的区别是一个是字节流,一个是字符流。字符流在内存中已经以char类型表示了,不涉及编码问题。
小结
Java集合库
提供的Properties
用于读写配置文件.properties
。.properties
文件可以使用UTF-8
编码。
可以从文件系统、classpath
或其他任何地方读取.properties
文件。
读写Properties
时,注意仅使用getProperty()
和setProperty()
方法,不要调用继承而来的get()
和put()
等方法。