文字列のXMLからorg.w3c.dom.Documentをロードするにはどうすればよいですか?


103

文字列の完全なXMLドキュメントがあり、Documentオブジェクトが必要です。Googleはあらゆる種類のゴミを発生させます。最も簡単な解決策は何ですか?(Java 1.5の場合)

解決策Matt McMinnのおかげで、この実装に落ち着きました。それは私にとって適切なレベルの入力の柔軟性と例外の細分性を持っています。(エラーが不正なXMLから発生したのSAXExceptionか-または単にIOが悪かったのかを知るのは良いことですIOException

public static org.w3c.dom.Document loadXMLFrom(String xml)
    throws org.xml.sax.SAXException, java.io.IOException {
    return loadXMLFrom(new java.io.ByteArrayInputStream(xml.getBytes()));
}

public static org.w3c.dom.Document loadXMLFrom(java.io.InputStream is) 
    throws org.xml.sax.SAXException, java.io.IOException {
    javax.xml.parsers.DocumentBuilderFactory factory =
        javax.xml.parsers.DocumentBuilderFactory.newInstance();
    factory.setNamespaceAware(true);
    javax.xml.parsers.DocumentBuilder builder = null;
    try {
        builder = factory.newDocumentBuilder();
    }
    catch (javax.xml.parsers.ParserConfigurationException ex) {
    }  
    org.w3c.dom.Document doc = builder.parse(is);
    is.close();
    return doc;
}

解決策を修正できるとよいでしょう。String.getByptesとInputStreamを使用すると、i18nの問題が発生します。私の友人の一人がここからコードを取得しましたが、これは間違っています。findbugsが問題を検出したことは幸運です。エリクソンが提供する正しいソリューションは、InputSourceを使用することです。
Kenneth Xu

回答:


80

これは私にとってJava 1.5で機能します-読みやすさのために特定の例外を削除しました。

import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.DocumentBuilder;
import org.w3c.dom.Document;
import java.io.ByteArrayInputStream;

public Document loadXMLFromString(String xml) throws Exception
{
    DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();

    factory.setNamespaceAware(true);
    DocumentBuilder builder = factory.newDocumentBuilder();

    return builder.parse(new ByteArrayInputStream(xml.getBytes()));
}

28
sylvarkingの回答で述べたように、このコードはgetBytes()エンコードを考慮せずに使用します。
McDowell、

2
エリクソンの答えですか?それとも彼は自分のプロファイルの名前を変更しましたか?
rogerdpack、

1
キャストしてはいけませんreturn (Document) builder.parse(new ByteArrayInputStream(xml.getBytes()));か?
InfantPro'Aravind '

150

あらあら!

で指定された文字エンコードString(デフォルトではUTF-8)を無視するため、このコードには潜在的に深刻な問題があります。String.getBytes()プラットフォームを呼び出すと、デフォルトのエンコーディングを使用してUnicode文字がバイトにエンコードされます。したがって、パーサーは実際にEBCDICまたは何かを取得しているときに、UTF-8データを取得していると考える可能性があります。

代わりに、次のように、Readerで構築できるInputSourceを使用するparseメソッドを使用します。

import java.io.StringReader;
import org.xml.sax.InputSource;

        return builder.parse(new InputSource(new StringReader(xml)));

大したことではないように思えるかもしれませんが、文字エンコーディングの問題を知らないと、y2kのような油断のならないコードの腐敗につながります。


3
とてもシンプルですが、Googleのソリューションはとらえどころがありません。ありがとう+1
pat8719

6
今では、受け入れられた回答をコピーして貼り付けるだけではなく、読み通す必要があることに気づきました。
Vitaly Sazanovich 2013年

1
驚くばかり!次の設定ファイルを使用してJDK8に私たちの命を救いました。encoding= ISO-8859_1、javax.servlet.request.encoding = UTF-8 PS正しいというラベルの付いた答えが機能しませんでした
kosta5

9

同様の問題がありましたが、ドキュメントではなくNodeListが必要だったことを除けば、これが私が思いついたものです。これは以前とほとんど同じソリューションであり、ルート要素をNodeListとして取得するように拡張され、文字エンコーディングの問題ではなく、InputSourceを使用するというエリクソンの提案を使用しています。

private String DOC_ROOT="root";
String xml=getXmlString();
Document xmlDoc=loadXMLFrom(xml);
Element template=xmlDoc.getDocumentElement();
NodeList nodes=xmlDoc.getElementsByTagName(DOC_ROOT);

public static Document loadXMLFrom(String xml) throws Exception {
        InputSource is= new InputSource(new StringReader(xml));
        DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
        factory.setNamespaceAware(true);
        DocumentBuilder builder = null;
        builder = factory.newDocumentBuilder();
        Document doc = builder.parse(is);
        return doc;
    }

1

JavaでXMLを操作するために、私は常にTransformer APIを使用する傾向があります。

import javax.xml.transform.Source;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMResult;
import javax.xml.transform.stream.StreamSource;

public static Document loadXMLFrom(String xml) throws TransformerException {
    Source source = new StreamSource(new StringReader(xml));
    DOMResult result = new DOMResult();
    TransformerFactory.newInstance().newTransformer().transform(source , result);
    return (Document) result.getNode();
}   
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.