org.w3c.dom.Document を XML から文字列としてロードするにはどうすればよいですか?
質問
文字列内の完全な XML ドキュメントがあり、 Document
物体。Google はあらゆる種類のゴミを見つけます。最も簡単な解決策は何ですか?(Java 1.5の場合)
解決 おかげで マット・マクミン, この実装に落ち着きました。私にとって、適切なレベルの入力の柔軟性と例外の粒度が備わっています。(エラーの原因が不正な 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;
}
解決
これは 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()));
}
他のヒント
おっと!
このコードは、 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 に似た陰湿なコード腐敗が発生します。
ドキュメントではなく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;
}
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();
}