我在XSLT中更加敏捷。我已经看了一些我感兴趣的任务的代码,并构建了一些逻辑,但我无法进入所需的输出。如果能得到帮助,我很高兴。
输入XML:
<?xml version="1.0" encoding="UTF-8"?>
<t>
<Data>
<CD>
<Artist>xxx.yyy</Artist>
<song>abc</song>
</CD>
<CD>
<Artist>xxx.zzz</Artist>
<song>cba</song>
</CD>
<CD>
<Artist>aaa.kkk</Artist>
<song>123</song>
</CD>
<CD>
<Artist>aaa.lll</Artist>
<song>456</song>
</CD>
<CD>
<Artist>ddd</Artist>
<song>1234</song>
</CD>
</Data>
<Music>
<music_no>E123</music_no>
<music_type>outdoor</music_type>
<Artist>bat.ball</Artist>
<value>0000</value>
</Music>
<Music>
<music_no>E123</music_no>
<music_type>outdoor</music_type>
<Artist>bat.stone</Artist>
<value>0001</value>
</Music>
<Music>
<music_no>E111</music_no>
<music_type>outdoor</music_type>
<Artist>board.coins</Artist>
<value>0002</value>
</Music>
<Music>
<music_no>E111</music_no>
<music_type>outdoor</music_type>
<Artist>board.ball</Artist>
<value>0003</value>
</Music>
<Music>
<music_no>E001</music_no>
<music_type>indoor</music_type>
<Artist>bat.ball</Artist>
<value>8888</value>
</Music>
<Music>
<music_no>E001</music_no>
<music_type>indoor</music_type>
<Artist>bat.stone</Artist>
<value>9999</value>
</Music>
<Music>
<music_no>E111</music_no>
<music_type>indoor</music_type>
<Artist>board.coins</Artist>
<value>0001</value>
</Music>
<Music>
<music_no>E111</music_no>
<music_type>indoor</music_type>
<Artist>bat</Artist>
<value>0001</value>
</Music>
</t>
预期产出:
<?xml version="1.0" encoding="UTF-8"?>
<version_3>
<information>
<xxx>
<yyy>abc</yyy>
<zzz>cba</zzz>
</xxx>
<aaa>
<kkk>123</kkk>
<lll>456</lll>
</aaa>
<ddd>1234</ddd>
</information>
<information>
<bat>
<ball>0000</ball>
<stone>0001</stone>
</bat>
<board>
<coins>0002</coins>
<ball>0003</ball>
</board>
<bat>
<ball>8888</ball>
<stone>9999</stone>
</bat>
<board>
<coins>0001</coins>
</board>
</information>
<information>
<bat>0001</bat>
</information>
</version_3>
编辑预期成果:
Expected Output:
<?xml version="1.0" encoding="UTF-8"?>
<version_3>
<information>
<xxx>
<yyy>abc</yyy>
<zzz>cba</zzz>
</xxx>
<aaa>
<kkk>123</kkk>
<lll>456</lll>
</aaa>
<ddd>1234</ddd>
</information>
<information>
<bat>
<ball>0000</ball>
<stone>0001</stone>
</bat>
</information>
<information>
<board>
<coins>0002</coins>
<ball>0003</ball>
</board>
<board>
<coins>0001</coins>
<bat>0001</bat>
</board>
</information>
<information>
<bat>
<ball>8888</ball>
<stone>9999</stone>
</bat>
</information>
</version_3>
在上面的输入XML中:您可以注意到“CD”元素和值,类似“音乐”。我想获得一个看起来像
的输出“CD”的示例输出:
<information>
<xxx>
<yyy>abc</yyy>
<zzz>cba</zzz>
</xxx>
<aaa>
<kkk>123</kkk>
<lll>456</lll>
</aaa>
<ddd>1234</ddd>
</information>
我可以在一定程度上通过Muenchian Grouping实现这一目标。但是,
下一个元素“音乐”有子元素,其中前两个元素“music_no”&amp; “music_type”匹配,然后必须对“艺术家”中的值进行分组。如果它们不匹配,则必须单独分组。
音乐的样本o / p:
<information>
<bat>
<ball>0000</ball>
<stone>0001</stone>
</bat>
<board>
<coins>0002</coins>
<ball>0003</ball>
</board>
<bat>
<ball>8888</ball>
<stone>9999</stone>
</bat>
<board>
<coins>0001</coins>
</board>
</information>
<information>
<bat>0001</bat>
</information>
我无法实现第二部分,因为迭代有点棘手。帮助得到了赞赏。
注意:对于“Music”元素,如果Artist中的Value与相应的节点相似但没有“。”然后,该值必须单独分组,在“信息”之外,并且应该有新的“信息”
我工作的代码:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:ext="http://exslt.org/common" exclude-result-prefixes="ext">
<xsl:key name="kBychildName" match="CD" use="name(Artist/*[1])"/>
<xsl:key name="kByAttribs" match="Artist" use="concat(../@music_no, '+', ../@music_type)"/>
<xsl:key name="kChildByAttribsAndArtist" match="Artist/*" use="concat(../../@music_no, '+', ../../@music_type, '+', name())"/>
<xsl:template match="/">
<version_3>
<information>
<xsl:variable name="var1">
<xsl:apply-templates/>
</xsl:variable>
<xsl:apply-templates mode="pass2" select="ext:node-set($var1)/* [generate-id()=generate-id(key('kBychildName',name(Artist/*[1]))[1]) or not(Artist/*)]"/>
</information>
<information>
<xsl:variable name="var2">
<xsl:apply-templates/>
</xsl:variable>
<xsl:apply-templates mode="pass3" select="ext:node-set($var2)/*/* [generate-id() = generate-id(key('kByAttribs', concat(../@music_no, '+', ../@music_type) ) [1])] "/>
</information>
</version_3>
<!--xsl:copy-of select="//msg_debug"/-->
</xsl:template>
<xsl:template match="CD[contains(Artist,'.')]">
<CD>
<Artist>
<xsl:element name="{substring-before(Artist, '.')}">
<xsl:element name="{substring-after(Artist, '.')}">
<xsl:value-of select="song"/>
</xsl:element>
</xsl:element>
</Artist>
</CD>
</xsl:template>
<xsl:template match="CD">
<CD>
<Artist>
<xsl:element name="{Artist}">
<xsl:value-of select="song"/>
</xsl:element>
</Artist>
</CD>
</xsl:template>
<xsl:template match="CD" mode="pass2">
<xsl:apply-templates select="*/*[1]" mode="pass2"/>
</xsl:template>
<xsl:template match="Artist/*" mode="pass2">
<xsl:copy>
<xsl:copy-of select="self::*[not(*)]/text()|key('kBychildName', name())/*/*/*"/>
</xsl:copy>
</xsl:template>
<xsl:template match="Music[contains(Artist, '.')]">
<Music music_no="{music_no}" music_type="{music_type}">
<Artist>
<xsl:element name="{substring-before(Artist, '.')}">
<xsl:element name="{substring-after(Artist, '.')}">
<xsl:value-of select="value"/>
</xsl:element>
</xsl:element>
</Artist>
</Music>
</xsl:template>
<xsl:template match="Music">
<Music music_no="{music_no}" music_type="{music_type}">
<Artist>
<xsl:element name="{Artist}">
<xsl:value-of select="value"/>
</xsl:element>
</Artist>
</Music>
</xsl:template>
<xsl:template match="Artist" mode="pass3">
<!--Artist-->
<xsl:apply-templates mode="pass3" select="*[generate-id() =generate-id(key('kChildByAttribsAndArtist', concat(../../@music_no, '+', ../../@music_type,'+', name()))[1] ) ]"/>
<xsl:copy-of select="key('kByAttribs',concat(../@music_no, '+', ../@music_type) )/*[not(*)] "/>
<!--/Artist-->
</xsl:template>
<xsl:template match="Artist/*" mode="pass3">
<xsl:element name="{name()}">
<xsl:copy-of select="key('kChildByAttribsAndArtist', concat(../../@music_no, '+', ../../@music_type, '+', name()) )/* "/>
</xsl:element>
</xsl:template>
</xsl:stylesheet>
如果发现困难,我很高兴再次解释这个问题。作为XSLT的初学者,我希望你能帮助我。
答案 0 :(得分:1)
一个简单的问题值得一个简单的答案。这个XSLT 1.0样式表...
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" indent="yes"/>
<xsl:key name="artist-group" match="CD|Music" use="
concat( substring-before(Artist,'.'),'|',music_no,'|',music_type)"/>
<xsl:template match="/">
<version_3>
<information>
<xsl:apply-templates select="t/Data/CD
[generate-id(.) = generate-id( key('artist-group',
concat( substring-before(Artist,'.'),'||'))[1])]
[substring-before(Artist,'.')]" />
<xsl:apply-templates select="key('artist-group','||')/self::CD" />
</information>
<information>
<xsl:apply-templates select="t/Music
[generate-id(.) = generate-id( key('artist-group',
concat( substring-before(Artist,'.'),'|',music_no,'|',music_type))[1])]
[substring-before(Artist,'.')]" />
</information>
<information>
<xsl:apply-templates select="t/Music[substring-before(Artist,'.') = '']" />
</information>
</version_3>
</xsl:template>
<xsl:template match="CD[ substring-before(Artist,'.') != ''] |
Music[ substring-before(Artist,'.') != '']">
<xsl:element name="{substring-before(Artist,'.')}">
<xsl:for-each select="key('artist-group',
concat( substring-before(Artist,'.'),'|',music_no,'|',music_type))" >
<xsl:element name="{substring-after(Artist,'.')}">
<xsl:value-of select="song|value" />
</xsl:element>
</xsl:for-each>
</xsl:element>
</xsl:template>
<xsl:template match="CD[ substring-before(Artist,'.') = ''] |
Music[ substring-before(Artist,'.') = '']">
<xsl:element name="{Artist}">
<xsl:value-of select="song|value" />
</xsl:element>
</xsl:template>
</xsl:stylesheet>
...当应用于您的样本输入时,将产生您所需的预期输出。
单个键用于对Music和CD元素进行分组。在音乐的情况下,进一步划分为music_no和music_type。第一输出信息节点是从CD元素的muenchian分组导出的。第一个信息节点中的孤立艺术家是由指令产生的。
常见的对CD |音乐模板用于所有分组艺术家和孤儿艺术家节点制作。第一个模板对,谓词[substring-before(艺术家,&#39;。&#39;)!=&#39;&#39;]用于分组艺术家,另一个用于孤儿艺术家。
第二个信息节点的制作与第一个节点类似,不同之处在于它基于输入音乐节点并且不包括孤立的艺术家。
第三个信息节点的生成源自孤立的音乐节点。
“孤儿&#39;”是指其Artist值不包含点字符的节点。