14

我正在尝试将 ResultSet 转换为 XML 文件。我首先使用这个例子进行序列化。

import  org.w3c.dom.bootstrap.DOMImplementationRegistry;
import  org.w3c.dom.Document;
import  org.w3c.dom.ls.DOMImplementationLS;
import  org.w3c.dom.ls.LSSerializer;

...

DOMImplementationRegistry registry = DOMImplementationRegistry.newInstance();

DOMImplementationLS impl = 
    (DOMImplementationLS)registry.getDOMImplementation("LS");

...     

LSSerializer writer = impl.createLSSerializer();
String str = writer.writeToString(document);

完成这项工作后,我尝试验证我的 XML 文件,但出现了一些警告。一个关于没有文档类型的问题。所以我尝试了另一种方法来实现这一点。我遇到了 Transformer 类。这个类让我设置编码、文档类型等。

以前的实现支持自动命名空间修复。以下没有。

private static Document toDocument(ResultSet rs) throws Exception {   
    DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
    factory.setNamespaceAware(true);
    DocumentBuilder builder = factory.newDocumentBuilder();
    Document doc = builder.newDocument();

    URL namespaceURL = new URL("http://www.w3.org/2001/XMLSchema-instance");
    String namespace = "xmlns:xsi="+namespaceURL.toString();

    Element messages = doc.createElementNS(namespace, "messages");
    doc.appendChild(messages);

    ResultSetMetaData rsmd = rs.getMetaData();
    int colCount = rsmd.getColumnCount();

    String attributeValue = "true";
    String attribute = "xsi:nil";

    rs.beforeFirst();

    while(rs.next()) {
        amountOfRecords = 0;
        Element message = doc.createElement("message");
        messages.appendChild(message);

        for(int i = 1; i <= colCount; i++) {

            Object value = rs.getObject(i);
            String columnName = rsmd.getColumnName(i);

            Element messageNode = doc.createElement(columnName);

            if(value != null) {
                messageNode.appendChild(doc.createTextNode(value.toString()));
            } else {
                messageNode.setAttribute(attribute, attributeValue);
            }
            message.appendChild(messageNode);
        }
        amountOfRecords++;
    }
    logger.info("Amount of records archived: " + amountOfRecords);

    TransformerFactory tff = TransformerFactory.newInstance();
    Transformer tf = tff.newTransformer();
    tf.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION, "yes");
    tf.setOutputProperty(OutputKeys.ENCODING, "UTF-8");
    tf.setOutputProperty(OutputKeys.INDENT, "yes");

    BufferedWriter bf = createFile();
    StreamResult sr = new StreamResult(bf);
    DOMSource source = new DOMSource(doc);
    tf.transform(source, sr);

    return doc;
}

当我在测试之前的实现时,我得到了一个 TransformationException: Namespace for prefix 'xsi' has not been declared。如您所见,我尝试将带有 xsi 前缀的命名空间添加到文档的根元素中。经过测试,我仍然得到了异常。设置命名空间及其前缀的正确方法是什么?

编辑:我在第一个实现中遇到的另一个问题是 XML 文档中的最后一个元素没有最后三个结束标记。

4

3 回答 3

34

在 namespaceAware 文档上设置节点的正确方法是使用:

rootNode.createElementNS("http://example/namespace", "PREFIX:aNodeName");

因此,您可以将“PREFIX”替换为您自己的自定义前缀,并将“aNodeName”替换为您的节点名称。为了避免每个节点都有自己的命名空间声明,您可以将命名空间定义为根节点上的属性,如下所示:

rootNode.setAttribute("xmlns:PREFIX", "http://example/namespace");

请务必设置:

documentBuilderFactory.setNamespaceAware(true)

否则你没有命名空间意识。

于 2012-10-24T11:41:43.673 回答
10

请注意,使用 setAttribute 设置 xmlns 前缀是错误的。如果你想签署你的 DOM,你必须使用 setAttributeNS: element.setAttributeNS("http://www.w3.org/2000/xmlns/", "xmlns:PREFIX", "http://example/namespace");

于 2014-10-16T07:18:30.090 回答
9

您还没有在根节点中添加命名空间声明;你刚刚在命名空间中声明了根节点,两个完全不同的东西。在构建 DOM 时,您需要引用每个相关节点上的命名空间。换句话说,当你添加你的属性时,你需要定义它的命名空间(例如,setAttributeNS)。

旁注:虽然 XML 名称空间看起来像 URL,但实际上并非如此。这里不需要使用 URL 类。

于 2012-05-14T14:20:06.843 回答