如何在Java中加载和解析(蚂蚁风格)属性文件?
How to load and parse (ant style) properties file in Java?
如何将 属性 文件加载到 Java 中的 属性 对象中并获取解析的 属性 值(${x}
被替换为 ant特性)?例如使用这个 属性 文件:
foo=1
bar=${foo}.0
我需要 bar
属性 作为 1.0
而不是 ${foo}.0
。有没有简单的方法?
编辑:
Alex 的解决方案适用于简单的场景。就我而言,我必须解决导致此问题的另一个问题:Pulling values from a Java Properties file in order?。
加载和解析属性的结果示例代码:
import java.util.*;
import java.io.*;
import org.apache.commons.text.StringSubstitutor;
public class Prop {
Properties parsedProperties = null;
public static Properties parseProperties(String filename) {
// inner helper class keeping order of properties:
class LinkedProperties extends Properties {
private static final long serialVersionUID = 1L;
private final HashSet<Object> keys = new LinkedHashSet<Object>();
public LinkedProperties() {
}
public Iterable<Object> orderedKeys() {
return Collections.list(keys());
}
public Enumeration<Object> keys() {
return Collections.<Object>enumeration(keys);
}
public Object put(Object key, Object value) {
keys.add(key);
return super.put(key, value);
}
}
LinkedProperties result = new LinkedProperties();
try (InputStream input = new FileInputStream(filename)) {
result.load(input);
@SuppressWarnings("unchecked")
StringSubstitutor sub = new StringSubstitutor((Map) result);
for (Object k : result.orderedKeys()) {
result.setProperty((String)k, sub.replace(result.getProperty((String)k)));
}
} catch (IOException ex) { ex.printStackTrace(); }
return ((Properties)result);
}
public static void main(String[] args) {
Prop app = new Prop();
// test - write sample properties file:
try {
PrintWriter writer = new PrintWriter(new FileWriter("config.properties"));
writer.println("foo=1");
writer.println("bar=1.${foo}");
writer.println("baz=${bar}.0");
writer.println("xxx=V.${baz}");
writer.close();
} catch (IOException ex) { ex.printStackTrace(); }
// read and parse properties:
app.parsedProperties = parseProperties("config.properties");
// debug print:
for (Object k : app.parsedProperties.keySet()) {
System.out.println((String)k + " = " + app.parsedProperties.getProperty((String)k));
}
}
}
您可以使用 Apache Commons Text 中的 StringSubstitutor
,它的 Maven 依赖性非常适中(~200K):
<!-- https://mvnrepository.com/artifact/org.apache.commons/commons-text -->
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-text</artifactId>
<version>1.8</version>
</dependency>
代码示例:
// init sample properties
Properties p = new Properties();
p.setProperty("foo", "${baz}.${baz}");
p.setProperty("bar", "${foo}.0");
p.setProperty("baz", "5");
Properties resolved = parseProperties(p);
System.out.println("resolved: " + resolved);
/////
public static Properties parseProperties(Properties orig) {
Properties result = new Properties();
StringSubstitutor sub = new StringSubstitutor((Map) orig);
orig.entrySet().forEach(e -> result.put(e.getKey(), sub.replace(e.getValue())));
return result;
}
输出:
resolved: {bar=5.5.0, foo=5.5, baz=5}
如何将 属性 文件加载到 Java 中的 属性 对象中并获取解析的 属性 值(${x}
被替换为 ant特性)?例如使用这个 属性 文件:
foo=1
bar=${foo}.0
我需要 bar
属性 作为 1.0
而不是 ${foo}.0
。有没有简单的方法?
编辑: Alex 的解决方案适用于简单的场景。就我而言,我必须解决导致此问题的另一个问题:Pulling values from a Java Properties file in order?。
加载和解析属性的结果示例代码:
import java.util.*;
import java.io.*;
import org.apache.commons.text.StringSubstitutor;
public class Prop {
Properties parsedProperties = null;
public static Properties parseProperties(String filename) {
// inner helper class keeping order of properties:
class LinkedProperties extends Properties {
private static final long serialVersionUID = 1L;
private final HashSet<Object> keys = new LinkedHashSet<Object>();
public LinkedProperties() {
}
public Iterable<Object> orderedKeys() {
return Collections.list(keys());
}
public Enumeration<Object> keys() {
return Collections.<Object>enumeration(keys);
}
public Object put(Object key, Object value) {
keys.add(key);
return super.put(key, value);
}
}
LinkedProperties result = new LinkedProperties();
try (InputStream input = new FileInputStream(filename)) {
result.load(input);
@SuppressWarnings("unchecked")
StringSubstitutor sub = new StringSubstitutor((Map) result);
for (Object k : result.orderedKeys()) {
result.setProperty((String)k, sub.replace(result.getProperty((String)k)));
}
} catch (IOException ex) { ex.printStackTrace(); }
return ((Properties)result);
}
public static void main(String[] args) {
Prop app = new Prop();
// test - write sample properties file:
try {
PrintWriter writer = new PrintWriter(new FileWriter("config.properties"));
writer.println("foo=1");
writer.println("bar=1.${foo}");
writer.println("baz=${bar}.0");
writer.println("xxx=V.${baz}");
writer.close();
} catch (IOException ex) { ex.printStackTrace(); }
// read and parse properties:
app.parsedProperties = parseProperties("config.properties");
// debug print:
for (Object k : app.parsedProperties.keySet()) {
System.out.println((String)k + " = " + app.parsedProperties.getProperty((String)k));
}
}
}
您可以使用 Apache Commons Text 中的 StringSubstitutor
,它的 Maven 依赖性非常适中(~200K):
<!-- https://mvnrepository.com/artifact/org.apache.commons/commons-text -->
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-text</artifactId>
<version>1.8</version>
</dependency>
代码示例:
// init sample properties
Properties p = new Properties();
p.setProperty("foo", "${baz}.${baz}");
p.setProperty("bar", "${foo}.0");
p.setProperty("baz", "5");
Properties resolved = parseProperties(p);
System.out.println("resolved: " + resolved);
/////
public static Properties parseProperties(Properties orig) {
Properties result = new Properties();
StringSubstitutor sub = new StringSubstitutor((Map) orig);
orig.entrySet().forEach(e -> result.put(e.getKey(), sub.replace(e.getValue())));
return result;
}
输出:
resolved: {bar=5.5.0, foo=5.5, baz=5}