我有一个像
这样的XML文件<node>
<properties>
<property name="titleENG">English title</property>
<property name="titleFRE">French title</property>
<property name="descENG">English description</property>
<property name="descFRE">French description</property>
...
</properties>
</node>
作为输出,我想要一个像
这样的CSV文件title, English title, French title
desc, English description, French description
...
我们假设我知道可用的语言,但我不知道可用的属性(title,desc,...)
我可以使用以下XSLT获取英文字符串
<?xml version="1.0" encoding="ISO-8859-1"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" >
<xsl:template match="@* | node()">
<xsl:apply-templates select="@* | node()"/>
</xsl:template>
<xsl:template match="property">
<xsl:if test="@name[ends-with(.,'ENG')]">
<xsl:variable name="key" select="substring(@name,1,string-length(@name)-3)" />
<xsl:value-of select="$key"/>, <xsl:value-of select="./text()"/><xsl:text> </xsl:text>
</xsl:if>
</xsl:template>
</xsl:stylesheet>
但在'if'中,我还需要name=concat($key,'FRE')
属性的值,我该怎么做?
任何帮助表示赞赏。
修改 我想我可以将我的问题改为(因为我可以很容易地将我当前的集合转换为这个集合):假设我有一个类似XML的
<node>
<properties>
<property name="dummy"></property>
<property name="title" lang="ENG">English title</property>
<property name="title" lang="FRE">French title</property>
<property name="desc" lang="ENG">English description</property>
<property name="desc" lang="FRE">French description</property>
...
</properties>
</node>
如何创建一个(CSV)输出,在这种情况下有两行,第一列中的名称和下一列中的英语,法语翻译?
答案 0 :(得分:1)
XSLT 2.0没问题
实际上,我没有时间切换到XSLT 2.0模式,这将同时适用于:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:my="http://www.example.com/my">
<xsl:output method="text" encoding="UTF-8"/>
<xsl:variable name="root" select="/" />
<xsl:key name="prop-by-prefix" match="property" use="substring(@name, 1, string-length(@name)-3)" />
<xsl:key name="prop-by-name" match="property" use="@name" />
<my:languages>
<lang>ENG</lang>
<lang>FRE</lang>
</my:languages>
<xsl:template match="/">
<xsl:for-each select="node/properties/property[count(. | key('prop-by-prefix', substring(@name, 1, string-length(@name)-3))[1]) = 1]">
<xsl:variable name="prefix" select="substring(@name, 1, string-length(@name)-3)" />
<xsl:value-of select="$prefix"/>
<xsl:text>,</xsl:text>
<xsl:for-each select="document('')/xsl:stylesheet/my:languages/lang">
<xsl:variable name="name" select="concat($prefix, .)"/>
<!-- switch context back to source document in order to use key -->
<xsl:for-each select="$root">
<xsl:value-of select="key('prop-by-name', $name)"/>
</xsl:for-each>
<xsl:if test="position()!=last()">
<xsl:text>,</xsl:text>
</xsl:if>
</xsl:for-each>
<xsl:if test="position()!=last()">
<xsl:text> </xsl:text>
</xsl:if>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
应用于您的输入:
<node>
<properties>
<property name="titleENG">English title</property>
<property name="titleFRE">French title</property>
<property name="descENG">English description</property>
<property name="descFRE">French description</property>
</properties>
</node>
获得以下结果:
title,English title,French title
desc,English description,French description
我刚才注意到你已经改变了你的输入。这使它更简单:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text" encoding="UTF-8"/>
<xsl:key name="prop-by-name" match="property" use="@name" />
<xsl:template match="/">
<xsl:for-each select="node/properties/property[count(. | key('prop-by-name', @name)[1]) = 1][@lang]">
<xsl:value-of select="@name"/>
<xsl:text>,</xsl:text>
<xsl:for-each select="key('prop-by-name', @name)">
<xsl:value-of select="."/>
<xsl:if test="position()!=last()">
<xsl:text>,</xsl:text>
</xsl:if>
</xsl:for-each>
<xsl:if test="position()!=last()">
<xsl:text> </xsl:text>
</xsl:if>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
如果您使用的是XSLT 2.0(或支持EXSLT set:distinct的XSLT 1.0处理器)函数,则可以使用它而不是Muenchian分组。
答案 1 :(得分:0)
您可以在if:
之外声明$key
变量
<xsl:variable name="key" select="substring(@name,1,string-length(@name)-3)" />
使用substring-after
$key
属性中的@name
获取后缀:
<xsl:variable name="suffix" select="substring-after(@name, $key)"/>
然后,您可以独立于知道后缀进行测试:
<xsl:if test="@name[ends-with(.,$suffix)]">