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

时间:2012-06-21 07:49:46

标签: java xml dom

我正在使用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 个答案:

答案 0 :(得分:24)

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

setAttributesetPrefix替换为此(第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。声明的命名空间应该进入第3个参数

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

答案 1 :(得分:0)

Bellow方法也适用于我,但可能不应该用于性能危急情况。

  1. 将名称空间添加到文档根元素作为属性。
  2. 将文档转换为XML字符串。此步骤的目的是使XML字符串中的子元素继承父元素名称空间。
  3. 现在xml字符串有名称空间。
  4. 您可以使用XML字符串再次构建文档或用于JAXB unmarshal等。
  5. 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;
    }
    

答案 2 :(得分:0)

此处中部分收集,以及从上面的评论中,我能够使其正常工作(将任意DOM节点,并为其及其所有子代添加前缀),

 私有字符串addNamespacePrefix(文档doc,节点节点)引发TransformerException {
    元素mainRootElement = doc.createElementNS(
            “ http://abc.de/x/y/z”,//命名空间
            “ my-prefix:fake-header-element” //前缀以在DOM中“注册”它,因此以后我们不会再出现异常...
    );
    List 后代= nodeListToArrayRecurse(node.getChildNodes()); //由于某种原因,在执行第一个“ renameNode”之前,我们必须抓住所有这些……不知道为什么……

    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()));
  }

  私人List  nodeListToArrayRecurse(NodeList entryNodes){
    List  allEntries =新的ArrayList <>();
    for(int i = 0; i  

如果它可以帮助任何人。然后,我将其转换为字符串,然后手动删除多余的标题和结束行。