我正在编写修改xml文件的几个脚本。有问题的文件使用xml:lang元素。 Groovy(XmlSlurper)似乎插入了一个tag0命名空间,我不介意这么做,除非它似乎打破了以后使用XmlUtil处理。
一个例子:
import groovy.xml.StreamingMarkupBuilder
import groovy.xml.XmlUtil
String source = """<?xml version='1.0' encoding='UTF-8'?>
<root>
<one xml:lang="en">First</one>
<one xml:lang="de">Second</one>
</root>
"""
def root = new XmlSlurper().parseText(source).declareNamespace(xml: "http://www.w3.org/XML/1998/namespace")
println root
String xml = new StreamingMarkupBuilder().bind{
mkp.xmlDeclaration()
out << root
}
println xml
println XmlUtil.serialize(xml)
结果
[Fatal Error] :2:44: The value of the attribute "prefix="xmlns",localpart="tag0",rawname="xmlns:tag0"" is invalid. Prefixed namespace bindings may not be empty.
默认情况下,xml:namespace应该存在,我尝试使用.declareNamespace()添加它,但它似乎没有帮助。我觉得我错过了一些显而易见的东西,但Google无法告诉我它是什么。
答案 0 :(得分:8)
几年前发现了this thread,其中说:
问题是原始文档使用默认命名空间。
SMB通常不使用默认命名空间,因此它会发明一个标记并使用它来显式标记命名空间中的每个元素。就XML解析器而言,如何指示命名空间并不重要。但是,为什么有时需要使用默认命名空间,这是美化的原因。
如果将
mkp.declareNamespace("": "http://java.sun.com/xml/ns/j2ee")
作为构建器闭包中的第一行,则应获得所需的输出。
但是,这似乎不起作用
我找到的唯一解决方案是让Slurper忽略命名空间和验证;
def root = new XmlSlurper(false,false).parseText(source)
答案 1 :(得分:2)
将默认命名空间设置为空标记对我有用(没有&#34; tag0&#34;已添加)。 我使用默认的XmlSlurper构造函数来进行工作命名空间和验证,例如:
def root = new XmlSlurper().parseText(source).declareNamespace(xml: "http://www.w3.org/XML/1998/namespace")
绑定时,声明空命名空间:
def writer = new StreamingMarkupBuilder().bind {
mkp.declareNamespace("": "") //get rid of "tag0"
mkp.declareNamespace(xml: "http://www.w3.org/XML/1998/namespace")
mkp.yield root
}