ホームページ  >  記事  >  Java  >  Java 永続性 XML ファイルの構成方法

Java 永続性 XML ファイルの構成方法

WBOY
WBOY転載
2023-05-04 09:40:06926ブラウズ

形式の選択

設定ファイルの作成はかなり複雑です。構成項目をカンマ区切りのテキスト ファイルに保存してみたり、構成項目を非常に冗長な YAML および XML で保存してみたりしました。構成ファイルの場合、最も重要なことは一貫性と規則性を持つことです。これにより、コードを簡単かつ迅速に記述し、構成ファイルのデータを解析できるようになります。同時に、ユーザーが変更を加える場合は、簡単に保存できます。そして設定を更新します。

現在、一般的な構成ファイル形式がいくつかあります。最も一般的な構成ファイル形式に対応するライブラリが Java にあります。この記事では XML 形式を使用します。プロジェクトによっては、含まれるデータに関連するメタデータを豊富に提供できることがその優れた機能の 1 つであるため、XML の使用を選択する場合もありますが、他のプロジェクトでは、その冗長さのために XML を選択しない場合もあります。 Java にはデフォルトで堅牢な XML ライブラリが多数含まれているため、Java での XML の操作は非常に簡単です。

XML の基本

XML についての議論は大きなトピックです。私は 700 ページを超える XML に関する本を持っています。幸いなことに、XML の操作には、その多くの機能についての詳しい知識は必要ありません。 HTML と同様に、XML は開始タグと終了タグを備えた階層化されたマークアップ言語であり、各タグには 0 個以上のデータを含めることができます。 XML の簡単なサンプル スニペットを次に示します。

<xml>
  <node>
    <element>Penguin</element>
  </node>
</xml>

この自己記述的な例では、XML パーサーは次の概念を使用します。

  • Documentation ドキュメント: b2a0af5a8fd26276da50279a1c63a57a タグはドキュメントの始まりを示し、21118965b89073f60271ef4a3b5d3c58 タグはこのドキュメントの終わりを示します。

  • ノード ノード: 37493a7a244a2c0740719da11c0bbc72 ラベルはノードを表します。

  • 要素: 1dd643d91c792abd0a34d1b312e2f207ペンギンa24c0203f0ae689239f065103120aae7、先頭 1ea0503f97b84b66e1cfd4f600833e26 は要素を表します。

  • コンテンツ: 1dd643d91c792abd0a34d1b312e2f207 要素の文字列 Penguin がコンテンツです。

信じられないかもしれませんが、上記の概念を理解していれば、XML ファイルの作成と解析を始めることができます。

サンプル構成ファイルを作成する

XML ファイルを解析する方法を学ぶために必要なのは、最小限のサンプル ファイルだけです。グラフィカル インターフェイス ウィンドウのプロパティを保存する構成ファイルがあるとします。

<xml>
  <window>
    <theme>Dark</theme>
    <fullscreen>0</fullscreen>
    <icons>Tango</icons>
</window>
</xml>

~/.config/DemoXMLParser という名前のディレクトリを作成します。

$ mkdir ~/.config/DemoXMLParser

Linux の場合~/.config ディレクトリは、構成ファイルを保存するためのデフォルトの場所であり、Free Desktop Working Group の仕様で定義されています。 Freedesktop 標準に準拠していないオペレーティング システムを使用している場合でも、このディレクトリを使用できますが、これらのディレクトリを自分で作成する必要があります。

XML サンプル構成ファイルをコピーし、貼り付けて、~/.config/DemoXMLParser/myconfig.xml ファイルとして保存します。

Java を使用して XML を解析する

Java の初心者の方は、まず、初心者の Java 開発者向けに私が書いた 7 つのヒントを読んでください。 Java に慣れてきたら、お気に入りの統合開発ツール (IDE) を開いて新しいプロジェクトを作成します。新しいプロジェクトに myConfigParser という名前を付けます。

最初は依存関係のインポートや例外キャッチにあまり注意を払わないでください。まず、javax および java.io の標準 Java 拡張機能を使用してインスタンス化を試みることができます。 パッケージ、パーサー。 IDE を使用している場合は、適切な依存関係をインポートするように求められます。そうでない場合は、依存関係の完全なリストを含む完全なコードをこの記事の後半で見つけることができます。

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();

このサンプル コードでは、java.nio.Paths クラスを使用してユーザーのホーム ディレクトリを検索し、そのパスをデフォルトの構成ファイルに接続します。次に、java.io.File クラスを使用して、構成ファイルを File オブジェクトとして定義します。

次に、2 つのクラス javax.xml.parsers.DocumentBuilderjavax.xml.parsers.DocumentBuilderFactory を使用して内部ドキュメント構造 Converter を作成します。プログラムは XML データをインポートして解析できます。

最後に、Java は doc というドキュメント オブジェクトを作成し、configFile ファイルをこのオブジェクトにロードします。 org.w3c.dom パッケージを使用して、XML データを読み取り、正規化します。

基本的にはそれだけです。理論的には、データ分析作業は完了しました。ただし、データにアクセスできない場合、データ解析はほとんど役に立ちません。そこで、構成から重要なプロパティ値を読み取るためのクエリをさらにいくつか書いてみましょう。

使用 Java 访问 XML 的值

从你已经读取的 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

使用 Java 更新 XML

用户时不时地会改变某个偏好项,这时候 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 这种高度结构化的格式帮助了代码保持可靠性,这对用户和开发者来说都是有好处的。

以上がJava 永続性 XML ファイルの構成方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事はyisu.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。