我正在编写一个XSLT 1.0样式表,将多命名空间
XML文档转换为HTML.在结果HTML的某个地方,我想列出文档中出现的所有命名空间.
这可能吗
我想到了像
<xsl:for-each select="//*|//@*"> <xsl:value-of select="namespace-uri(.)" /> </xsl:for-each>
但是我当然会收到重复的数据.所以我必须过滤,我已经打印了.
递归调用模板可以工作,但是我不能围绕如何覆盖所有元素.
访问// @ xmlns:*直接不起作用,因为无法通过XPath访问(不允许将任何前缀绑定到xmlns:namespace).
另一个没有扩展功能:
原文链接:https://www.f2er.com/xml/292301.html<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <xsl:template match="*"> <xsl:param name="pNamespaces" select="'
'"/> <xsl:variable name="vNamespaces"> <xsl:variable name="vMyNamespaces"> <xsl:value-of select="$pNamespaces"/> <xsl:for-each select="namespace::* [not(contains( $pNamespaces,concat('
',.,'
')))]"> <xsl:value-of select="concat(.,'
')"/> </xsl:for-each> </xsl:variable> <xsl:variable name="vChildsNamespaces"> <xsl:apply-templates select="*[1]"> <xsl:with-param name="pNamespaces" select="$vMyNamespaces"/> </xsl:apply-templates> </xsl:variable> <xsl:value-of select="concat(substring($vMyNamespaces,1 div not(*)),substring($vChildsNamespaces,1 div boolean(*)))"/> </xsl:variable> <xsl:variable name="vFollowNamespaces"> <xsl:apply-templates select="following-sibling::*[1]"> <xsl:with-param name="pNamespaces" select="$vNamespaces"/> </xsl:apply-templates> </xsl:variable> <xsl:value-of select="concat(substring($vNamespaces,1 div not(following-sibling::*)),substring($vFollowNamespaces,1 div boolean(following-sibling::*)))"/> </xsl:template> </xsl:stylesheet>
输出(使用Dimitre的输入样本):
http://www.w3.org/XML/1998/namespace mynamespace mynamespace2 mynamespace3
编辑:此XPath表达式:
//*/namespace::*[not(. = ../../namespace::*|preceding::*/namespace::*)]
作为证明,此样式表:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <xsl:output method="text"/> <xsl:template match="/"> <xsl:for-each select="//*/namespace::* [not(. = ../../namespace::*| preceding::*/namespace::*)]"> <xsl:value-of select="concat(.,'
')"/> </xsl:for-each> </xsl:template> </xsl:stylesheet>
输出:
http://www.w3.org/XML/1998/namespace mynamespace mynamespace2 mynamespace3
编辑4:与双程转换相同的效率.
这个样式表:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <xsl:output method="text"/> <xsl:key name="kElemByNSURI" match="*[namespace::*[not(. = ../../namespace::*)]]" use="namespace::*[not(. = ../../namespace::*)]"/> <xsl:template match="/"> <xsl:for-each select= "//namespace::*[not(. = ../../namespace::*)] [count(..|key('kElemByNSURI',.)[1])=1]"> <xsl:value-of select="concat(.,'
')"/> </xsl:for-each> </xsl:template> </xsl:stylesheet>
输出:
http://www.w3.org/XML/1998/namespace mynamespace mynamespace2 mynamespace3
编辑5:当您处理没有命名空间ax实现的XSLT处理器(像TransforMiix)时,您只能提取与此样式表实际使用的命名空间:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <xsl:output method="text"/> <xsl:key name="kElemByNSURI" match="*|@*" use="namespace-uri()"/> <xsl:template match="/"> <xsl:for-each select= "(//*|//@*)[namespace-uri()!=''] [count(.|key('kElemByNSURI',namespace-uri())[1])=1]"> <xsl:value-of select="concat(namespace-uri(),'
')"/> </xsl:for-each> </xsl:template> </xsl:stylesheet>
TransforMix输出:
mynamespace2