很早前就遇到这个空值的属性,它既出现在 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 属性。
要去除此属性有两个方法:
1.去除 age 祖先节点默认的命名空间
2.创建 age 节点时指明其空间为祖先定下的默认空间
第一种方案一般是不会被采纳的,为了儿孙辈不太可能废掉祖先定下的规矩。第二种方案最简单:使用 createNode 方法替换 createElement 方法。
- .Set age = dom.createNode(1,"age","http://www.example.com")
效果立现。
除了程序会遇到命名空间继承的问题,xsl 模板中创建的元素也有此问题。比如创建元素:
- <XSL:ELEMENT name="a">
- <XSL:ATTRIBUTE name="href">http://www.example.com</XSL:ATTRIBUTE>
- <XSL:VALUE-OF select="'Hello world'" />
- </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 没有任何作用。
解决的方法同样有两种:
1.给 xslt 页面的根节点 xsl:stylesheet 声明默认空间,令它与 html 元素的默认空间值一致
2.使用 xsl 方法创建元素时指明其空间,只要空间与默认空间一致,就不会出现 xmlns=""
这里推荐采用第一种方案,因为只需在 xsl 文档的根节点声明了与 html 元素一致的默认命名空间,用 xsl 方法创建的元素就与祖先的空间一致,就不会出现 xmlns="",一劳永逸。
方法二就需要每次都使用
- <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 了。
【编辑推荐】