`
234390216
  • 浏览: 10229857 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
博客专栏
A5ee55b9-a463-3d09-9c78-0c0cf33198cd
Oracle基础
浏览量:462460
Ad26f909-6440-35a9-b4e9-9aea825bd38e
springMVC介绍
浏览量:1775244
Ce363057-ae4d-3ee1-bb46-e7b51a722a4b
Mybatis简介
浏览量:1398182
Bdeb91ad-cf8a-3fe9-942a-3710073b4000
Spring整合JMS
浏览量:394948
5cbbde67-7cd5-313c-95c2-4185389601e7
Ehcache简介
浏览量:679879
Cc1c0708-ccc2-3d20-ba47-d40e04440682
Cas简介
浏览量:530772
51592fc3-854c-34f4-9eff-cb82d993ab3a
Spring Securi...
浏览量:1183598
23e1c30e-ef8c-3702-aa3c-e83277ffca91
Spring基础知识
浏览量:467458
4af1c81c-eb9d-365f-b759-07685a32156e
Spring Aop介绍
浏览量:151279
2f926891-9e7a-3ce2-a074-3acb2aaf2584
JAXB简介
浏览量:68022
社区版块
存档分类
最新评论

实现对properties文件的有序读写

    博客分类:
  • java
阅读更多

实现对properties文件的有序读写

 

         最近遇到一项需求,要求把properties文件中的内容读取出来供用户修改,修改完后需要再重新保存到properties文件中。很简单的需求吧,可问题是Properties是继承自HashTable的,直接通过keySet()、keys()或entrySet()方法对Properties中的元素进行遍历时取出来的内容顺序与properties文件中的顺序不一致,这是问题一;问题二是就算取出来的时候是有序的,保存到文件中时又是无序的了。

         当然,解决这两个问题的方法有很多。我最终采用的方法是自定义一个PropertiesUtil类,该类继承自Properties。PropertiesUtil提供一个返回由key按照存入顺序组成的List的方法,getKeyList(),这样问题一就解决了。那如何保证getKeyList()方法返回的就是有序的key组成的集合呢?我查看了一下Properties方法的源码,发现其setProperty()方法实际上就是调用了父类HashTable的put()方法,其次Properties在从文件中加载内容时是按照文件顺序进行读取,然后调用父类HashTable的put()方法进行储存。所以问题的解决办法就是PropertiesUtil持有一个私有的可以有序存储key的集合,然后重写父类的put()方法,在方法体中照常通过super.put()进行属性的存储,同时将key添加到存储key的集合中。

         Properties提供有save()方法和store()方法可以将当前对象的内容存放到指定的输出流中,但它们的底层逻辑都是一样的。通过调用keys()方法获取一个Enumeration,然后对该Enumeration进行遍历,依次将对应的key和value写入到输出流中,所以要保证写入是有序的,就要保证遍历keys()返回的Enumeration时取出的元素key是有序的。所以解决方法是重写keys()方法,保证遍历返回的Enumeration时得到的key是有序的。完整代码如下:

 

import java.io.BufferedWriter;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.List;
import java.util.Properties;

public class PropertiesUtil extends Properties {

	private static final long serialVersionUID = 1L;

	private List<Object> keyList = new ArrayList<Object>();
	
	/**
	 * 默认构造方法
	 */
	public PropertiesUtil() {
		
	}
	
	/**
	 * 从指定路径加载信息到Properties
	 * @param path
	 */
	public PropertiesUtil(String path) {
		try {
			InputStream is = new FileInputStream(path);
			this.load(is);
		} catch (FileNotFoundException e) {
			e.printStackTrace();
			throw new RuntimeException("指定文件不存在!");
		} catch (IOException e) {
			e.printStackTrace();
		}
	}
	
	/**
	 * 重写put方法,按照property的存入顺序保存key到keyList,遇到重复的后者将覆盖前者。
	 */
	@Override
	public synchronized Object put(Object key, Object value) {
		this.removeKeyIfExists(key);
		keyList.add(key);
		return super.put(key, value);
	}
	

	/**
	 * 重写remove方法,删除属性时清除keyList中对应的key。
	 */
	@Override
	public synchronized Object remove(Object key) {
		this.removeKeyIfExists(key);
		return super.remove(key);
	}
	
	/**
	 * keyList中存在指定的key时则将其删除
	 */
	private void removeKeyIfExists(Object key) {
		keyList.remove(key);
	}
	
	/**
	 * 获取Properties中key的有序集合
	 * @return
	 */
	public List<Object> getKeyList() {
		return keyList;
	}
	
	/**
	 * 保存Properties到指定文件,默认使用UTF-8编码
	 * @param path 指定文件路径
	 */
	public void store(String path) {
		this.store(path, "UTF-8");
	}
	
	/**
	 * 保存Properties到指定文件,并指定对应存放编码
	 * @param path 指定路径
	 * @param charset 文件编码
	 */
	public void store(String path, String charset) {
		if (path != null && !"".equals(path)) {
			try {
				OutputStream os = new FileOutputStream(path);
				BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(os, charset));
				this.store(bw, null);
				bw.close();
			} catch (FileNotFoundException e) {
				e.printStackTrace();
			} catch (IOException e) {
				e.printStackTrace();
			}
		} else {
			throw new RuntimeException("存储路径不能为空!");
		}
	}

	/**
	 * 重写keys方法,返回根据keyList适配的Enumeration,且保持HashTable keys()方法的原有语义,
	 * 每次都调用返回一个新的Enumeration对象,且和之前的不产生冲突
	 */
	@Override
	public synchronized Enumeration<Object> keys() {
		return new EnumerationAdapter<Object>(keyList);
	}
    
	/**
	 * List到Enumeration的适配器
	 */
    private class EnumerationAdapter<T> implements Enumeration<T> {
		private int index = 0;
		private final List<T> list;
		private final boolean isEmpty;
		
		public EnumerationAdapter(List<T> list) {
			this.list = list;
			this.isEmpty = list.isEmpty();
		}
		
		public boolean hasMoreElements() {
			//isEmpty的引入是为了更贴近HashTable原有的语义,在HashTable中添加元素前调用其keys()方法获得一个Enumeration的引用,
			//之后往HashTable中添加数据后,调用之前获取到的Enumeration的hasMoreElements()将返回false,但如果此时重新获取一个
			//Enumeration的引用,则新Enumeration的hasMoreElements()将返回true,而且之后对HashTable数据的增、删、改都是可以在
			//nextElement中获取到的。
			return !isEmpty && index < list.size();
		}

		public T nextElement() {
			if (this.hasMoreElements()) {
				return list.get(index++);
			}
			return null;
		}
		
    }

}

 

 

 

 

 

3
0
分享到:
评论
3 楼 xiazhongwei 2016-10-19  
这个util怎么用的,能怎么读取项目中的.property文件进行修改
2 楼 234390216 2014-09-17  
asialee 写道
    解决这个问题看似简单,其实需要对properties有了解,lz采取的是继承的办法,不过我每每看到put(Object,Object)的时候就感觉不爽,properties本身对外暴露的api的key只能为string。
   

put(Object,Object)重写的是Properties的父类HashTable的put方法,再者Properties也没有对外开放的put(String,String)方法,设置属性是通过setProperty(String,String)进行的。如有任何疑问请参看Properties的源码。
1 楼 asialee 2014-09-17  
    解决这个问题看似简单,其实需要对properties有了解,lz采取的是继承的办法,不过我每每看到put(Object,Object)的时候就感觉不爽,properties本身对外暴露的api的key只能为string。
   

相关推荐

    Java代码实现对properties文件有序的读写的示例

    Java代码实现对properties文件有序的读写的示例 本篇文章主要介绍了Java代码实现对properties文件有序的读写的示例。Properties文件是一种常用的配置文件格式,用于存储应用程序的配置信息。然而,在读写Properties...

    day10-IO流&Properties集合1

    在本案例中,我们探讨的是如何结合使用IO流和Properties集合来实现数据的读写以及排序。下面我们将详细讲解相关知识点。 首先,我们看到一个名为"Student"的类,它代表一个学生对象,包含姓名、语文、数学和英语四...

    图书进销管理系统

    通过加载和保存.properties文件,系统能够灵活地读取和更新这些配置信息。 2. **List**: List是Java集合框架的一部分,它是一个有序的、可变大小的集合,允许重复元素。在图书管理系统中,List可以用来存储图书对象...

    TMS320C6748开发例程(1)

    操作步骤包括将HexAIS工具拷贝到C盘下,并在工程的Properties中设置Build-&gt;Variables-&gt;Add,填入HexAIS的路径,然后在Build-&gt;Steps-&gt;Post-build steps-&gt;Command中输入相应的命令,来生成.ais文件。 总的来说,本文...

    mybatis+redis实现二级缓存

    2. **配置Redis**:在应用的配置文件(如application.properties)中,设置Redis服务器的连接信息,如主机地址、端口、密码等。 3. **配置MyBatis**:在MyBatis的配置文件mybatis-config.xml中,启用二级缓存并指定...

    springboot面试题aop.docx

    在缓存处理中,可以通过 AOP 来实现缓存的读写操作。在事务管理中,可以通过 AOP 来实现事务的提交、回滚等。 Spring Boot 是一个非常流行的 Java 框架,提供了许多实用的功能和特性,包括简化的配置、自动配置、...

    28个java常用的工具类

    10. **Properties**: 用来加载和保存键值对,常用于配置文件的读写。 11. **UUID**: 生成全局唯一的128位标识符,常用于生成唯一ID。 12. **Scanner**: 提供了从各种输入源(如键盘、文件)读取基本类型和字符串的...

    Collectiion与Map类图

    4. Properties:Properties是特殊类型的Map,主要用于存储配置信息,键和值都是字符串类型,它支持读写文件。 在Java 8中,集合框架引入了一些新特性: - Lambda表达式和流API:这些新特性让集合的操作更加简洁和...

    Java代码管理器

    开发者可以通过Java的JGit库与Git进行交互,实现对本地Git仓库的操作,如添加、提交、拉取和推送。 2. 文件操作:Java的`java.io`和`java.nio`包提供了处理文件的基础功能,如读写文件、创建目录、重命名等。对于...

    springboot + redis实现session共享

    而Redis是一个高性能的键值数据库,常用于缓存和消息代理,它的特点是支持多种数据结构,如字符串、哈希、列表、集合、有序集合等,这使得它非常适合存储session数据。 要实现Spring Boot与Redis的session共享,...

    java常见集合继承结构图及核心知识点.pdf

    `Properties`类继承自`Hashtable`,并且是线程安全的,常用于配置文件的读写,键值对形式存储。 总的来说,Java集合框架通过各种接口和实现类提供了丰富的数据结构选择,以满足不同场景下的需求。了解它们的特性和...

    RocksDB的术语表.docx

    6. **LSM树(Log-Structured Merge-Tree)**:RocksDB的底层数据结构,所有写操作先写入内存中的Memtable,再定期刷入磁盘上的SST文件,通过定期的压缩操作保持数据的有序性。 7. **Write-Ahead-Log (WAL)**:预写式...

    一个java tcp服务器的基础框架

    `zys`目录可能包含了项目的源代码,其中的类和方法实现了TCP服务器的核心功能,如初始化服务器、处理连接、读写数据等。通常会有一个主类作为入口,启动服务器监听;其他类可能包括线程池管理、数据处理逻辑等。 六...

    SSH经典面试题汇总

    - 常见实现包括`HashMap`(基于哈希表)、`LinkedHashMap`(保持键值对插入顺序)、`Hashtable`(线程安全的`HashMap`)以及`Properties`(专用于读写配置文件),还有`SortedMap`接口的实现`TreeMap`。 ### 二、...

    javaIO流知识大总结

    - **创建、读取和写入文件:**通过File类操作文件,如new File()创建文件对象,FileInputStream和FileOutputStream实现文件读写。 - **复制文件:**通过流进行文件复制,通常使用循环读取源文件,然后写入目标文件...

    sharding-jdbc-example

    Sharding-JDBC是ShardingSphere项目的一部分,它定位为一个数据库中间件,可在不改变现有业务代码和数据库结构的前提下,实现对数据库的水平扩展。通过数据分片策略,将大规模数据分散到多个数据库中,从而提升系统...

    tomcat日志分割

    然而,随着服务器运行时间的增长,日志文件可能会变得非常大,这不仅占用大量磁盘空间,还可能影响系统的读写效率。因此,对Tomcat日志进行分割是非常必要的。 日志分割通常涉及以下方面: 1. **日志滚动策略**:...

    三、Nginx+Tomcat+Redis实现Session共享

    Redis 是一个内存中的数据结构存储系统,支持多种数据类型(如字符串、哈希、列表、集合、有序集合)和操作。在Session共享场景中,Redis作为中间缓存,用于存储用户的Session信息,以确保用户在多台服务器间切换时...

    集群 redis实现 session的 jar包之tomcat7

    接着,配置Spring Boot的`application.properties`文件,设置Redis连接信息: ```properties spring.redis.host=your.redis.host spring.redis.port=your.redis.port ``` 最后,创建一个配置类,启用Spring ...

    ssm+redis的增删改查的demo.zip

    - `application.properties`或`redis.properties`:配置Redis连接信息。 这个示例项目可以帮助开发者理解如何在实际项目中设置SSM+Redis的环境,进行数据操作,以及如何在出现异常时保持数据的一致性。通过研究和...

Global site tag (gtag.js) - Google Analytics