我有两个XML文件,我目前正在合并一个新的结果XML文件。
第一档:
<T>
<T1>AA</T1>
<T2>1</T2>
</T>
<T>
<T1>BB</T1>
<T2>1</T2>
</T>
<T>
<T1>AC</T1>
<T2>1</T2>
</T>
第二档:
<T>
<T1>BB</T1>
<T2>3</T2>
</T>
<T>
<T1>AB</T1>
<T2>3</T2>
</T>
我希望结果是:
<T>
<T1>AA</T1>
<T2>1</T2>
</T>
<T>
<T1>BB</T1>
<T2>3</T2>
</T>
<T>
<T1>AC</T1>
<T2>1</T2>
</T>
<T>
<T1>AB</T1>
<T2>3</T2>
</T>
IE中。我想将父节点<T>
添加到结果文件中,如果它们在我的第一个文件中不存在(取决于<T1>
),但如果它们确实存在则替换。
我希望解决方案是通用的,解决方案应该适用于&#34; new&#34;第二个档案。
任何人都可以提供帮助,这就是我现在所拥有的:
<xsl:output indent="yes"/>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()" />
</xsl:copy>
</xsl:template>
<xsl:template match="T">
<xsl:copy>
<xsl:apply-templates select="*"/>
<xsl:apply-templates select="document('./AB.xml')/T/*" />
</xsl:copy>
</xsl:template>
我有两个XML文件,我目前正在合并一个新的结果XML文件。 第一个档案:
<TT>
<T>
<T1>AA</T1>
<T2>1</T2>
</T>
<T>
<T1>BB</T1>
<T2>1</T2>
<T3>1</T3>
</T>
</TT>
第二档:
<TT>
<T>
<T1>AA</T1>
<T2>3</T2>
<T3>3</T3>
</T>
<T>
<T1>BB</T1>
<T3>3</T3>
</T>
</TT>
我希望结果是:
<TT>
<T>
<T1>AA</T1>
<T2>3</T2>
<T3>3</T3>
</T>
<T>
<T1>BB</T1>
<T2>1</T2>
<T3>3</T3>
</T>
</TT>
换句话说,<T1>
将是决定因素。到目前为止收到的答案很棒我刚才意识到我的问题实际上比我最初想的要复杂得多。
答案 0 :(得分:0)
给出两个有效 XML文件:
<强> file1.xml 强>
<input>
<T>
<T1>AA</T1>
<T2>1</T2>
</T>
<T>
<T1>BB</T1>
<T2>1</T2>
</T>
<T>
<T1>AC</T1>
<T2>1</T2>
</T>
</input>
<强> file2.xml 强>
<input>
<T>
<T1>BB</T1>
<T2>2</T2>
</T>
<T>
<T1>AB</T1>
<T2>3</T2>
</T>
</input>
将以下样式表应用于file1.xml:
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:variable name="file2" select="document('file2.xml')" />
<xsl:variable name="localT1s" select="/input/T/T1" />
<xsl:template match="/input">
<result>
<xsl:apply-templates select="T"/>
<xsl:apply-templates select="$file2/input/T[not(T1=$localT1s)]"/>
</result>
</xsl:template>
<xsl:template match="T">
<xsl:copy>
<xsl:copy-of select="T1"/>
<T2>
<xsl:choose>
<xsl:when test="$file2/input/T[T1=current()/T1]">
<xsl:value-of select="$file2/input/T[T1=current()/T1]/T2"/>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="T2"/>
</xsl:otherwise>
</xsl:choose>
</T2>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
结果将是:
<?xml version="1.0" encoding="UTF-8"?>
<result>
<T>
<T1>AA</T1>
<T2>1</T2>
</T>
<T>
<T1>BB</T1>
<T2>2</T2>
</T>
<T>
<T1>AC</T1>
<T2>1</T2>
</T>
<T>
<T1>AB</T1>
<T2>3</T2>
</T>
</result>
回答您修正后的问题:
你不清楚你对输入文件中<T#>
元素的名称有什么了解。如果(我假设)什么都没有,那么这变得相当复杂:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:exsl="http://exslt.org/common"
extension-element-prefixes="exsl">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:variable name="file2" select="document('file2.xml')" />
<xsl:template match="/TT">
<xsl:variable name="localT1s" select="T/T1" />
<xsl:copy>
<xsl:apply-templates select="T"/>
<xsl:apply-templates select="$file2/TT/T[not(T1=$localT1s)]" mode="import"/>
</xsl:copy>
</xsl:template>
<xsl:template match="T">
<xsl:variable name="elemNames">
<xsl:for-each select="*">
<name><xsl:value-of select="name()"/></name>
</xsl:for-each>
</xsl:variable>
<xsl:copy>
<xsl:apply-templates select="*"/>
<xsl:apply-templates select="$file2/TT/T[T1=current()/T1]/*[not(name()=exsl:node-set($elemNames)/name)]" mode="import"/>
</xsl:copy>
</xsl:template>
<xsl:template match="T/*">
<xsl:variable name="T1" select="../T1" />
<xsl:variable name="Tx" select="name()" />
<xsl:copy>
<xsl:choose>
<xsl:when test="$file2/TT/T[T1=$T1]/*[name()=$Tx]">
<xsl:value-of select="$file2/TT/T[T1=$T1]/*[name()=$Tx]"/>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="."/>
</xsl:otherwise>
</xsl:choose>
</xsl:copy>
</xsl:template>
<xsl:template match="*" mode="import">
<xsl:copy-of select="."/>
</xsl:template>
</xsl:stylesheet>
我用以下两个文件测试了上述内容:
<强> file1.xml 强>
<TT>
<T>
<T1>A</T1>
<T2>a2</T2>
</T>
<T>
<T1>B</T1>
<T2>b2</T2>
<T3>b3</T3>
</T>
</TT>
<强> file2.xml 强>
<TT>
<T>
<T1>A</T1>
<T2>A2</T2>
<T3>A3</T3>
</T>
<T>
<T1>B</T1>
<T3>B3</T3>
</T>
<T>
<T1>C</T1>
<T2>C2</T2>
</T>
</TT>
并收到了这个结果:
<?xml version="1.0" encoding="UTF-8"?>
<TT>
<T>
<T1>A</T1>
<T2>A2</T2>
<T3>A3</T3>
</T>
<T>
<T1>B</T1>
<T2>b2</T2>
<T3>B3</T3>
</T>
<T>
<T1>C</T1>
<T2>C2</T2>
</T>
</TT>
答案 1 :(得分:0)
使用XSLT 2.0(然后使用像Saxon 9这样的XSLT 2.0处理器),您可以很好地在两个文档之间进行交叉引用:
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema" exclude-result-prefixes="xs" version="2.0">
<xsl:param name="doc2-url" select="'file2.xml'"/>
<xsl:variable name="doc2" select="doc($doc2-url)"/>
<xsl:variable name="main-doc" select="/"/>
<xsl:output indent="yes"/>
<xsl:key name="by-T1" match="T" use="T1"/>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()" />
</xsl:copy>
</xsl:template>
<xsl:template match="*[T]">
<xsl:copy>
<xsl:apply-templates select="T, $doc2//T[not(key('by-T1', T1, $main-doc))]"/>
</xsl:copy>
</xsl:template>
<xsl:template match="T[$main-doc is /][key('by-T1', T1, $doc2)]/T2">
<xsl:copy-of select="key('by-T1', ../T1, $doc2)/T2"/>
</xsl:template>
</xsl:stylesheet>