提问者:小点点

将命名空间添加到已创建的XML文档


我正在使用String值创建一个W3C Document对象。创建Document对象后,我想在此文档的根元素中添加一个命名空间。这是我当前的代码:

Document document = builder.parse(new InputSource(new StringReader(xmlString)));
document.getDocumentElement().setAttributeNS("http://com", "xmlns:ns2", "Test");
document.setPrefix("ns2");
TransformerFactory tranFactory = TransformerFactory.newInstance();
Transformer aTransformer = tranFactory.newTransformer();
Source src = new DOMSource(document);
Result dest = new StreamResult(new File("c:\\xmlFileName.xml"));
aTransformer.transform(src, dest);

我用什么作为输入:

<product>
    <arg0>DDDDDD</arg0>
    <arg1>DDDD</arg1>
</product>

输出应该是什么样子:

<ns2:product xmlns:ns2="http://com">
    <arg0>DDDDDD</arg0>
    <arg1>DDDD</arg1>
</ns2:product>

我需要将前缀值和命名空间也添加到输入xml字符串中。如果我尝试上面的代码,我会得到这个异常:

NAMESPACE_ERR: An attempt is made to create or change an object in a way which is incorrect with regard to namespaces.

感谢您的帮助!


共3个答案

匿名用户

由于没有简单的方法来重命名根元素,我们必须用具有正确命名空间和属性的元素替换它,然后将所有原始子元素复制到其中。不需要强制命名空间声明,因为通过为元素提供正确的命名空间(URI)并设置前缀,声明将是自动的。

setAt的setPrefix替换为这个(第2,3行)

String namespace = "http://com";
String prefix = "ns2";
// Upgrade the DOM level 1 to level 2 with the correct namespace
Element originalDocumentElement = document.getDocumentElement();
Element newDocumentElement = document.createElementNS(namespace, originalDocumentElement.getNodeName());
// Set the desired namespace and prefix
newDocumentElement.setPrefix(prefix);
// Copy all children
NodeList list = originalDocumentElement.getChildNodes();
while(list.getLength()!=0) {
    newDocumentElement.appendChild(list.item(0));
}
// Replace the original element
document.replaceChild(newDocumentElement, originalDocumentElement);

在原始代码中,作者试图像这样声明一个元素命名空间:

.setAttributeNS("http://com", "xmlns:ns2", "Test");

第一个参数是属性的命名空间,因为它是一个命名空间属性,所以它需要有http://www.w3.org/2000/xmlns/URI。声明的命名空间应该进入第三个参数

.setAttributeNS("http://www.w3.org/2000/xmlns/", "xmlns:ns2", "http://com");

匿名用户

波纹管方法也适用于我,但在性能关键的情况下可能不应该使用。

  1. 将名称空间作为属性添加到文档根元素。
  2. 将文档转换为XML字符串,这一步的目的是使XML字符串中的子元素继承父元素命名空间。
  3. 现在xml字符串有名称空间。
  4. 您可以使用XML字符串再次构建文档或用于JAXB散集等。
private static String addNamespaceToXml(InputStream in)
        throws ParserConfigurationException, SAXException, IOException,
        TransformerException {
    DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
    /*
     * Must not namespace aware, otherwise the generated XML string will
     * have wrong namespace
     */
    // dbf.setNamespaceAware(true);
    DocumentBuilder db = dbf.newDocumentBuilder();
    Document document = db.parse(in);
    Element documentElement = document.getDocumentElement();
    // Add name space to root element as attribute
    documentElement.setAttribute("xmlns", "http://you_name_space");
    String xml = transformXmlNodeToXmlString(documentElement);
    return xml;
}

private static String transformXmlNodeToXmlString(Node node)
        throws TransformerException {
    TransformerFactory transFactory = TransformerFactory.newInstance();
    Transformer transformer = transFactory.newTransformer();
    StringWriter buffer = new StringWriter();
    transformer.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION, "yes");
    transformer.transform(new DOMSource(node), new StreamResult(buffer));
    String xml = buffer.toString();
    return xml;
}

匿名用户

部分从这里收集,也从上面的评论中,我能够让它工作(转换任意DOM节点并为其及其所有子节点添加前缀),因此:

  private String addNamespacePrefix(Document doc, Node node) throws TransformerException {
    Element mainRootElement = doc.createElementNS(
            "http://abc.de/x/y/z", // namespace
            "my-prefix:fake-header-element" // prefix to "register" it with the DOM so we don't get exceptions later...
    );
    List<Element> descendants = nodeListToArrayRecurse(node.getChildNodes()); // for some reason we have to grab all these before doing the first "renameNode" ... no idea why ...

    mainRootElement.appendChild(node);
    doc.renameNode(node, "http://abc.de/x/y/z", "my-prefix:" + node.getNodeName());
    descendants.stream().forEach(c -> doc.renameNode(c, "http://abc.de/x/y/z", "my-prefix:" + c.getNodeName()));
  }

  private List<Element> nodeListToArrayRecurse(NodeList entryNodes) {
    List<Element> allEntries = new ArrayList<>();
    for (int i = 0; i < entryNodes.getLength(); i++) {
      Node child = entryNodes.item(i);
      if (child.getNodeType() == Node.ELEMENT_NODE) {
        allEntries.add((Element) child);
        allEntries.addAll(nodeListToArray(child.getChildNodes())); // recurse
      } // ignore other [i.e. text] nodes https://stackoverflow.com/questions/14566596/loop-through-all-elements-in-xml-using-nodelist
    }
    return allEntries;
  }

如果它对任何人都有帮助。然后我将其转换为字符串,然后手动删除额外的标题和关闭行。多么痛苦,我一定做错了什么…