Come faccio a caricare un documento org.w3c.dom.Document da XML in una stringa?


103

Ho un documento XML completo in una stringa e vorrei un Documentoggetto. Google scopre ogni sorta di spazzatura. Qual è la soluzione più semplice? (In Java 1.5)

Soluzione Grazie a Matt McMinn , ho scelto questa implementazione. Ha il giusto livello di flessibilità di input e granularità delle eccezioni per me. (È utile sapere se l'errore proviene da XML non valido - SAXException- o solo da IO non valido - 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;
}

Sarebbe bello se tu potessi correggere la soluzione. L'uso di String.getByptes e InputStream impone problemi i18n. Uno dei miei amici ha ricevuto il codice da qui perché è sbagliato. Fortunatamente findbugs ha rilevato il problema. La soluzione corretta fornita da erickson consiste nell'usare InputSource.
Kenneth Xu

Risposte:


80

Questo funziona per me in Java 1.5: ho eliminato eccezioni specifiche per la leggibilità.

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
Come notato nella risposta di sylvarking, questo codice viene utilizzato getBytes()senza alcuna considerazione per la codifica.
McDowell

2
vuoi dire la risposta di Erickson? o forse ha ribattezzato il suo profilo?
rogerdpack

1
non dovrebbe esserci il casting return (Document) builder.parse(new ByteArrayInputStream(xml.getBytes()));?
InfantPro'Aravind '16

150

Whoa là!

C'è un problema potenzialmente serio con questo codice, perché ignora la codifica dei caratteri specificata in String(che è UTF-8 per impostazione predefinita). Quando chiami String.getBytes()la piattaforma, la codifica predefinita viene utilizzata per codificare i caratteri Unicode in byte. Quindi, il parser potrebbe pensare che stia ottenendo dati UTF-8 quando in realtà sta ottenendo EBCDIC o qualcosa del genere ... non carino!

Utilizza invece il metodo di analisi che accetta un InputSource, che può essere costruito con un Reader, in questo modo:

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

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

Potrebbe non sembrare un grosso problema, ma l'ignoranza dei problemi di codifica dei caratteri porta a un codice insidioso simile a y2k.


3
Una soluzione così semplice ma così sfuggente su Google. Grazie +1
pat8719

6
Mi rendo conto ora che non dovrei semplicemente copiare e incollare la risposta accettata, ma piuttosto leggere.
Vitaly Sazanovich

1
Eccezionale! Ci ha salvato la vita su JDK8 con il seguente file di installazione.encoding = ISO-8859_1, javax.servlet.request.encoding = UTF-8 PS la risposta etichettata come corretta non ha funzionato per noi
kosta5

9

Ho appena avuto un problema simile, tranne che avevo bisogno di un NodeList e non di un documento, ecco cosa mi è venuto in mente. È per lo più la stessa soluzione di prima, aumentata per ottenere l'elemento radice come NodeList e utilizzando il suggerimento di erickson di utilizzare invece un InputSource per problemi di codifica dei caratteri.

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

Per manipolare XML in Java, tendo sempre a utilizzare l'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();
}   
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.