如何使用xslt将xml节点与特定条件结合起来?
我想合并两个XML文档 forSale.xml:如何使用xslt将xml节点与特定条件结合起来?,xml,xslt,xpath,conditional,Xml,Xslt,Xpath,Conditional,我想合并两个XML文档 forSale.xml: <forSale> <game> <cover>me3_cover.jpg</cover> <title>Mass Effect 3</title> <publisher>Electronic Arts</publisher> <developer>BioWare</developer>
<forSale>
<game>
<cover>me3_cover.jpg</cover>
<title>Mass Effect 3</title>
<publisher>Electronic Arts</publisher>
<developer>BioWare</developer>
<genre>Role-Playing</genre>
<releaseDate>
<yyyy>2012</yyyy>
<mm>03</mm>
<dd>06</dd>
</releaseDate>
<esrbRating>M</esrbRating>
<platforms>
<platform>X360</platform>
<platform>PC</platform>
<platform>PS3</platform>
<platform>WIIU</platform>
</platforms>
</game>
<!--more games-->
</forSale>
其次,我不知道如何像那样复制
任何帮助都将不胜感激!谢谢大家! 问题第一部分的答案是: 您可以使用与访问第一个文档相同的想法来访问第二个文档: 这样做应该可以:
<xsl:for-each select="document('reviews.xml')/reviews/game[title = $title]">
<xsl:copy>
<xsl:apply-templates select="review" />
</xsl:copy>
</xsl:for-each>
更新:生成CDATA
我不知道是否以及如何在输入中测试CDATA。但您可以测试是否存在将被转义的字符,但您不喜欢这样做并使用CDATA。
试着这样做:
<forSale>
<game>
<cover>me3_cover.jpg</cover>
<title>Mass Effect 3</title>
<publisher>Electronic Arts</publisher>
<developer>BioWare</developer>
<genre>Role-Playing</genre>
<releaseDate>
<yyyy>2012</yyyy>
<mm>03</mm>
<dd>06</dd>
</releaseDate>
<esrbRating>M</esrbRating>
<platforms>
<platform>X360</platform>
<platform>PC</platform>
<platform>PS3</platform>
<platform>WIIU</platform>
</platforms>
<review>
<critic>Kevin VanOrd</critic>
<synopsis>
<![CDATA[some data]]>
</synopsis>
<pros>
<pro><![CDATA[some data]]></pro>
<pro><![CDATA[some data]]></pro>
<pro><![CDATA[some data]]></pro>
<pro><![CDATA[some data]]></pro>
<pro><![CDATA[some data]]></pro>
</pros>
<cons>
<con><![CDATA[some data]]></con>
<con><![CDATA[some data]]></con>
</cons>
<content>
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
<p />
<img url="me3_img1.jpg"><![CDATA[some data]]></img>
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
<p />
<img url="me3_img2.jpg"><![CDATA[some data]]></img>
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
<p />
<img url="me3_img3.jpg"><![CDATA[some data]]></img>
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
<p />
<img url="me3_img4.jpg"><![CDATA[some data]]></img>
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
<p />
<img url="me3_img5.jpg"><![CDATA[some data]]></img>
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
<p />
<img url="me3_img6.jpg"><![CDATA[some data]]></img>
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
<p />
<img url="me3_img7.jpg"><![CDATA[some data]]></img>
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
<p />
<img url="me3_img8.jpg"><![CDATA[some data]]></img>
<![CDATA[some data]]>
<p />
<![CDATA[some data]]>
</content>
</review>
</game>
<!--more games-->
</forSale>
<xsl:template match="text()">
<xsl:choose>
<xsl:when test="contains(.,'<')">
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of disable-output-escaping="yes" select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="."/>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
![CDATA[
]]
基本上,XSLT数据模型不区分纯文本节点和CDATA节,因此在XSLT处理器使用的输入树中不会对您的输入(例如,某些数据
或
进行建模。因此,您无法确保这样的元素按原样复制。但是,您可以指示XSLT处理器将某些结果元素的内容输出为CDATA节,例如
<xsl:output cdata-section-elements="foo bar"/>
将确保XSLT转换的序列化结果树使用
和
。但是,这些名称的所有元素都会出现这种情况,而不仅仅是从包含CDATA节内容的输入中复制的元素。问题的完整解决方案如下所示:
<?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" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="/">
<forSale>
<xsl:for-each select="document('Untitled1.xml')/forSale/game">
<game>
<xsl:copy-of select="cover"/>
<xsl:copy-of select="title"/>
<xsl:variable name="title" select="title"/>
<xsl:copy-of select="publisher"/>
<xsl:copy-of select="developer"/>
<xsl:copy-of select="genre"/>
<xsl:copy-of select="releaseDate"/>
<xsl:copy-of select="esrbRating"/>
<xsl:copy-of select="platforms"/>
<xsl:for-each select="document('Untitled2.xml')/reviews/game">
<xsl:choose>
<xsl:when test="./title = $title">
<review>
<xsl:call-template name="checkChild">
<xsl:with-param name="self" select="self::node()/review"/>
</xsl:call-template>
</review>
</xsl:when>
</xsl:choose>
</xsl:for-each>
</game>
</xsl:for-each>
</forSale>
</xsl:template>
<xsl:template name="checkChild">
<xsl:param name="self"/>
<xsl:for-each select="$self/child::node()">
<xsl:message>
<xsl:value-of select="name()"/>
</xsl:message>
<xsl:choose>
<xsl:when test="self::node()/name() = 'pros'">
<pros>
<xsl:for-each select="pro">
<pro>
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</pro>
</xsl:for-each>
</pros>
</xsl:when>
<xsl:when test="self::node()/name() = 'synopsis'">
<xsl:copy>
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</xsl:copy>
</xsl:when>
<xsl:when test="self::node()/name() = 'cons'">
<cons>
<xsl:for-each select="con">
<con>
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</con>
</xsl:for-each>
</cons>
</xsl:when>
<xsl:when test="self::node()/name() = 'content'">
<content>
<xsl:for-each select="child::node()">
<xsl:choose>
<xsl:when test="self::text()">
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</xsl:when>
<xsl:when test="self::node()/name()= 'img'">
<xsl:element name="{name()}">
<xsl:if test="@url">
<xsl:attribute name="url">
<xsl:value-of select="@url"/>
</xsl:attribute>
</xsl:if>
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</xsl:element>
</xsl:when>
<xsl:otherwise>
<xsl:copy>
<xsl:copy-of select="node()"/>
</xsl:copy>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</content>
</xsl:when>
<xsl:otherwise>
<xsl:copy>
<xsl:copy-of select="node()"/>
</xsl:copy>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
![CDATA[
]]
![CDATA[
]]
![CDATA[
]]
![CDATA[
]]
![CDATA[
]]
Martin,没有什么比得上“XSLT数据模型”。您可能想说“XPath数据模型”或“XML信息集”?XSLT()使用了“XPath数据模型”,因此我认为使用“XSLT数据模型”一词没有任何错误。
<xsl:output cdata-section-elements="foo bar"/>
<?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" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="/">
<forSale>
<xsl:for-each select="document('Untitled1.xml')/forSale/game">
<game>
<xsl:copy-of select="cover"/>
<xsl:copy-of select="title"/>
<xsl:variable name="title" select="title"/>
<xsl:copy-of select="publisher"/>
<xsl:copy-of select="developer"/>
<xsl:copy-of select="genre"/>
<xsl:copy-of select="releaseDate"/>
<xsl:copy-of select="esrbRating"/>
<xsl:copy-of select="platforms"/>
<xsl:for-each select="document('Untitled2.xml')/reviews/game">
<xsl:choose>
<xsl:when test="./title = $title">
<review>
<xsl:call-template name="checkChild">
<xsl:with-param name="self" select="self::node()/review"/>
</xsl:call-template>
</review>
</xsl:when>
</xsl:choose>
</xsl:for-each>
</game>
</xsl:for-each>
</forSale>
</xsl:template>
<xsl:template name="checkChild">
<xsl:param name="self"/>
<xsl:for-each select="$self/child::node()">
<xsl:message>
<xsl:value-of select="name()"/>
</xsl:message>
<xsl:choose>
<xsl:when test="self::node()/name() = 'pros'">
<pros>
<xsl:for-each select="pro">
<pro>
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</pro>
</xsl:for-each>
</pros>
</xsl:when>
<xsl:when test="self::node()/name() = 'synopsis'">
<xsl:copy>
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</xsl:copy>
</xsl:when>
<xsl:when test="self::node()/name() = 'cons'">
<cons>
<xsl:for-each select="con">
<con>
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</con>
</xsl:for-each>
</cons>
</xsl:when>
<xsl:when test="self::node()/name() = 'content'">
<content>
<xsl:for-each select="child::node()">
<xsl:choose>
<xsl:when test="self::text()">
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</xsl:when>
<xsl:when test="self::node()/name()= 'img'">
<xsl:element name="{name()}">
<xsl:if test="@url">
<xsl:attribute name="url">
<xsl:value-of select="@url"/>
</xsl:attribute>
</xsl:if>
<xsl:text disable-output-escaping="yes"><![CDATA[</xsl:text>
<xsl:value-of select="."/>
<xsl:text disable-output-escaping="yes">]]></xsl:text>
</xsl:element>
</xsl:when>
<xsl:otherwise>
<xsl:copy>
<xsl:copy-of select="node()"/>
</xsl:copy>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</content>
</xsl:when>
<xsl:otherwise>
<xsl:copy>
<xsl:copy-of select="node()"/>
</xsl:copy>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>