我正在尝试比较以下两个XML节并生成第三个节。
1)
<Drive>
<Folders>
<Folder>
<name>folder1</name>
<Files>
<File>
<State>Writable</State>
<name>file1</name>
</File>
</Files>
</Folder>
</Folders>
<Files>
<File>
<State>Readable</State>
<name>file2</name>
</File>
</Files>
</Drive>2)
<Partition>
<Directories>
<Directory>
<name>folder1</name>
<Unix-File>
<name>file1</name>
</Unix-File>
</Directory>
<Directory>
<name>folder-5</name>
<Unix-File>
<name>file-5</name>
</Unix-File>
</Directory>
</Directories>
<Unix-Files>
<Unix-File>
<name>file2</name>
</Unix-File>
</UnixFiles>
</Partition> 现在,
1)生成的XML节应该包含所有的Unix-file和Directory元素,这些元素与第一个XML节中的元素同名,并且State为Writable。
2)对于没有匹配文件的匹配文件夹,应将其省略。
3)这里我假设文件结构的深度是2。也就是说,文件夹将不再包含文件夹。
因此,对于上面的场景,结果应该是
<Partition>
<Directories>
<Directory>
<name>folder1</name>
<Unix-File>
<name>file1</name>
</Unix-File>
</Directory>
</Directories>
</Partition> 我正在使用下面的XSL文件。
<xsl:if test="count(Drive/Folders/Folder) > 0">
<Directories>
<xsl:for-each select="Partition/Directories/Directory">
<xsl:variable name="directoryName" select="name"/>
<xsl:if test="../../../Drive/Folders/Folder[name=$directoryName]">
<Directory>
<name><xsl:value-of select="name"/>
</name>
<Unix-Files>
<xsl:for-each select="Unix-Files/Unix-File">
<xsl:variable name="fileName" select="name"/>
<xsl:if test="../../../../../Drive/Folders/Folder[name=$directoryName]/Files/File[name=$fileName]">
<xsl:if test="../../../../../Drive/Folders/Folder[name=$directoryName]/Files/File[name=$fileName]/State != 'Writable'">
<xsl:value-of select="."/>
</xsl:if>
</xsl:if>
</xsl:for-each>
</Unix-Files>
</Directory>
</xsl:if>
</xsl:for-each>
</Directories>
</xsl:if>除了上面的要求2之外,上面的XSLT工作得很好。即,当文件夹中的文件不匹配时,它也应该省略该文件夹。
发布于 2013-05-25 23:33:58
您可以尝试这个基于密钥的解决方案:
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" >
<xsl:output method="xml" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:key name="kFile" match="Files/File[State='Writable']" use="concat(../../name, '/' , name)"/>
<xsl:template match="@*|node()">
<xsl:copy >
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="Directory">
<xsl:if test="Unix-File[ key( 'kFile', concat( ../name, '/', name ) )]">
<xsl:apply-templates select="name"/>
<xsl:copy>
<xsl:for-each select="Unix-File[ key( 'kFile', concat( ../name, '/', name ) )]" >
<xsl:apply-templates select="."/>
</xsl:for-each>
</xsl:copy>
</xsl:if>
</xsl:template>
<xsl:template match="/*" >
<Partition>
<xsl:if test="count(Drive/Folders/Folder) > 0">
<xsl:element name="Directories" >
<xsl:apply-templates select="Partition/Directories/Directory" />
</xsl:element>
</xsl:if>
</Partition>
</xsl:template>
</xsl:stylesheet>使用此输入:
<?xml version="1.0" encoding="utf-8" ?>
<xml>
<Drive>
<Folders>
<Folder>
<name>folder1</name>
<Files>
<File>
<State>Writable</State>
<name>file1</name>
</File>
</Files>
</Folder>
</Folders>
<Files>
<File>
<State>Readable</State>
<name>file2</name>
</File>
</Files>
</Drive>
<Partition>
<Directories>
<Directory>
<name>folder1</name>
<Unix-File>
<name>file1</name>
</Unix-File>
</Directory>
<Directory>
<name>folder-5</name>
<Unix-File>
<name>file-5</name>
</Unix-File>
</Directory>
</Directories>
<Unix-Files>
<Unix-File>
<name>file2</name>
</Unix-File>
</Unix-Files>
</Partition>
</xml>它将生成以下输出:
<?xml version="1.0"?>
<Partition>
<Directories>
<name>folder1</name>
<Directory>
<Unix-File>
<name>file1</name>
</Unix-File>
</Directory>
</Directories>
</Partition>键"kFile“为每个文件保存一个条目,状态为”文件夹名/文件名“。
目录模板检查此目录中是否有任何Unix文件的kFile密钥。
https://stackoverflow.com/questions/16750090
复制相似问题