删除多余的属性 xmlns=""
作者:快乐笛子 来源:快乐笛子blog 时间:2010-08-24 18:41:00
很早前就遇到这个空值的属性,它既出现在 html 文档中,也出现在 xml 中,一直都回避,放之任之,反正也不影响文档的正确性。隐隐约约过了大半年,终于有一天下定决心,一定要把这个问题彻彻底底搞个透,认真一分析,却发现这是一个 xml 命名空间基础知识,不是一个需要捋起两管衣袖再加深呼吸两口才能解决的问题。
首先来重现此属性,现有的XML数据(people.xml):
<ROOT xmlns\="http://www.example.com">
<NAME>张三</NAME>
</ROOT>(注意上面的xmlns后面带有一个反斜杠,是为了防止语法高亮组件SyntaxHighlighter异常而加的,请透明视之,以下同)程序给root增加一个age节点用于描述“张三”的年龄(VBScript):
Dim dom Set dom = CreateObject("MSXML2.DOMDocument") dom.async = False dom.load "people.xml" Dim age Set age = dom.createElement("age") age.text = 25 dom.documentElement.appendChild age dom.save "people.xml" 修改后的 people.xml 如下:
<ROOT xmlns\="http://www.example.com">
<NAME>张三</NAME>
<AGE xmlns="">25</AGE>
</ROOT> 文档的root节点定义了一个默认的命名空间 http://www.example.com,即root元素的所有子孙元素都在此空间之下(除非它们另行定义有空间)。程序创建 age 节点时没有指明此节点具有哪个空间,即 age 的命名空间为空值,空值不等于 http://www.example.com,相当于 age 有自己的私有空间,它与父辈的空间不同,理所当然地会出现 xmlns 属性。要去除此属性有两个方法:
去除 age 祖先节点默认的命名空间
创建 age 节点时指明其空间为祖先定下的默认空间
第一种方案一般是不会被采纳的,为了儿孙辈不太可能废掉祖先定下的规矩。第二种方案最简单:使用 createNode 方法替换 createElement 方法。
Set age = dom.createNode(1,"age","http://www.example.com") 效果立现。
除了程序会遇到命名空间继承的问题,xsl 模板中创建的元素也有此问题。比如创建元素:
<?xml:namespace prefix = xsl /><xsl:element name="a">
<xsl:attribute name="href">http://www.example.com</xsl:attribute>
<xsl:value-of select="'Hello world'"></xsl:value-of>
</xsl:element>该 XSL 模板是由 Dreamweaver 创建,在经历 XSL 处理器转换后,输出的 a 元素同样也包含一个 xmlns="" 属性。根本原因还是创建的元素与其祖先的命名空间不一致。用 Dreamweaver 创建一个空的 XSLT(整页) 页面,其根节点
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
只声明了名为 xsl 的空间,它用于实现所有的xslt元素: <xsl:xxxx />, 因此用 xsl 的方法创建出来的元素,它们的空间自然就在根节点声明空间之下,该死的是根节点并没有定义有默认空间,所以 xsl 方法创建的元素最终都将有一个 xmlns="" 属性。
这里需要注意到 XSLT 模板里面的 html 元素为了适合 w3c 所推荐的 xhtml 标准,它声明了一个默认空间
<html xmlns="http://www.w3.org/1999/xhtml">
这个空间是对最终的 html 代码有效的,它对 xslt 没有任何作用。
解决的方法同样有两种:
给 xslt 页面的根节点 xsl:stylesheet 声明默认空间,令它与 html 元素的默认空间值一致
使用 xsl 方法创建元素时指明其空间,只要空间与默认空间一致,就不会出现 xmlns=""
这里推荐采用第一种方案,因为只需在 xsl 文档的根节点声明了与 html 元素一致的默认命名空间,用 xsl 方法创建的元素就与祖先的空间一致,就不会出现 xmlns="",一劳永逸。
方法二就需要每次都使用
<?xml:namespace prefix = xsl />
<xsl:element name="xx" namespace="http://www.w3.org/1999/xhtml">
...
</xsl:element>
明显第一种方案胜出。
继续深入,怎样让 Dreamweaver 创建的 XSLT(整页) 模板根元素 xsl:stylesheet 自动带有默认命名空间xmlns="http://www.w3.org/1999/xhtml"。
找到文件:
C:\Program Files\Adobe\Adobe Dreamweaver CS3\configuration\DocumentTypes\MMDocumentTypeDeclarations.xml
用记事本打开,在文件底部,可以看到 Dreamweaver 创建 xslt 页面时所使用的模板,只需把默认的命名空间加在此处就 OK 了。 猛击此处下载修改后的文件,下载解压后覆盖原文件即可。