Writing a configuration file is a rather complicated matter. I've tried saving configuration items in a comma-separated text file, and I've tried saving configuration items in very verbose YAML and XML. For configuration files, the most important thing is to have consistency and regularity. They allow you to write code easily and quickly and parse data from the configuration file; at the same time, when the user decides to make changes, it is easy to save them. and update configuration.
There are currently several popular configuration file formats. For most common configuration file formats, Java has corresponding libraries. In this article, I will use XML format. For some projects, you might choose to use XML because one of its standout features is its ability to provide a wealth of relevant metadata for the data it contains, while for other projects, you might not choose XML because of its verbosity. Working with XML in Java is very easy as it includes many robust XML libraries by default.
Discussing XML is a big topic. I have a book on XML that is over 700 pages long. Fortunately, working with XML doesn't require an intimate knowledge of its many features. Just like HTML, XML is a layered markup language with opening and closing tags, each of which can contain zero or more data. Here is a simple example snippet of XML:
Penguin
In this self-descriptive example, the XML parser uses the following concepts:
Documentation Document: The
tag marks the beginning of a document, and the tag marks the end of this document.
Node Node:
The label represents a node.
Element:
, from the beginning <
to the last > ;
represents an element.
Content: In the
element, the string Penguin
is the content.
Believe it or not, as long as you understand the above concepts, you can start writing and parsing XML files.
To learn how to parse XML files, all you need is a minimalist sample file. Suppose there is a configuration file, which saves the properties of a graphical interface window:
Dark 0 Tango
Create a directory named ~/.config/DemoXMLParser
:
$ mkdir ~/.config/DemoXMLParser
In Linux, the ~/.config
directory is the default location for storing configuration files, which is defined in the specifications of the Free Desktop Working Group. If you are using an operating system that does not adhere to the Freedesktop standards, you can still use this directory, but you will need to create these directories yourself.
Copy the XML sample configuration file, paste and save it as ~/.config/DemoXMLParser/myconfig.xml
file.
If you are a beginner in Java, you can first read the 7 tips I wrote for beginner Java developers. Once you are more familiar with Java, open your favorite integrated development tool (IDE) and create a new project. I will name my new project myConfigParser
.
Don’t pay too much attention to dependency imports and exception catching at the beginning. You can first try to instantiate using the standard Java extensions in the javax
and java.io
packages. A parser. If you are using an IDE, it will prompt you to import the appropriate dependencies. If not, you can find the complete code later in the article, which has a complete list of dependencies.
Path configPath = Paths.get(System.getProperty("user.home"), ".config", "DemoXMLParser"); File configFile = new File(configPath.toString(), "myconfig.xml"); DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); DocumentBuilder builder = null; builder = factory.newDocumentBuilder(); Document doc = null; doc = builder.parse(configFile); doc.getDocumentElement().normalize();
This sample code uses the java.nio.Paths
class to find the user's home directory, and then splices the path to the default configuration file. Then, it uses the java.io.File
class to define the configuration file as a File
object.
Next, it uses the two classes javax.xml.parsers.DocumentBuilder
and javax.xml.parsers.DocumentBuilderFactory
to create an internal document structure Converter so that Java programs can import and parse XML data.
Finally, Java creates a document object called doc
and loads the configFile
file into this object. Using the org.w3c.dom
package, it reads and normalizes XML data.
That’s basically it. In theory, you have completed the data analysis work. However, data parsing is of little use if you don't have access to the data. So, let's write some more queries to read important property values from your configuration.
从你已经读取的 XML 文档中获取数据,其实就是要先找到一个特定的节点,然后遍历它包含的所有元素。通常我们会使用多个循环语句来遍历节点中的元素,但是为了保持代码可读性,我会尽可能少地使用循环语句:
NodeList nodes = doc.getElementsByTagName("window"); for (int i = 0; i < nodes.getLength(); i++) { Node mynode = nodes.item(i); System.out.println("Property = " + mynode.getNodeName()); if (mynode.getNodeType() == Node.ELEMENT_NODE) { Element myelement = (Element) mynode; System.out.println("Theme = " + myelement.getElementsByTagName("theme").item(0).getTextContent()); System.out.println("Fullscreen = " + myelement.getElementsByTagName("fullscreen").item(0).getTextContent()); System.out.println("Icon set = " + myelement.getElementsByTagName("icons").item(0).getTextContent()); } }
这段示例代码使用了 org.w3c.dom.NodeList
类,创建了一个名为 nodes
的 NodeList
对象。这个对象包含了所有名字匹配字符串 window
的子节点,实际上这样的节点只有一个,因为本文的示例配置文件中只配置了一个。
紧接着,它使用了一个 for
循环来遍历 nodes
列表。具体过程是:根据节点出现的顺序逐个取出,然后交给一个 if-then
子句处理。这个 if-then
子句创建了一个名为 myelement
的 Element
对象,其中包含了当前节点下的所有元素。你可以使用例如 getChildNodes
和 getElementById
方法来查询这些元素,项目中还 记录了 其他查询方法。
在这个示例中,每个元素就是配置的键。而配置的值储存在元素的内容中,你可以使用 .getTextContent
方法来提取出配置的值。
在你的 IDE 中运行代码(或者运行编译后的二进制文件):
$ java ./DemoXMLParser.java Property = window Theme = Dark Fullscreen = 0 Icon set = Tango
下面是完整的代码示例:
package myConfigParser; import java.io.File; import java.io.IOException; import java.nio.file.Path; import java.nio.file.Paths; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import javax.xml.parsers.ParserConfigurationException; import org.w3c.dom.Document; import org.w3c.dom.Element; import org.w3c.dom.NamedNodeMap; import org.w3c.dom.Node; import org.w3c.dom.NodeList; import org.xml.sax.SAXException; public class ConfigParser { public static void main(String[] args) { Path configPath = Paths.get(System.getProperty("user.home"), ".config", "DemoXMLParser"); File configFile = new File(configPath.toString(), "myconfig.xml"); DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); DocumentBuilder builder = null; try { builder = factory.newDocumentBuilder(); } catch (ParserConfigurationException e) { e.printStackTrace(); } Document doc = null; try { doc = builder.parse(configFile); } catch (SAXException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } doc.getDocumentElement().normalize(); NodeList nodes = doc.getElementsByTagName("window"); for (int i = 0; i < nodes.getLength(); i++) { Node mynode = nodes.item(i); System.out.println("Property = " + mynode.getNodeName()); if (mynode.getNodeType() == Node.ELEMENT_NODE) { Element myelement = (Element) mynode; System.out.println("Theme = " + myelement.getElementsByTagName("theme").item(0).getTextContent()); System.out.println("Fullscreen = " + myelement.getElementsByTagName("fullscreen").item(0).getTextContent()); System.out.println("Icon set = " + myelement.getElementsByTagName("icons").item(0).getTextContent()); } // close if } // close for } // close method } //close class
用户时不时地会改变某个偏好项,这时候 org.w3c.dom
库就可以帮助你更新某个 XML 元素的内容。你只需要选择这个 XML 元素,就像你读取它时那样。不过,此时你不再使用 .getTextContent
方法,而是使用 .setTextContent
方法。
updatePref = myelement.getElementsByTagName("fullscreen").item(0); updatePref.setTextContent("1"); System.out.println("Updated fullscreen to " + myelement.getElementsByTagName("fullscreen").item(0).getTextContent());
这么做会改变应用程序内存中的 XML 文档,但是还没有把数据写回到磁盘上。配合使用 javax
和 w3c
库,你就可以把读取到的 XML 内容写回到配置文件中。
TransformerFactory transformerFactory = TransformerFactory.newInstance(); Transformer xtransform; xtransform = transformerFactory.newTransformer(); DOMSource mydom = new DOMSource(doc); StreamResult streamResult = new StreamResult(configFile); xtransform.transform(mydom, streamResult);
这么做会没有警告地写入转换后的数据,并覆盖掉之前的配置。
下面是完整的代码,包括更新 XML 的操作:
package myConfigParser; import java.io.File; import java.io.IOException; import java.nio.file.Path; import java.nio.file.Paths; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import javax.xml.parsers.ParserConfigurationException; import javax.xml.transform.Transformer; import javax.xml.transform.TransformerException; import javax.xml.transform.TransformerFactory; import javax.xml.transform.dom.DOMSource; import javax.xml.transform.stream.StreamResult; import org.w3c.dom.Document; import org.w3c.dom.Element; import org.w3c.dom.Node; import org.w3c.dom.NodeList; import org.xml.sax.SAXException; public class ConfigParser { public static void main(String[] args) { Path configPath = Paths.get(System.getProperty("user.home"), ".config", "DemoXMLParser"); File configFile = new File(configPath.toString(), "myconfig.xml"); DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); DocumentBuilder builder = null; try { builder = factory.newDocumentBuilder(); } catch (ParserConfigurationException e) { // TODO Auto-generated catch block e.printStackTrace(); } Document doc = null; try { doc = builder.parse(configFile); } catch (SAXException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } doc.getDocumentElement().normalize(); Node updatePref = null; // NodeList nodes = doc.getChildNodes(); NodeList nodes = doc.getElementsByTagName("window"); for (int i = 0; i < nodes.getLength(); i++) { Node mynode = nodes.item(i); System.out.println("Property = " + mynode.getNodeName()); if (mynode.getNodeType() == Node.ELEMENT_NODE) { Element myelement = (Element) mynode; System.out.println("Theme = " + myelement.getElementsByTagName("theme").item(0).getTextContent()); System.out.println("Fullscreen = " + myelement.getElementsByTagName("fullscreen").item(0).getTextContent()); System.out.println("Icon set = " + myelement.getElementsByTagName("icons").item(0).getTextContent()); updatePref = myelement.getElementsByTagName("fullscreen").item(0); updatePref.setTextContent("2"); System.out.println("Updated fullscreen to " + myelement.getElementsByTagName("fullscreen").item(0).getTextContent()); } // close if }// close for // write DOM back to the file TransformerFactory transformerFactory = TransformerFactory.newInstance(); Transformer xtransform; DOMSource mydom = new DOMSource(doc); StreamResult streamResult = new StreamResult(configFile); try { xtransform = transformerFactory.newTransformer(); xtransform.transform(mydom, streamResult); } catch (TransformerException e) { e.printStackTrace(); } } // close method } //close class
编写配置文件看上去是一个还挺简单的任务。一开始,你可能会用一个简单的文本格式,因为你的应用程序只要寥寥几个配置项而已。但是,随着你引入了更多的配置项,读取或者写入错误的数据可能会给你的应用程序带来意料之外的错误。一种帮助你保持配置过程安全、不出错的方法,就是使用类似 XML 的规范格式,然后依靠你用的编程语言的内置功能来处理这些复杂的事情。
这也正是我喜欢使用 Java 和 XML 的原因。每当我试图读取错误的配置值时,Java 就会提醒我。通常,这是由于我在代码中试图获取的节点,并不存在于我期望的 XML 路径中。XML 这种高度结构化的格式帮助了代码保持可靠性,这对用户和开发者来说都是有好处的。
The above is the detailed content of How to configure Java persistence XML file. For more information, please follow other related articles on the PHP Chinese website!