Как мне загрузить org.w3c.dom.Документ из XML в виде строки?
Вопрос
У меня есть полный XML-документ в виде строки, и я хотел бы Document
объект.Google выдает всевозможный мусор.Каково самое простое решение?(В Java 1.5)
Решение Благодаря Мэтт Макминн, Я остановился на этой реализации.Для меня он обладает нужным уровнем гибкости ввода и детализации исключений.(Полезно знать, не произошла ли ошибка из-за искаженного XML - SAXException
- или просто плохой ввод - вывод - 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()
кодировка платформы по умолчанию используется для кодирования символов Юникода в байты.Таким образом, анализатор может думать, что он получает данные UTF-8, когда на самом деле он получает EBCDIC или что-то в этом роде ... не очень красиво!
Вместо этого используйте метод parse, который принимает InputSource, который может быть создан с помощью Reader, например:
import java.io.StringReader;
import org.xml.sax.InputSource;
…
return builder.parse(new InputSource(new StringReader(xml)));
Это может показаться несущественным, но незнание проблем с кодировкой символов приводит к коварной гнили кода, подобной y2k.
Только что была похожая проблема, за исключением того, что мне нужен был список узлов, а не Документ, вот что я придумал.В основном это то же решение, что и раньше, дополненное для удаления корневого элемента в виде списка узлов и использующее предложение Эриксона использовать 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;
}
Чтобы манипулировать XML в Java, я всегда предпочитаю использовать 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();
}