在我的XSLT中,我正在预处理大型xml文件,并且必须操作某些值(因为源系统没有按预期交付它们)。
属性"name"one_answers"nm"都应该包含相同的文本。但是,在原始XML中,它们是空的。
我需要使用另一个属性"description"和一个硬编码的查找列表来生成它们(例如description="Some value"意味着nm和name都应该是"NameABC")。因为我的查找列表很长,我真的不想在两个模板中实现它,一个用于属性"nm",另一个用于属性"name"。相反,我想在一个地方实现我的查找列表,并且总是同时更改两个属性。
有什么办法可以做到吗?
这是我的原始XML(当然是简化的示例):
<?xml version="1.0" encoding="UTF-8"?>
<Sample>
<Header>
<Type>A</Type>
</Header>
<DataSet name="">
<Info description="Some value" nm="" other="123"/>
</DataSet>
<DataSet name="">
<Info description="Another value" nm="" other="456"/>
</DataSet>
</Sample>
所需输出:<?xml version="1.0" encoding="UTF-8"?>
<Sample>
<Header>
<Type>A</Type>
</Header>
<DataSet name="NameABC">
<Info description="Some value" other="123" nm="NameABC"/>
</DataSet>
<DataSet name="NameXYZ">
<Info description="Another value" other="456" nm="NameXYZ"/>
</DataSet>
</Sample>
我当前的XSLT(只改变属性"nm"):
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:fo="http://www.w3.org/1999/XSL/Format">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:template match="Sample">
<xsl:copy>
<xsl:apply-templates/>
</xsl:copy>
</xsl:template>
<xsl:template match="DataSet">
<xsl:copy>
<xsl:for-each select="@*">
<xsl:attribute namespace="" name="{name()}"><xsl:value-of select="."/></xsl:attribute>
</xsl:for-each>
<xsl:apply-templates/>
</xsl:copy>
</xsl:template>
<xsl:template match="Info">
<xsl:element name="Info">
<xsl:for-each select="@*">
<xsl:attribute namespace="" name="{name()}"><xsl:value-of select="."/></xsl:attribute>
</xsl:for-each>
<xsl:apply-templates select="@nm"/>
<xsl:copy-of select="node()"/>
</xsl:element>
</xsl:template>
<xsl:template match="Info/@nm">
<xsl:choose>
<xsl:when test="/Sample/Header/Type='A' and .=''">
<xsl:attribute name="nm">
<xsl:choose>
<xsl:when test="../@description = 'Some value'">NameABC</xsl:when>
<xsl:when test="../@description = 'Another value'">NameXYZ</xsl:when>
</xsl:choose>
</xsl:attribute>
</xsl:when>
<xsl:otherwise>
<xsl:copy-of select="."/>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
<xsl:template match="*">
<xsl:copy-of select="."/>
</xsl:template>
</xsl:stylesheet>
如何:
XSLT 1.0<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<!-- identity transform -->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="DataSet">
<xsl:variable name="nm">
<xsl:choose>
<xsl:when test="Info/@description = 'Some value'">NameABC</xsl:when>
<xsl:when test="Info/@description = 'Another value'">NameXYZ</xsl:when>
</xsl:choose>
</xsl:variable>
<xsl:copy>
<xsl:apply-templates select="@*"/>
<xsl:attribute name="name"><xsl:value-of select="$nm"/></xsl:attribute>
<xsl:apply-templates select="node()" >
<xsl:with-param name="nm" select="$nm"/>
</xsl:apply-templates>
</xsl:copy>
</xsl:template>
<xsl:template match="Info">
<xsl:param name="nm"/>
<xsl:copy>
<xsl:apply-templates select="@*"/>
<xsl:attribute name="nm"><xsl:value-of select="$nm"/></xsl:attribute>
<xsl:apply-templates select="node()"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
选择:
XSLT 1.0<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<!-- identity transform -->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="DataSet/@name">
<xsl:attribute name="name">
<xsl:call-template name="getName">
<xsl:with-param name="desc" select="../Info/@description"/>
</xsl:call-template>
</xsl:attribute>
</xsl:template>
<xsl:template match="Info/@nm">
<xsl:attribute name="nm">
<xsl:call-template name="getName">
<xsl:with-param name="desc" select="../@description"/>
</xsl:call-template>
</xsl:attribute>
</xsl:template>
<xsl:template name="getName">
<xsl:param name="desc"/>
<xsl:choose>
<xsl:when test="$desc = 'Some value'">NameABC</xsl:when>
<xsl:when test="$desc = 'Another value'">NameXYZ</xsl:when>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
在这种情况下,我更喜欢使用外部XML文件来实现查找表。
<lut>
<entry desc="Some value" name="NameABC" number="123"/>
<entry desc="Another value" name="NameXYZ" number="456"/>
<!-- and another dozen entries -->
</lut>
放在lut.xml
中,可以为DataSet和Info应用以下样式表模板(其中other
也从lut中读取,以给出另一个例子):
<xsl:template match="DataSet">
<xsl:variable name="desc" select="./Info/@description"/>
<xsl:copy>
<xsl:apply-templates select="@*"/>
<xsl:attribute name="name">
<xsl_value-of select="document('lut.xml')/lut/entry[@desc=$desc]/@name"/>
</xsl:attribute>
<xsl:apply-templates/>
</xsl:copy>
</xsl:template>
<xsl:template match="Info">
<xsl:variable name="desc" select="./@description"/>
<xsl:copy>
<xsl:apply-templates select="@*"/>
<xsl:attribute name="nm">
<xsl:value-of select="document('lut.xml')/lut/entry[@desc=$desc]/@name"/>
</xsl:attribute>
<xsl:attribute name="other">
<xsl:value-of select="document('lut.xml')/lut/entry[@desc=$desc]/@number"/>
</xsl:attribute>
<xsl:apply-templates/>
</xsl:copy>
</xsl:template>
实现对Header
值的依赖也很容易:要么使用另一个属性,要么将entries
分组在"类型"块中。