Java 如何使用XSLT将递归XML数据打印到CSV文件中
这是我的示例XML数据,它只有属性数据和多个子节点Java 如何使用XSLT将递归XML数据打印到CSV文件中,java,xml,csv,xslt,Java,Xml,Csv,Xslt,这是我的示例XML数据,它只有属性数据和多个子节点 <FIXML s="...." v="...."> <Batch ID="...."> <MktDef MktID="XEUR" MktSegID="14" EfctvBizDt="2017-05-11" NxtEfctvBizDt="2017-05-15" MktSeg="CONF" MarketSegmentDesc="FUT 8-13 Y. SWISS GOV.BONDS 6%" Sym="CH000
<FIXML s="...." v="....">
<Batch ID="....">
<MktDef MktID="XEUR" MktSegID="14" EfctvBizDt="2017-05-11" NxtEfctvBizDt="2017-05-15" MktSeg="CONF" MarketSegmentDesc="FUT 8-13 Y. SWISS GOV.BONDS 6%" Sym="CH0002741988" ParentMktSegmID="FBND" Ccy="CHF" MktSegStat="1" USFirmFlag="Y" PartID="2">
<MtchRules MtchRuleProdCmplx="5" MtchAlgo="PT"/>
<MtchRules MtchRuleProdCmplx="1" MtchAlgo="PT"/>
<FlexProdEligs FlexProdEligCmplx="5" FlexProdElig="Y"/>
<BaseTrdgRules QtSideInd="1" FastMktPctg="0">
<TickRules TickRuleProdCmplx="1" StartTickPxRng="0" EndTickPxRng="99999.9999" TickIncr="0.01"/>
<TickRules TickRuleProdCmplx="5" StartTickPxRng="0" EndTickPxRng="99999.9999" TickIncr="0.01"/>
<QuotSizeRules MinBidSz="1" MinOfrSz="1" FastMktInd="0"/>
<QuotSizeRules MinBidSz="1" MinOfrSz="1" FastMktInd="1"/>
<PxRngRules PxRngRuleID="75" PxRngProdCmplx="1" StartPxRng="0" EndPxRng="99999.9999" PxRngValu="0.15"/>
<PxRngRules PxRngRuleID="347" PxRngProdCmplx="5" StartPxRng="0" EndPxRng="99999.9999" PxRngValu="0.12"/>
</BaseTrdgRules>
<MDFeedTyps MDFeedTyp="HS" MDBkTyp="2" MktDepth="10" MDRcvryTmIntvl="120000" SvcLctnID1="224.0.50.102" SvcLctnSubID1="59500" SvcLctnID2="224.0.50.230" SvcLctnSubID2="59500"/>
<MDFeedTyps MDFeedTyp="HI" MDBkTyp="2" MktDepth="10" MktDepthTmIntvl="0" SvcLctnID1="224.0.50.103" SvcLctnSubID1="59501" SvcLctnID2="224.0.50.231" SvcLctnSubID2="59501"/>
<MDFeedTyps MDFeedTyp="HI" MDBkTyp="3" MktDepthTmIntvl="0" SvcLctnID1="224.0.114.97" SvcLctnSubID1="59501" SvcLctnID2="224.0.114.113" SvcLctnSubID2="59501"/>
<MDFeedTyps MDFeedTyp="HS" MDBkTyp="3" SvcLctnID1="224.0.114.96" SvcLctnSubID1="59500" SvcLctnID2="224.0.114.112" SvcLctnSubID2="59500"/>
<MDFeedTyps MDFeedTyp="L" MDBkTyp="2" MktDepth="5" MktDepthTmIntvl="3500" MDRcvryTmIntvl="30000" SvcLctnID1="224.0.50.89" SvcLctnSubID1="59500" SvcLctnID2="224.0.50.217" SvcLctnSubID2="59500"/>
</MktDef>
<SecDef PriSetPx="158.39">
<Instrmt ID="408805" Src="M" SecTyp="FUT" Status="1" ProdCmplx="1" CFI="FFMPSX" MatDt="2017-06-08" MMY="201706" Mult="1" ValMeth="FUT" SettlMeth="P" SettlSubMeth="4" PxPrcsn="2" MinPxIncr="0.01" MinPxIncrAmt="10">
<AID AltID="1048612" AltIDSrc="M"/>
<AID AltID="XF000001RQD8" AltIDSrc="4"/>
<Evnt EventTyp="7" Dt="2017-06-08"/>
</Instrmt>
<MktSegGrp MktSegID="14">
<SecTrdgRules>
<BaseTrdgRules>
<PxRngRules PxRngRuleID="75"/>
</BaseTrdgRules>
</SecTrdgRules>
</MktSegGrp>
</SecDef>
.
.
.
.
</Batch>
</FIXML>
"<MktDef> and <SecDef>" will be continued for N number of times.
在这里,我想得到以下格式的输出
MktID MktSegID PxRngRuleID PxRngProdCmplx StartPxRng EndPxRng PxRngValu
XEUR 19629 75 1 0 9999.9999 0.15
XEUR 65 75 1 0 99999.9999 0.15
XEUR 66 75 1 0 99999.9999 0.15
XEUR 67 75 1 0 99999.9999 0.15
我的问题是,我只获取第一个“PxRngRules”标记数据。我无法获取所有出现的“PxRngRules”信息。我可能误解了您的要求,但如果您希望在
MktDef下为每个PxRngRules
多行,可以尝试使用此XSLT
<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="text"/>
<xsl:template match="/">
<xsl:text>MktID,MktSegID,PxRngRuleID,PxRngProdCmplx,StartPxRng,EndPxRng,PxRngValu</xsl:text>
<xsl:text>
</xsl:text>
<xsl:for-each select="FIXML/Batch/MktDef">
<xsl:variable name="mktDef" select="concat(@MktID,',',@MktSegID)" />
<xsl:choose>
<xsl:when test="BaseTrdgRules/PxRngRules">
<xsl:for-each select="BaseTrdgRules/PxRngRules">
<xsl:value-of select="concat($mktDef, ',', @PxRngRuleID,',',@PxRngProdCmplx,',',@StartPxRng,',',@EndPxRng,',',@PxRngValu,'
')"/>
</xsl:for-each>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="concat($mktDef, ',,,,,
')"/>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
您是否只对具有PxRngRules子项的MktDef条目感兴趣?这与您的问题无关,但不要仅仅为了向转换提供输入而创建DOM。让XSLT处理器通过提供StreamSource以自己的方式构建树。您好NMGod,是的,我只需要具有PxRngRules子项的MktDef条目,您好Michael Kay,您能提供一个示例程序或POC来使用StreamSource进行转换吗。非常感谢。那么,您真的希望在MktDef
下为每个PxRngRules
指定一行吗?每个MktDef
元素下是否总是至少有一个PxRngRules
?嗨,Tim C,谢谢你的回复。太完美了。它工作得很好。我能问你更多的问题吗?如果我想在BaseTrdgRules标记中添加两个值为“QtSideInd和FastMktPctg”的列,你能告诉我怎么做吗?非常感谢您的支持。您可以在concat station中使用。/@qtsidend
来获取父级的属性值。非常感谢Tim C。
<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="text"/>
<xsl:template match="/">
<xsl:text>MktID,MktSegID,PxRngRuleID,PxRngProdCmplx,StartPxRng,EndPxRng,PxRngValu</xsl:text>
<xsl:text>
</xsl:text>
<xsl:for-each select="FIXML/Batch/MktDef">
<xsl:variable name="mktDef" select="concat(@MktID,',',@MktSegID)" />
<xsl:choose>
<xsl:when test="BaseTrdgRules/PxRngRules">
<xsl:for-each select="BaseTrdgRules/PxRngRules">
<xsl:value-of select="concat($mktDef, ',', @PxRngRuleID,',',@PxRngProdCmplx,',',@StartPxRng,',',@EndPxRng,',',@PxRngValu,'
')"/>
</xsl:for-each>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="concat($mktDef, ',,,,,
')"/>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
<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="text"/>
<xsl:template match="/">
<xsl:text>MktID,MktSegID,PxRngRuleID,PxRngProdCmplx,StartPxRng,EndPxRng,PxRngValu</xsl:text>
<xsl:text>
</xsl:text>
<xsl:for-each select="FIXML/Batch/MktDef">
<xsl:variable name="mktDef" select="concat(@MktID,',',@MktSegID)" />
<xsl:for-each select="BaseTrdgRules/PxRngRules">
<xsl:value-of select="concat($mktDef, ',', @PxRngRuleID,',',@PxRngProdCmplx,',',@StartPxRng,',',@EndPxRng,',',@PxRngValu,'
')"/>
</xsl:for-each>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>