在Java开发中,配置信息存储是常见需求,除了常用的properties文件,XML文件也可以作为属性文件使用,它的层级结构能更清晰地组织复杂配置,适合存储多模块、多层级的应用配置信息。

XML作为属性文件的优势
和普通properties文件相比,XML作为属性文件有两个明显优势:一是支持嵌套结构,可以把同模块的配置放在同一个节点下,逻辑更清晰;二是可以存储非字符串类型的配置,比如列表、嵌套对象等,扩展性更强。
Java解析XML属性文件的常用方式
Java中解析XML的常用方式有DOM和SAX两种,DOM会把整个XML文档加载到内存形成树结构,方便随机访问和修改,适合配置文件较小的场景;SAX是事件驱动的解析方式,逐行读取XML,内存占用低,适合大文件解析,但只能顺序读取,不支持修改。
使用DOM解析XML属性文件
下面是用DOM解析XML属性文件的完整示例,首先准备一个存储配置的XML文件config.xml:
<?xml version="1.0" encoding="UTF-8"?>
<config>
<database>
<url>jdbc:mysql://127.0.0.1:3306/test</url>
<username>root</username>
<password>123456</password>
</database>
<app>
<name>测试应用</name>
<version>1.0</version>
</app>
</config>然后编写DOM解析代码读取配置:
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NodeList;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import java.io.File;
public class XmlPropertyReader {
public static void main(String[] args) {
try {
// 创建DocumentBuilderFactory实例
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
// 创建DocumentBuilder实例
DocumentBuilder builder = factory.newDocumentBuilder();
// 解析XML文件得到Document对象
Document document = builder.parse(new File("config.xml"));
// 获取根节点
Element root = document.getDocumentElement();
// 读取数据库配置
NodeList dbNodes = root.getElementsByTagName("database");
if (dbNodes.getLength() > 0) {
Element dbElement = (Element) dbNodes.item(0);
String url = dbElement.getElementsByTagName("url").item(0).getTextContent();
String username = dbElement.getElementsByTagName("username").item(0).getTextContent();
String password = dbElement.getElementsByTagName("password").item(0).getTextContent();
System.out.println("数据库地址:" + url);
System.out.println("数据库用户名:" + username);
System.out.println("数据库密码:" + password);
}
// 读取应用配置
NodeList appNodes = root.getElementsByTagName("app");
if (appNodes.getLength() > 0) {
Element appElement = (Element) appNodes.item(0);
String appName = appElement.getElementsByTagName("name").item(0).getTextContent();
String appVersion = appElement.getElementsByTagName("version").item(0).getTextContent();
System.out.println("应用名称:" + appName);
System.out.println("应用版本:" + appVersion);
}
} catch (Exception e) {
e.printStackTrace();
}
}
}使用SAX解析XML属性文件
如果需要解析大体积的XML配置文件,可以使用SAX解析,以下是SAX解析的示例代码:
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import java.io.File;
public class SaxXmlReader {
public static void main(String[] args) {
try {
SAXParserFactory factory = SAXParserFactory.newInstance();
SAXParser saxParser = factory.newSAXParser();
// 自定义处理器
DefaultHandler handler = new DefaultHandler() {
private String currentElement = "";
private boolean isDatabaseNode = false;
private boolean isAppNode = false;
@Override
public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
currentElement = qName;
if ("database".equals(qName)) {
isDatabaseNode = true;
} else if ("app".equals(qName)) {
isAppNode = true;
}
}
@Override
public void characters(char[] ch, int start, int length) throws SAXException {
String value = new String(ch, start, length).trim();
if (value.isEmpty()) {
return;
}
if (isDatabaseNode) {
if ("url".equals(currentElement)) {
System.out.println("数据库地址:" + value);
} else if ("username".equals(currentElement)) {
System.out.println("数据库用户名:" + value);
} else if ("password".equals(currentElement)) {
System.out.println("数据库密码:" + value);
}
} else if (isAppNode) {
if ("name".equals(currentElement)) {
System.out.println("应用名称:" + value);
} else if ("version".equals(currentElement)) {
System.out.println("应用版本:" + value);
}
}
}
@Override
public void endElement(String uri, String localName, String qName) throws SAXException {
if ("database".equals(qName)) {
isDatabaseNode = false;
} else if ("app".equals(qName)) {
isAppNode = false;
}
currentElement = "";
}
};
saxParser.parse(new File("config.xml"), handler);
} catch (Exception e) {
e.printStackTrace();
}
}
}注意事项
使用XML作为属性文件时,需要注意XML文件的编码要和解析时指定的编码一致,避免出现乱码问题。另外如果配置需要频繁修改,DOM方式更方便,因为可以直接修改内存中的节点再写回文件;如果只需要读取配置,SAX方式的效率更高。
实际开发中可以根据配置文件的规模和修改频率选择合适的解析方式,把XML作为属性文件使用能更好地管理复杂配置,提升配置的可维护性。