Как мне загрузить org.w3c.dom.Document из 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;
}

person Frank Krueger    schedule 28.08.2008    source источник
comment
Было бы неплохо, если бы можно было исправить решение. Использование String.getByptes и InputStream создает проблемы с i18n. Один из моих друзей получил код отсюда как есть, что неверно. К счастью, findbugs обнаружил проблему. Правильное решение, предлагаемое erickson, - использовать InputSource.   -  person Kenneth Xu    schedule 16.08.2013


Ответы (4)


У меня это работает в 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()));
}
person Matt McMinn    schedule 28.08.2008
comment
Как отмечено в ответе sylvarking, этот код использует getBytes() без учета кодировки. - person McDowell; 24.11.2009
comment
ты имеешь в виду ответ Эриксона? а может он переименовал свой профиль? - person rogerdpack; 05.12.2012
comment
разве не должно быть литья return (Document) builder.parse(new ByteArrayInputStream(xml.getBytes())); ?? - person InfantPro'Aravind'; 16.01.2013

Эй, там!

В этом коде есть потенциально серьезная проблема, поскольку он игнорирует кодировку символов, указанную в String (по умолчанию это UTF-8). Когда вы вызываете String.getBytes(), для кодирования символов Юникода в байты используется кодировка по умолчанию. Итак, синтаксический анализатор может подумать, что он получает данные UTF-8, хотя на самом деле он получает EBCDIC или что-то не очень красивое!

Вместо этого используйте метод синтаксического анализа, который принимает InputSource, который можно создать с помощью Reader, например:

import java.io.StringReader;
import org.xml.sax.InputSource;
…
        return builder.parse(new InputSource(new StringReader(xml)));

Это может показаться не таким уж большим делом, но незнание проблем с кодировкой символов приводит к коварному гниению кода, подобному y2k.

person erickson    schedule 28.08.2008
comment
Такое простое, но столь неуловимое решение в Google. Спасибо +1 - person pat8719; 16.02.2012
comment
Теперь я понимаю, что мне нужно не просто скопировать и вставить принятый ответ, а лучше прочитать. - person Vitaly Sazanovich; 13.11.2013
comment
Потрясающий! Спас нам жизнь на JDK8 с помощью следующего установочного файла. Encoding = ISO-8859_1, javax.servlet.request.encoding = UTF-8 PS ответ, помеченный как правильный, не сработал для нас - person kosta5; 12.04.2017

У меня была аналогичная проблема, за исключением того, что мне нужен был 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;
    }
person shsteimer    schedule 05.09.2008

Для управления 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();
}   
person Xavier Dury    schedule 05.08.2014