<h4>Choisir un format </h4>
<p>Écrire un fichier de configuration est une chose assez compliquée. J'ai essayé d'enregistrer les éléments de configuration dans un fichier texte séparé par des virgules et j'ai essayé d'enregistrer les éléments de configuration dans YAML et XML très détaillés. Pour les fichiers de configuration, le plus important est d'avoir de la cohérence et de la régularité. Ils permettent d'écrire du code facilement et rapidement et d'analyser les données du fichier de configuration en même temps, lorsque l'utilisateur décide d'apporter des modifications, il est facile de les enregistrer ; eux et mettre à jour la configuration. </p>
<p>Il existe actuellement plusieurs formats de fichiers de configuration populaires. Pour les formats de fichiers de configuration les plus courants, Java dispose de bibliothèques correspondantes. Dans cet article, j'utiliserai le format XML. Pour certains projets, vous pouvez choisir d'utiliser XML car l'une de ses fonctionnalités remarquables est sa capacité à fournir une multitude de métadonnées pertinentes pour les données qu'il contient, tandis que pour d'autres projets, vous ne pouvez pas choisir XML en raison de sa verbosité. Travailler avec XML en Java est très simple car il inclut par défaut de nombreuses bibliothèques XML robustes. </p>
<h4>Bases de XML</h4>
<p>Discuter de XML est un sujet important. J'ai un livre sur XML de plus de 700 pages. Heureusement, travailler avec XML ne nécessite pas une connaissance approfondie de ses nombreuses fonctionnalités. Tout comme HTML, XML est un langage de balisage en couches avec des balises d'ouverture et de fermeture, chacune pouvant contenir zéro ou plusieurs données. Voici un exemple simple d'extrait de XML : </p><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:xml;"><xml>
<node>
<element>Penguin</element>
</node>
</xml></pre><div class="contentsignin">Copier après la connexion</div></div><p> Dans cet exemple autodescriptif, l'analyseur XML utilise les concepts suivants : </p><ul class=" list-paddingleft-2">#🎜🎜 #<li>Document Document : La balise <code><xml></code> marque le début d'un document et la balise <code></xml></code> marque la fin de ce document. <p></li><code><xml></code> 标签标志着一个 文档 的开始,<code></xml></code> 标签标志着这个文档的结束。</p></li><li><p>节点Node:<code><node></code> 标签代表了一个 节点。</p></li><li><p>元素Element:<code><element>Penguin</element></code> 中,从开头的 <code><</code> 到最后的 <code>></code> 表示了一个 元素。</p></li><li><p>内容Content: 在 <code><element></code> 元素里,字符串 <code>Penguin</code> 就是 内容。</p></li></ul><p>不管你信不信,只要了解了以上几个概念,你就可以开始编写、解析 XML 文件了。</p><h4>创建一个示例配置文件</h4><p>要学习如何解析 XML 文件,只需要一个极简的示例文件就够了。假设现在有一个配置文件,里面保存的是关于一个图形界面窗口的属性:</p><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:xml;"><xml>
<window>
<theme>Dark</theme>
<fullscreen>0</fullscreen>
<icons>Tango</icons>
</window>
</xml></pre><div class="contentsignin">Copier après la connexion</div></div><p>创建一个名为 <code>~/.config/DemoXMLParser</code> 的目录:</p><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:bash;">$ mkdir ~/.config/DemoXMLParser</pre><div class="contentsignin">Copier après la connexion</div></div><p>在 Linux 中,<code>~/.config</code> 目录是存放配置文件的默认位置,这是在 自由桌面工作组 的规范中定义的。如果你正在使用一个不遵守 自由桌面工作组Freedesktop标准的操作系统,你也仍然可以使用这个目录,只不过你需要自己创建这些目录了。</p><p>复制 XML 的示例配置文件,粘贴并保存为 <code>~/.config/DemoXMLParser/myconfig.xml</code> 文件。</p><h4>使用 Java 解析 XML</h4><p>如果你是 Java 的初学者,你可以先阅读我写的 面向 Java 入门开发者的 7 个小技巧。一旦你对 Java 比较熟悉了,打开你最喜爱的集成开发工具(IDE),创建一个新工程。我会把我的新工程命名为 <code>myConfigParser</code>。</p><p>刚开始先不要太关注依赖导入和异常捕获这些,你可以先尝试用 <code>javax</code> 和 <code>java.io</code> 包里的标准 Java 扩展来实例化一个解析器。如果你使用了 IDE,它会提示你导入合适的依赖。如果没有,你也可以在文章稍后的部分找到完整的代码,里面就有完整的依赖列表。</p><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">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();</pre><div class="contentsignin">Copier après la connexion</div></div><p>这段示例代码使用了 <code>java.nio.Paths</code> 类来找到用户的主目录,然后在拼接上默认配置文件的路径。接着,它用 <code>java.io.File</code> 类来把配置文件定义为一个 <code>File</code> 对象。</p><p>紧接着,它使用了 <code>javax.xml.parsers.DocumentBuilder</code> 和 <code>javax.xml.parsers.DocumentBuilderFactory</code> 这两个类来创建一个内部的文档构造器,这样 Java 程序就可以导入并解析 XML 数据了。</p><p>最后,Java 创建一个叫 <code>doc</code> 的文档对象,并且把 <code>configFile</code> 文件加载到这个对象里。通过使用 <code>org.w3c.dom</code></p>Node Node : <code><node></code> Le label représente un nœud. <p></li></p>#🎜🎜#Element : <code><element>Penguin</element></code> , du début <code><</code> à la finale <code>></code> représente un élément. #🎜🎜#</li>#🎜🎜##🎜🎜#Content : Dans l'élément <code><element></code>, la chaîne <code>Penguin</code> est le contenu. #🎜🎜#</li></ul>#🎜🎜#Croyez-le ou non, tant que vous comprenez les concepts ci-dessus, vous pouvez commencer à écrire et à analyser des fichiers XML. #🎜🎜##🎜🎜#Créez un exemple de fichier de configuration #🎜🎜##🎜🎜#Pour apprendre à analyser les fichiers XML, tout ce dont vous avez besoin est un exemple de fichier minimaliste. Supposons qu'il existe un fichier de configuration qui enregistre les propriétés d'une fenêtre d'interface graphique : #🎜🎜#<div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">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());
}
}</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div>#🎜🎜#Créez un répertoire nommé <code>~/.config/DemoXMLParser</code> : #🎜 🎜#<div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">$ java ./DemoXMLParser.java
Property = window
Theme = Dark
Fullscreen = 0
Icon set = Tango</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div># 🎜🎜#Sous Linux, le répertoire <code>~/.config</code> est l'emplacement par défaut pour stocker les fichiers de configuration, qui est défini dans les spécifications du Free Desktop Working Group. Si vous utilisez un système d'exploitation qui n'adhère pas aux standards Freedesktop, vous pouvez toujours utiliser ce répertoire, mais vous devrez créer vous-même ces répertoires. #🎜🎜##🎜🎜#Copiez l'exemple de fichier de configuration XML, collez-le et enregistrez-le sous le nom de fichier <code>~/.config/DemoXMLParser/myconfig.xml</code>. #🎜🎜##🎜🎜#Utiliser Java pour analyser XML#🎜🎜##🎜🎜#Si vous êtes débutant en Java, vous pouvez d'abord lire mes 7 conseils pour les développeurs Java débutants. Une fois que vous êtes plus familier avec Java, ouvrez votre outil de développement intégré (IDE) préféré et créez un nouveau projet. Je nommerai mon nouveau projet <code>myConfigParser</code>. #🎜🎜##🎜🎜#Ne prêtez pas trop attention aux importations de dépendances et à la capture d'exceptions au début. Vous pouvez essayer d'utiliser Java standard dans <code>javax</code> et <code>java.io<. /code> paquets en premier Extension pour instancier un analyseur. Si vous utilisez un IDE, il vous demandera d'importer les dépendances appropriées. Sinon, vous pouvez trouver le code complet plus loin dans l’article, qui contient une liste complète des dépendances. #🎜🎜#<div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">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</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div>#🎜🎜#Cet exemple de code utilise la classe <code>java.nio.Paths</code> pour trouver le répertoire personnel de l'utilisateur, puis découpe le chemin d'accès au fichier de configuration par défaut. Ensuite, il utilise la classe <code>java.io.File</code> pour définir le fichier de configuration en tant qu'objet <code>File</code>. #🎜🎜##🎜🎜#Ensuite, il utilise les deux classes <code>javax.xml.parsers.DocumentBuilder</code> et <code>javax.xml.parsers.DocumentBuilderFactory</code> pour créer un constructeur de document interne afin que les programmes Java puissent importer et analyser des données XML. #🎜🎜##🎜🎜#Enfin, Java crée un objet document appelé <code>doc</code> et charge le fichier <code>configFile</code> dans cet objet. Il lit et normalise les données XML à l'aide du package <code>org.w3c.dom</code>. #🎜🎜##🎜🎜#C'est essentiellement ça. En théorie, vous avez terminé le travail d’analyse des données. Cependant, l'analyse des données est de peu d'utilité si vous n'avez pas accès aux données. Alors, écrivons quelques requêtes supplémentaires pour lire les valeurs de propriétés importantes de votre configuration. #🎜🎜#<h4>使用 Java 访问 XML 的值</h4><p>从你已经读取的 XML 文档中获取数据,其实就是要先找到一个特定的节点,然后遍历它包含的所有元素。通常我们会使用多个循环语句来遍历节点中的元素,但是为了保持代码可读性,我会尽可能少地使用循环语句:</p><div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">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());
}
}</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div><p>这段示例代码使用了 <code>org.w3c.dom.NodeList</code> 类,创建了一个名为 <code>nodes</code> 的 <code>NodeList</code> 对象。这个对象包含了所有名字匹配字符串 <code>window</code> 的子节点,实际上这样的节点只有一个,因为本文的示例配置文件中只配置了一个。</p><p>紧接着,它使用了一个 <code>for</code> 循环来遍历 <code>nodes</code> 列表。具体过程是:根据节点出现的顺序逐个取出,然后交给一个 <code>if-then</code> 子句处理。这个 <code>if-then</code> 子句创建了一个名为 <code>myelement</code> 的 <code>Element</code> 对象,其中包含了当前节点下的所有元素。你可以使用例如 <code>getChildNodes</code> 和 <code>getElementById</code> 方法来查询这些元素,项目中还 记录了 其他查询方法。</p><p>在这个示例中,每个元素就是配置的键。而配置的值储存在元素的内容中,你可以使用 <code>.getTextContent</code> 方法来提取出配置的值。</p><p>在你的 IDE 中运行代码(或者运行编译后的二进制文件):</p><div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">$ java ./DemoXMLParser.java
Property = window
Theme = Dark
Fullscreen = 0
Icon set = Tango</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div><p>下面是完整的代码示例:</p><div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">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</pre><div class="contentsignin">Copier après la connexion</div></div><div class="contentsignin">Copier après la connexion</div></div><h4>使用 Java 更新 XML</h4><p>用户时不时地会改变某个偏好项,这时候 <code>org.w3c.dom</code> 库就可以帮助你更新某个 XML 元素的内容。你只需要选择这个 XML 元素,就像你读取它时那样。不过,此时你不再使用 <code>.getTextContent</code> 方法,而是使用 <code>.setTextContent</code> 方法。</p><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">updatePref = myelement.getElementsByTagName("fullscreen").item(0);
updatePref.setTextContent("1");
System.out.println("Updated fullscreen to " + myelement.getElementsByTagName("fullscreen").item(0).getTextContent());</pre><div class="contentsignin">Copier après la connexion</div></div><p>这么做会改变应用程序内存中的 XML 文档,但是还没有把数据写回到磁盘上。配合使用 <code>javax</code> 和 <code>w3c</code> 库,你就可以把读取到的 XML 内容写回到配置文件中。</p><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">TransformerFactory transformerFactory = TransformerFactory.newInstance();
Transformer xtransform;
xtransform = transformerFactory.newTransformer();
DOMSource mydom = new DOMSource(doc);
StreamResult streamResult = new StreamResult(configFile);
xtransform.transform(mydom, streamResult);</pre><div class="contentsignin">Copier après la connexion</div></div><p>这么做会没有警告地写入转换后的数据,并覆盖掉之前的配置。</p><p>下面是完整的代码,包括更新 XML 的操作:</p><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:java;">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</pre><div class="contentsignin">Copier après la connexion</div></div><h4>如何保证配置不出问题</h4>
<p>编写配置文件看上去是一个还挺简单的任务。一开始,你可能会用一个简单的文本格式,因为你的应用程序只要寥寥几个配置项而已。但是,随着你引入了更多的配置项,读取或者写入错误的数据可能会给你的应用程序带来意料之外的错误。一种帮助你保持配置过程安全、不出错的方法,就是使用类似 XML 的规范格式,然后依靠你用的编程语言的内置功能来处理这些复杂的事情。</p>
<p>这也正是我喜欢使用 Java 和 XML 的原因。每当我试图读取错误的配置值时,Java 就会提醒我。通常,这是由于我在代码中试图获取的节点,并不存在于我期望的 XML 路径中。XML 这种高度结构化的格式帮助了代码保持可靠性,这对用户和开发者来说都是有好处的。</p>
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!