ฉันจะโหลด org.w3c.dom.Document จาก XML ในสตริงได้อย่างไร


103

ฉันมีเอกสาร XML ที่สมบูรณ์ในสตริงและต้องการDocumentวัตถุ Google เปลี่ยนขยะทุกประเภท ทางออกที่ง่ายที่สุดคืออะไร? (ใน Java 1.5)

วิธีแก้ไขต้องขอบคุณMatt McMinnฉันได้ตัดสินการใช้งานนี้แล้ว มีระดับความยืดหยุ่นในการป้อนข้อมูลและความละเอียดของข้อยกเว้นสำหรับฉัน (เป็นเรื่องที่ดีที่จะทราบว่าข้อผิดพลาดมาจาก XML ที่ผิดรูปแบบ - SAXException- หรือ IO - ไม่ดี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;
}

คงจะดีไม่น้อยหากคุณสามารถแก้ปัญหาได้ การใช้ String.getByptes และ InputStream ทำให้เกิดปัญหา i18n เพื่อนของฉันคนหนึ่งได้รหัสจากที่นี่ซึ่งเป็นสิ่งที่ไม่ถูกต้อง โชคดีที่ findbugs ตรวจพบปัญหา วิธีแก้ไขที่ถูกต้องโดย erickson คือการใช้ InputSource
Kenneth Xu

คำตอบ:


80

สิ่งนี้ใช้ได้กับฉันใน 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()));
}

28
ดังที่ระบุไว้ในคำตอบของ sylvarking รหัสนี้ใช้getBytes()โดยไม่คำนึงถึงการเข้ารหัส
McDowell

2
คุณหมายถึงคำตอบของ erickson หรือเปล่า? หรือบางทีเขาอาจเปลี่ยนชื่อโปรไฟล์ของเขา?
rogerdpack

1
ไม่ควรหล่อreturn (Document) builder.parse(new ByteArrayInputStream(xml.getBytes()));??
InfantPro'Aravind '16

150

โอ้โฮ!

รหัสนี้อาจมีปัญหาร้ายแรงเนื่องจากไม่สนใจการเข้ารหัสอักขระที่ระบุในString(ซึ่งเป็น UTF-8 โดยค่าเริ่มต้น) เมื่อคุณโทรString.getBytes()ใช้การเข้ารหัสเริ่มต้นของแพลตฟอร์มจะใช้ในการเข้ารหัสอักขระ Unicode เป็นไบต์ ดังนั้นโปรแกรมแยกวิเคราะห์อาจคิดว่ามันได้รับข้อมูล UTF-8 เมื่อในความเป็นจริงมันได้รับ EBCDIC หรืออะไรบางอย่าง ... ไม่สวยเลย!

ให้ใช้วิธีการแยกวิเคราะห์ที่ใช้ InputSource แทนซึ่งสามารถสร้างด้วย Reader ได้ดังนี้:

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

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

อาจดูเหมือนไม่ใช่เรื่องใหญ่ แต่การไม่รู้ปัญหาการเข้ารหัสอักขระทำให้เกิดการเน่าของรหัสที่ร้ายกาจคล้ายกับ y2k


3
วิธีแก้ปัญหาที่เรียบง่าย แต่เข้าใจยากใน Google ขอบคุณ +1
pat8719

6
ตอนนี้ฉันรู้แล้วว่าฉันไม่ควรแค่คัดลอกและวางคำตอบที่ยอมรับ แต่ควรอ่าน
Vitaly Sazanovich

1
สุดยอด! ช่วยชีวิตเราบน JDK8 ด้วยไฟล์ติดตั้งต่อไปนี้ encoding = ISO-8859_1, javax.servlet.request.encoding = UTF-8 PS คำตอบที่ระบุว่าถูกต้องไม่ได้ผลสำหรับเรา
kosta5

9

เพิ่งมีปัญหาที่คล้ายกันยกเว้นว่าฉันต้องการ NodeList ไม่ใช่เอกสารนี่คือสิ่งที่ฉันคิดขึ้นมา ส่วนใหญ่เป็นวิธีแก้ปัญหาเดียวกันกับก่อนหน้านี้เพิ่มขึ้นเพื่อให้องค์ประกอบรูทลงมาเป็น NodeList และใช้คำแนะนำของ erickson ในการใช้ 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;
    }

1

ในการจัดการ 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();
}   
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.