Como carrego um org.w3c.dom.Document de XML em uma string?


103

Tenho um documento XML completo em uma string e gostaria de um Documentobjeto. O Google encontra todo tipo de lixo. Qual é a solução mais simples? (Em Java 1.5)

Solução Graças a Matt McMinn , decidi por esta implementação. Ele tem o nível certo de flexibilidade de entrada e granularidade de exceção para mim. (É bom saber se o erro veio de XML malformado SAXException- ou apenas IO inválido - 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;
}

Seria bom se você pudesse corrigir a solução. Usar String.getByptes e InputStream impõe problemas de i18n. Um amigo meu pegou o código daqui, qual está errado. Sorte que o findbugs detectou o problema. A solução correta fornecida por erickson é usar InputSource.
Kenneth Xu de

Respostas:


80

Isso funciona para mim no 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()));
}

28
Conforme observado na resposta do sylvarking, este código usa getBytes()sem consideração para codificação.
McDowell

2
você quer dizer a resposta de Erickson? ou talvez ele renomeou seu perfil?
rogerdpack

1
não deveria haver elenco return (Document) builder.parse(new ByteArrayInputStream(xml.getBytes()));??
InfantPro'Aravind '16 de

150

Uau!

Há um problema potencialmente sério com esse código, porque ele ignora a codificação de caracteres especificada no String(que é UTF-8 por padrão). Quando você chama String.getBytes()a plataforma, a codificação padrão é 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 ... não é bonito!

Em vez disso, use o método de análise que usa um InputSource, que pode ser construído com um Reader, como este:

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 de código insidiosa semelhante ao y2k.


3
Uma solução tão simples, mas tão elusiva no Google. Obrigado +1
pat8719

6
Percebo agora que não devo apenas copiar e colar a resposta aceita, mas sim ler.
Vitaly Sazanovich

1
Impressionante! Salvou nossas vidas no JDK8 com a seguinte configuração file.encoding = ISO-8859_1, javax.servlet.request.encoding = UTF-8 PS a resposta rotulada como correta não funcionou para nós
kosta5

9

Acabei de ter um problema semelhante, exceto que precisava de uma NodeList e não de um Documento, aqui está o que eu vim. É 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 em vez de 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;
    }

1

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();
}   
Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.