import org.apache.commons.configuration.ConfigurationException;
import org.apache.commons.configuration.PropertiesConfiguration; import java.io.File;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.Date; public class PropertiesDemo { public static void main(String[] args) throws IOException, ConfigurationException {
String path = PropertiesDemo.class.getResource("/t.properties").getPath(); PropertiesConfiguration config = new PropertiesConfiguration(new File(path));
System.out.println(config.getEncoding());
config.setEncoding("gbk");
config.setAutoSave(false);
String value = new SimpleDateFormat("yyyy-mm-dd HH:mm:ss").format(new Date());
System.out.println(value);
config.setProperty("ums.version.endTime444", value);
config.setProperty("中文key", "中文value");
System.out.println(config.getEncoding());
config.save(); // Properties properties = new Properties();
// properties.load(new FileReader(new File(path)));
// properties.setProperty("hello", value);
// FileWriter fileWriter = new FileWriter(new File(path));
// properties.store(fileWriter, "Comment"); } }

在Java早期的开发中,常用*.properties文件存储一些配置信息。其文件中的信息主要是以key=value的方式进行存储,在早期受到广泛的应用。而后随着xml使用的广泛,其位置渐渐被取代,不过,目前仍有一些框架如log4J在使用它。最近在弄自己的小玩意儿的时候也用到了它,顺便加深了一下了解,在此分享。

Java在对*.properties文件进行操作的时候,实际上是通过IO对文档进行逐行的扫描,然后将文中非注释的部分存放在一个properties对象中。Properties 实际上是继承了hashtable,实现了Map接口。可以这样理解,它是进行了进一步封装的HashMap。存放到properties中后,可以对properties进行一系列的操作,此时的数据保存在内存中。最后,当需要保存数据的时候,是将properties中所有的键值重新写入到文件中去。 对properties文件的操作,jdk提供了一系列的API。一下是一个工具类,实现了对properties文件的增删查改的功能。

  1 package com.sean.file.properties;
2
3 import java.io.File;
4 import java.io.FileOutputStream;
5 import java.io.IOException;
6 import java.io.InputStreamReader;
7 import java.io.OutputStreamWriter;
8 import java.io.UnsupportedEncodingException;
9 import java.util.Enumeration;
10 import java.util.HashMap;
11 import java.util.Map;
12 import java.util.Properties;
13 /**
14 * Java 操作Properties的工具类
15 * 实现功能:
16 * 1、Properties文件的增删查改功能
17 * 2、解决读写中文乱码问题
18 * @author Sean
19 *
20 */
21 public class PropertiesUtil {
22
23 /**
24 * Properties地址值,不需要加根标记"/"
25 */
26 private String src = "";
27 private InputStreamReader inputStream = null;
28 private OutputStreamWriter outputStream = null;
29 private String encode="utf-8";
30 public Properties properties ;
31
32 /**
33 * 默认构造函数
34 */
35 public PropertiesUtil() {
36 }
37
38 /**
39 * 构造函数
40 *
41 * @param src 传入Properties地址值,不需要加根标记"/"
42 */
43 public PropertiesUtil(String src) {
44 this.src = src;
45 }
46
47
48 /**
49 * 构造函数,提供设置编码模式
50 * @param src 传入Properties地址值,不需要加根标记"/"
51 * @param encode 传入对应的编码模式,默认是utf-8
52 */
53 public PropertiesUtil(String src, String encode) {
54 this(src);
55 this.encode = encode;
56 }
57
58 /**
59 * 加载properties文件
60 * @author Sean
61 * @date 2015-6-5
62 * @return 返回读取到的properties对象
63 */
64 public Properties load(){
65 if(src.trim().equals("")){
66 throw new RuntimeException("The path of Properties File is need");
67 }
68 try {
69 inputStream=new InputStreamReader(ClassLoader.getSystemResourceAsStream(src),encode);
70 } catch (UnsupportedEncodingException e1) {
71 e1.printStackTrace();
72 }
73 properties=new Properties();
74 try {
75 properties.load(inputStream);
76 } catch (IOException e) {
77 e.printStackTrace();
78 }
79 return properties;
80 }
81
82 /**
83 * 将配置写入到文件
84 * @author Sean
85 * @date 2015-6-5
86 * @throws Exception
87 */
88 public void write2File() throws Exception{
89 //获取文件输出流
90 outputStream=new OutputStreamWriter(new FileOutputStream(new File(ClassLoader.getSystemResource(src).toURI())),encode);
91 properties.store(outputStream, null);
92 close();
93 }
94
95
96 /**
97 * 通过关键字获取值
98 * @author Sean
99 * @date 2015-6-5
100 * @param key 需要获取的关键字
101 * @return 返回对应的字符串,如果无,返回null
102 */
103 public String getValueByKey(String key){
104 properties=load();
105 String val =properties.getProperty(key.trim());
106 close();
107 return val;
108
109 }
110
111 /**
112 * 通过关键字获取值
113 * @author Sean
114 * @date 2015-6-5
115 * @param key 需要获取的关键字
116 * @param defaultValue 若找不到对应的关键字时返回的值
117 * @return 返回找到的字符串
118 */
119 public String getValueByKey(String key ,String defaultValue){
120 properties=load();
121 String val =properties.getProperty(key.trim(),defaultValue.trim());
122 close();
123 return val;
124 }
125
126 /**
127 * 关闭输入输出流
128 * @author Sean
129 * @date 2015-6-5
130 */
131 public void close(){
132 try {
133 if(inputStream!=null){inputStream.close();}
134 if(outputStream!=null){outputStream.close();}
135 } catch (IOException e) {
136 e.printStackTrace();
137 }
138 }
139
140 /**
141 * 获取Properties所有的值
142 * @author Sean
143 * @date 2015-6-5
144 * @return 返回Properties的键值对
145 */
146 public Map<String,String> getAllProperties(){
147 properties=load();
148 Map<String,String> map=new HashMap<String,String>();
149 //获取所有的键值
150 Enumeration enumeration=properties.propertyNames();
151 while(enumeration.hasMoreElements()){
152 String key=(String) enumeration.nextElement();
153 String value=getValueByKey(key);
154 map.put(key, value);
155 }
156 close();
157 return map;
158 }
159
160 /**
161 * 往Properties写入新的键值
162 * @author Sean
163 * @date 2015-6-5
164 * @param key 对应的键
165 * @param value 对应的值
166 */
167 public void addProperties(String key,String value){
168 properties=load();
169 properties.put(key, value);
170 try {
171 write2File();
172 } catch (Exception e) {
173 e.printStackTrace();
174 }
175 }
176
177 /**
178 * 添加Map中所有的值
179 * @author Sean
180 * @date 2015-6-5
181 * @param map 对应的键值对集合
182 */
183 public void addAllProperties(Map<String,String> map){
184 properties=load();
185 properties.putAll(map);
186 try {
187 write2File();
188 } catch (Exception e) {
189 e.printStackTrace();
190 throw new RuntimeException("write fail");
191 }
192 }
193
194 /**
195 * 更新配置文件
196 * @author Sean
197 * 2015-6-5
198 * @param key 需要更新的键值
199 * @param value 对应的值
200 */
201 public void update(String key,String value){
202 properties=load();
203 if(!properties.containsKey(key)){
204 throw new RuntimeException("not such key");
205 }
206 properties.setProperty(key, value);
207 try {
208 write2File();
209 } catch (Exception e) {
210 e.printStackTrace();
211 throw new RuntimeException("write fail");
212 }
213 }
214
215 /**
216 * 删除某一键值对
217 * @author Sean
218 * 2015-6-5
219 * @param key 对应的键值
220 */
221 public void deleteKey(String key){
222 properties=load();
223 if(!properties.containsKey(key)){
224 throw new RuntimeException("not such key");
225 }
226 properties.remove(key);
227 try {
228 write2File();
229 } catch (Exception e) {
230 e.printStackTrace();
231 throw new RuntimeException("write fail");
232 }
233 }
234
235 /**
236 * 设置path值
237 * @author Sean
238 * @date 2015-6-5
239 * @param src 对应文件值
240 */
241 public void setSrc(String src) {
242 this.src = src;
243 }
244 }

基本上,常用的对properties的操作采用这个工具类都能完成。值得注意的是,因为程序在对properties进行扫描的时候,忽略了注释的内容,而当重新写内容入文字的时候,程序也只会将properties中的值写入,这样会注释丢失的情况。这种情况,网上有人的解决方案是自己新建一个继承了properties的类,然后将程序读取到的信息包括注释放入到LinkedHashMap中,这样就可以保存注释了。

不过,同样的功能,通过Apache 上的开源项目commons-configuration也能实现。commons-configuration 主要是实现对如xml, properties等配置文件操作的API。通过它,同样能实现对properties的操作,同时,在操作的时候,也可以保存文件中的注释。

在使用commons-configuration进行properties的文件操作的时候,不仅需要导入commons-configuration.jar 包,还需要导入另外几个依赖包才能实现。

以下是本能采用commons-configuration提供的api进行properties操作的一个Demo

package com.sean;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.net.URISyntaxException; import org.apache.commons.configuration.Configuration;
import org.apache.commons.configuration.ConfigurationException;
import org.apache.commons.configuration.PropertiesConfiguration; public class PropertiesUtil {
private String src="";
private PropertiesConfiguration pcf=null;
private String encode="utf-8";
/**
* 默认构造函数
*/
public PropertiesUtil(){};
/**
* 传参构造函数
* @param src 传入对应文件地址
*/
public PropertiesUtil(String src){
this.src=src;
try {
pcf=new PropertiesConfiguration(src);
} catch (ConfigurationException e) {
e.printStackTrace();
}
pcf.setEncoding(encode);
} /**
* 获取特定key的值
* @param key 对应的键值
* @return 返回对应value值,找不到返回null;
*/
public String getValue(String key){
String s=pcf.getString(key);
return s;
}
/**
* 更新对应的值
* @param key 对应的关键字
* @param value 对应的值
*/
public void updateValue(String key,String value) {
if(!pcf.containsKey(key)){
throw new RuntimeException("not such key");
}
try {
pcf.save(new FileOutputStream(new File(ClassLoader.getSystemResource(src).toURI())),"utf-8");
} catch (ConfigurationException e) {
e.printStackTrace();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (URISyntaxException e) {
e.printStackTrace();
}
} /**
* 添加键值对
* @param key 关键字
* @param value 值
*/
public void addValue(String key,String value){
pcf.addProperty(key, value);
try {
pcf.save(new FileOutputStream(new File(ClassLoader.getSystemResource(src).toURI())),"utf-8");
} catch (ConfigurationException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (URISyntaxException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
/**
* 删除关键字
* @param key 关键字
*/
public void delValue(String key){
pcf.clearProperty(key);
try {
pcf.save(new FileOutputStream(new File(ClassLoader.getSystemResource(src).toURI())),"utf-8");
} catch (ConfigurationException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (URISyntaxException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} }

可以看出,commons-configuration提供的api操作起来简单多了。可是,因为commons-configuration是国外的开源项目,所以其对中文的支持存在一些问题。尽管API中提供了设置字符编码的功能,但是还是没有能够非常好的解决中文的问题。相对而言,原生的API实现起来比较的简单。

最后做下总结,关于对properties 的操作,jdk和commons-configuration都提供了较好的支持,关于使用原生还是框架,应该根据具体条件而定。不过我们在不重复造轮的情况下,还是应该保持对原理的探讨

http://www.cnblogs.com/Seanit/p/4555937.html

 我们在开发Java程序的时候,很多常量信息都存在配置文件中,比如数据库连接信息、ip黑名单,事件的超时时间等等。当需要该这些配置的值时都需要重新启动进程,改动的配置才会生效,有时候线上的应用不能容忍这种停服。

  还好,Apache Common Configuration给我们提供了可以检测文件修改后配置可短时间生效的功能。具体用法如下:

package com.netease.test.commons;

import org.apache.commons.configuration.ConfigurationException;
import org.apache.commons.configuration.PropertiesConfiguration;
import org.apache.commons.configuration.reloading.FileChangedReloadingStrategy;
import org.apache.log4j.Logger; /**
* User: hzwangxx
* Date: 14-3-13
* Time: 17:20
*/
public class SystemConfig {
private static Logger logger = Logger.getLogger(SystemConfig.class); private static PropertiesConfiguration config; static {
try {
//实例化一个PropertiesConfiguration
config = new PropertiesConfiguration("/Users/hzwangxx/IdeaProjects/app-test/src/main/resources/conf.properties");
//设置reload策略,这里用当文件被修改之后reload(默认5s中检测一次)
config.setReloadingStrategy(new FileChangedReloadingStrategy());
} catch (ConfigurationException e) {
logger.error("init static block error. ", e);
}
} public static synchronized String getProperty(String key) {
return (String) config.getProperty(key);
} public static void main(String[] args) throws InterruptedException {
for (;;) {
System.out.println(SystemConfig.getProperty("key"));
Thread.sleep(2000);
}
} }

http://www.cnblogs.com/nexiyi/p/how_to_reload_properties.html

PropertiesDemo的更多相关文章

  1. Day22_IO第四天

    1.合并流(序列流)-) { //不断的在a.txt上读取字节 fos.write(b1); //将读到的字节写到c.txt上 } fis1.close(); //关闭字节输入流 FileInputS ...

  2. IO流05--毕向东JAVA基础教程视频学习笔记

    Day20 10 创建java文件列表11 Properties简述12 Properties存取13 Properties存取配置文件14 Properties练习15 PrintWriter16 ...

  3. 【JAVA Properties类概述】

    一.概述. 之前说过,该对象是和IO流相结合的技术,所以和IO流结合在一起来讲比较合适. public class Propertiesextends Hashtable<Object,Obje ...

  4. Java常用类之Properties类

    1.特性 Properties类表示了一个持久的属性集,可保存在流中或从流中加载,实现内存和文件的交互.Properties继承了Hashtable<Object,Object>类,可以使 ...

  5. Java 文件IO续

    文件IO续 File类    用来将文件和文件夹封装成对象 方便对文件和文件夹的属性信息进行操作    File对象可以作为参数传递给流的构造函数 Demo1 File的构造方法 public cla ...

  6. javaSE第二十二天

    第二十二天    312 1:登录注册IO版本案例(掌握)    312 2:数据操作流(操作基本类型数据的流)(理解)    313 (1)定义:    313 (2)流对象名称    313 (3 ...

  7. Java:IO流其他类(字节数组流、字符数组流、数据流、打印流、Properities、对象流、管道流、随机访问、序列流、字符串读写流)

    一.字节数组流: 类 ByteArrayInputStream:在构造函数的时候,需要接受数据源,而且数据源是一个字节数组. 包含一个内部缓冲区,该缓冲区包含从流中读取的字节.内部计数器跟踪 read ...

  8. java基础知识回顾之java集合类-Properties集合

    /** java.lang.Object   |--java.util.Dictionary<K,V>      |--java.util.Hashtable<Object,Obje ...

  9. Java基础知识强化之IO流笔记66:Properties的概述 和 使用(作为Map集合使用)

    1. Properties的概述  Properties:属性集合类.是一个可以和IO流相结合使用的集合类. 该类主要用于读取以项目的配置文件(以.properties结尾的文件 和 xml文件). ...

随机推荐

  1. 深入浅出Hive企业级架构优化、Hive Sql优化、压缩和分布式缓存(企业Hadoop应用核心产品)

    一.本课程是怎么样的一门课程(全面介绍)    1.1.课程的背景       作为企业Hadoop应用的核心产品,Hive承载着FaceBook.淘宝等大佬 95%以上的离线统计,很多企业里的离线统 ...

  2. 关于ARM开发板与PC主机的网络设置问题

    直观来讲,ARM开发板多数情况下会有条网线与主机相连,所以最重要的一步是保证PC主机与ARM开发板能互通. 互通的意思进一步来讲就是互相能ping通.也就是说在瘟都死的dos下(假设主机是瘟都死系统) ...

  3. Window7下安装openssl完整版(亲测实现)

    安装环境: 操作系统:window7(64位) C++编译器:VS2010 -------------------------------------------------------------- ...

  4. oracle列合并

    在很多场合,我们会须要用到oracle列合并,oracle提供了一些方法用于把某列的多行数据合并成一行. 一.10G曾经使用WMSYS.WM_CONCAT   wmsys.wm_concat将字段的值 ...

  5. fake it till you become it

    fake it till you become it_你泛起山川烟波里的不是我._百度空间 fake it till you become it

  6. IE7IE8兼容性设置_服务器端设定

    用ASP.NET开发的web程序,客户端是IE7以上时,需要设置IE的兼容性,否则web页面可能混乱,但这样在每客户端设置不太现实, 对策: ①在站点的配置文件中,修改web.config文件,使得该 ...

  7. 如何关闭android studio开发环境自动保存

    使用DW习惯了现在转到学习开发android,请问怎样关闭android studio的自动保存功能,然后按ctrl+s进行保存,因为有时候代码不想让其保存,他也自动保存了. File -> S ...

  8. wikioi 1014 装箱问题

    来源:http://wikioi.com/problem/1014/ 1014 装箱问题 29人推荐 收藏 发题解 提交代码 报错 题目描写叙述 输入描写叙述 输出描写叙述 例子输入 例子输出 提示 ...

  9. STM8S---IO复用配置(STVP方式)

    1 说明 STM8S的IO复用用程序代码配置起来比較麻烦.通常是操作flash来操作option byte字节.配置寄存器更加麻烦,能够使用STM 标准外设驱动库来设置. 本文使用一种界面配置的方式来 ...

  10. hdu 5115 Dire Wolf

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5115 题目分类:区间dp 题意:有n只狼,每只狼有两种属性,一种攻击力一种附加值,我们没杀一只狼,那么 ...