使用XSL从XML文件重复第一个节点
发布时间:2020-12-16 23:00:25 所属栏目:百科 来源:网络整理
导读:我的智慧结束了,并想知道这是否可能与XSL.可以说我有这个 XML数据: ?xml version="1.0" standalone="yes"?Data Row F1Created By/F1 F2City/F2 /Row Row F1John Doe/F1 F2Los Angeles/F2 /Row Row F1Jane Doe/F1 F2San Diego/F2 /Row/Data 我想在迭代其余数
我的智慧结束了,并想知道这是否可能与XSL.可以说我有这个
XML数据:
<?xml version="1.0" standalone="yes"?> <Data> <Row> <F1>Created By</F1> <F2>City</F2> </Row> <Row> <F1>John Doe</F1> <F2>Los Angeles</F2> </Row> <Row> <F1>Jane Doe</F1> <F2>San Diego</F2> </Row> </Data> 我想在迭代其余数据时重复第一行元素.总之,我希望输出为: [Created By] [City] ----------- ------------ [John Doe] [Los Angeles] [Created By] [City] ---------- ------------ [Jane Doe] [San Diego] 什么是最好的方法?我尝试将第一个元素’Created By’设置为变量,但是当我尝试使用它时它不会渲染.我是XSL的新手,任何帮助都会受到赞赏. 谢谢 解决方法
这是一个解决方案,与当前接受的答案不同,它可以正确处理具有各种未知长度的数据 – 请参阅进一步处理无限数量列的更新:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <xsl:output method="text"/> <xsl:variable name="vMax1"> <xsl:call-template name="maxLength"> <xsl:with-param name="pNodes" select="/*/*/*[1]"/> </xsl:call-template> </xsl:variable> <xsl:variable name="vMax2"> <xsl:call-template name="maxLength"> <xsl:with-param name="pNodes" select="/*/*/*[2]"/> </xsl:call-template> </xsl:variable> <xsl:variable name="vLongest1" select= "/*/*/*[1][string-length() = $vMax1][1]"/> <xsl:variable name="vLongest2" select= "/*/*/*[2][string-length() = $vMax2][1]"/> <xsl:variable name="vUnderscores1" select= "concat('__',translate($vLongest1,'_',''),'_________________________________________________________') )"/> <xsl:variable name="vUnderscores2" select= "concat('__',translate($vLongest2,'_________________________________________________________') )"/> <xsl:variable name="vBlanks1" select="translate($vUnderscores1,' ')"/> <xsl:variable name="vBlanks2" select="translate($vUnderscores2,' ')"/> <xsl:variable name="vTitle1" select= "concat('[',/*/*/*[1],']',substring($vBlanks1,1,string-length($vBlanks1)-string-length(/*/*/*[1])) )"/> <xsl:variable name="vTitle2" select= "concat('[',/*/*/*[2],substring($vBlanks2,string-length($vBlanks2)-string-length(/*/*/*[2])) )"/> <xsl:template match="Row"> <xsl:value-of select= "concat('
',$vTitle1,$vTitle2)"/> <xsl:value-of select= "concat('
',$vUnderscores1,' ',$vUnderscores2,'
')"/> <xsl:value-of select= "concat(F1,string-length($vBlanks1)-string-length(F1)),F2,string-length($vBlanks1)-string-length(F2)),'
' )"/> </xsl:template> <xsl:template name="maxLength"> <xsl:param name="pNodes" select="/.."/> <xsl:for-each select="$pNodes"> <xsl:sort select="string-length()" data-type="number" order="descending"/> <xsl:if test="position() = 1"> <xsl:value-of select="string-length()"/> </xsl:if> </xsl:for-each> </xsl:template> <xsl:template match="Row[1]|text()"/> </xsl:stylesheet> 在提供的XML文档上应用此转换时: <Data> <Row> <F1>Created By</F1> <F2>City</F2> </Row> <Row> <F1>John Doe</F1> <F2>Los Angeles</F2> </Row> <Row> <F1>Jane Doe</F1> <F2>San Diego</F2> </Row> </Data> 产生了想要的正确结果: [Created By] [City] ____________ _____________ John Doe Los Angeles [Created By] [City] ____________ _____________ Jane Doe San Diego 更有趣的是,当应用于此XML文档时: <Data> <Row> <F1>Created By</F1> <F2>City</F2> </Row> <Row> <F1>John Doe</F1> <F2>La Villa Real de la Santa Fe de San Francisco de Asis</F2> </Row> <Row> <F1>Josiah Willard Gibbs</F1> <F2>San Diego</F2> </Row> </Data> 再次产生正确的结果: [Created By] [City] ______________________ _______________________________________________________ John Doe La Villa Real de la Santa Fe de San Francisco de Asis [Created By] [City] ______________________ _______________________________________________________ Josiah Willard Gibbs San Diego 将此正确结果与当前接受的答案产生的结果进行比较: Created By City ----------- ----------- John Doe La Villa Real de la Santa Fe de San Francisco de Asis Created By City ----------- ----------- Josiah Willard GibbsSan Diego II同一解决方案的XSLT 2.0变体:更短更容易编写: <xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:xs="http://www.w3.org/2001/XMLSchema"> <xsl:output method="text"/> <xsl:variable name="vMax" as="xs:integer*" select= "for $i in 1 to 2 return max(/*/*/*[$i]/string-length(.))"/> <xsl:variable name="vUnderscores" select= "for $i in 1 to 2 return concat('__',string-join((for $len in 1 to $vMax[$i] return '_'),'') )"/> <xsl:variable name="vBlanks" select= "for $i in 1 to 2 return translate($vUnderscores[$i],' ') "/> <xsl:variable name="vTitle" select= "for $i in 1 to 2 return concat('[',(*/*/*)[$i],substring($vBlanks[$i],$vMax[$i]+2 -string-length((/*/*/*)[$i])) ) "/> <xsl:template match="Row"> <xsl:value-of select= "concat('
',$vTitle[1],$vTitle[2])"/> <xsl:value-of select= "concat('
',$vUnderscores[1],$vUnderscores[2],substring($vBlanks[1],$vMax[1]+2 -string-length(F1)),substring($vBlanks[2],$vMax[1]+2 -string-length(F2)),'
' )"/> </xsl:template> <xsl:template match="Row[1]|text()"/> </xsl:stylesheet> III.处理有限数量的列: <xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:xs="http://www.w3.org/2001/XMLSchema"> <xsl:output method="text"/> <xsl:variable name="vNumCols" select="max(/*/*/count(*))"/> <xsl:variable name="vMax" as="xs:integer*" select= "for $i in 1 to $vNumCols return max(/*/*/*[$i]/string-length(.))"/> <xsl:variable name="vUnderscores" select= "for $i in 1 to $vNumCols return concat('__',string-join((for $len in 1 to $vMax[$i] return '_'),'') )"/> <xsl:variable name="vBlanks" select= "for $i in 1 to $vNumCols return translate($vUnderscores[$i],' ') "/> <xsl:variable name="vTitle" select= "for $i in 1 to $vNumCols return concat('[',$vMax[$i]+2 -string-length((/*/*/*)[$i])) ) "/> <xsl:template match="Row"> <xsl:value-of separator="" select= "'
',string-join($vTitle,'')"/> <xsl:value-of separator="" select= "'
',string-join($vUnderscores,' '),'
'"/> <xsl:value-of select= "string-join((for $i in 1 to $vNumCols,$vChild in *[$i] return ($vChild,$vMax[$i]+2 -string-length($vChild) ),' ' ),'
' ),'' )"/> </xsl:template> <xsl:template match="Row[1]|text()"/> </xsl:stylesheet> 应用于此XML文档(3列)时: <Data> <Row> <F1>Created By</F1> <F2>City</F2> <F3>Region</F3> </Row> <Row> <F1>Pablo Diego Ruiz y Picasso</F1> <F2>Los Angeles</F2> <F3>CA</F3> </Row> <Row> <F1>Jane Doe</F1> <F2>La Villa Real de la Santa Fe de San Francisco de Asis</F2> <F3>NM</F3> </Row> </Data> 产生了想要的正确结果: [Created By] [City] [Region] ____________________________ _______________________________________________________ ________ Pablo Diego Ruiz y Picasso Los Angeles CA [Created By] [City] [Region] ____________________________ _______________________________________________________ ________ Jane Doe La Villa Real de la Santa Fe de San Francisco de Asis NM IV.翻译成第III部分(上文)的XSLT 1.0: 请注意: JLR提供的解决方案作为对其初始答案的更正,如果每行有4列,则将执行400次排序. 以下代码中没有这样的低效率: <xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:ext="http://exslt.org/common"> <xsl:output omit-xml-declaration="yes" indent="yes"/> <xsl:variable name="vMaxCols"> <xsl:call-template name="maxChildren"> <xsl:with-param name="pNodes" select="/*/*"/> </xsl:call-template> </xsl:variable> <xsl:variable name="vrtfMax"> <xsl:for-each select="(/*/*/*)[not(position() > $vMaxCols)]"> <xsl:variable name="vPos" select="position()"/> <length> <xsl:call-template name="maxLength"> <xsl:with-param name="pNodes" select="/*/*/*[position()=$vPos]"/> </xsl:call-template> </length> </xsl:for-each> </xsl:variable> <xsl:variable name="vMax" select="ext:node-set($vrtfMax)/length"/> <xsl:variable name="vrtfUnderscores"> <xsl:for-each select="(/*/*/*)[not(position() > $vMaxCols)]"> <xsl:variable name="vPos" select="position()"/> <xsl:variable name="vLongestDataNode" select= "/*/*/*[position()=$vPos and string-length() = $vMax[position() = $vPos]][1]"/> <t> <xsl:value-of select= "concat('__',translate($vLongestDataNode,'_________________________________________________________') )"/> </t> </xsl:for-each> </xsl:variable> <xsl:variable name="vUnderscores" select="ext:node-set($vrtfUnderscores)/t"/> <xsl:variable name="vrtfBlanks"> <xsl:for-each select="$vUnderscores"> <xsl:variable name="vPos" select="position()"/> <t><xsl:value-of select= "translate($vUnderscores[position()=$vPos],' ')"/> </t> </xsl:for-each> </xsl:variable> <xsl:variable name="vBlanks" select="ext:node-set($vrtfBlanks)/t"/> <xsl:variable name="vrtfTitle"> <xsl:for-each select="/*/*[1]/*"> <xsl:variable name="vPos" select="position()"/> <t> <xsl:value-of select= "concat('[',.,substring($vBlanks[position()=$vPos],2+$vMax[position()=$vPos]-string-length()) ) "/> </t> </xsl:for-each> </xsl:variable> <xsl:variable name="vTitle" select="ext:node-set($vrtfTitle)/t"/> <xsl:template match="Row"> <xsl:text>
</xsl:text> <xsl:for-each select="$vTitle"> <xsl:value-of select="."/> </xsl:for-each> <xsl:text>
</xsl:text> <xsl:for-each select="$vUnderscores"> <xsl:value-of select="."/> <xsl:text> </xsl:text> </xsl:for-each> <xsl:text>
</xsl:text> <xsl:for-each select="*[not(position() > $vMaxCols)]"> <xsl:variable name="vPos" select="position()"/> <xsl:value-of select= "concat(.,string-length($vBlanks[position()=$vPos]) -string-length()),' ' )"/> </xsl:for-each> <xsl:text>
</xsl:text> </xsl:template> <xsl:template name="maxChildren"> <xsl:param name="pNodes" select="/.."/> <xsl:for-each select="$pNodes"> <xsl:sort select="count(*)" data-type="number" order="descending"/> <xsl:if test="position() = 1"> <xsl:value-of select="count(*)"/> </xsl:if> </xsl:for-each> </xsl:template> <xsl:template name="maxLength"> <xsl:param name="pNodes" select="/.."/> <xsl:for-each select="$pNodes"> <xsl:sort select="string-length()" data-type="number" order="descending"/> <xsl:if test="position() = 1"> <xsl:value-of select="string-length()"/> </xsl:if> </xsl:for-each> </xsl:template> <xsl:template match="Row[1]|text()"/> </xsl:stylesheet> (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |