Como carrego um org.w3c.dom.Document de XML em uma string?
Pergunta
Tenho um documento XML completo em uma string e gostaria de um Document
objeto.O Google revela todo tipo de lixo.Qual é a solução mais simples?(Em Java 1.5)
Solução Graças a Matt McMinn, eu decidi por esta implementação.Para mim, ele tem o nível certo de flexibilidade de entrada e granularidade de exceções.(É bom saber se o erro veio de XML malformado - SAXException
- ou apenas IO ruim - 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;
}
Solução
Isso funciona para mim em Java 1.5 - eliminei exceções específicas para facilitar a leitura.
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()));
}
Outras dicas
Uau!
Há um problema potencialmente sério com este código, porque ele ignora a codificação de caracteres especificada no String
(que é UTF-8 por padrão).Quando Você ligar String.getBytes()
a codificação padrão da plataforma é usada para codificar caracteres Unicode em bytes.Então, o analisador pode pensar que está obtendo dados UTF-8 quando na verdade está obtendo EBCDIC ou algo assim... nada bonito!
Em vez disso, use o método parse que usa um InputSource, que pode ser construído com um Reader, assim:
import java.io.StringReader;
import org.xml.sax.InputSource;
…
return builder.parse(new InputSource(new StringReader(xml)));
Pode não parecer grande coisa, mas a ignorância dos problemas de codificação de caracteres leva a uma podridão insidiosa do código semelhante ao ano 2000.
Acabei de ter um problema semelhante, exceto que eu precisava de um NodeList e não de um Documento, eis o que descobri.É basicamente a mesma solução de antes, aumentada para obter o elemento raiz como um NodeList e usando a sugestão de Erickson de usar um InputSource para problemas de codificação de caracteres.
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;
}
Para manipular XML em Java, sempre costumo usar a API Transformer:
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();
}