使用 XSLT 节点值的 XML 到 CSV 未显示

Posted

技术标签:

【中文标题】使用 XSLT 节点值的 XML 到 CSV 未显示【英文标题】:XML to CSV using XSLT node values not showing up 【发布时间】:2019-09-22 03:59:30 【问题描述】:

我是一个新手,正在编写从 XML 到 CSV 的 XSLT 转换。下面是它应该看起来像的输出文件(期望的结果):

21072,Charles,Bradley,401K,377.96
21072,Charles,Bradley,401K-R,94.49
21082,Alex,Grossman,401K,10657.15
3,11129.6

但我得到了这个,不知道如何获得计数和总数,还完全排除了 401(k) T 的行。

21072,Charles,Bradley,401(k) [USA],377.96,
401(k) Roth [USA],94.49,
401(k) T,94.49,
21082,Alex,Grossman,401(k) [USA],10657.15,
401(k) T,10657.15,

我在 XSLT 之后尝试过这个

<xsl:stylesheet 
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema" 
    xmlns:wd="urn:com.workday.report/CR-TestReport" 
    exclude-result-prefixes="xs wd"
    version="2.0">    
    <xsl:output method="text"/>
    <xsl:param name="separator" select="','"/>
    <xsl:param name="linefeed" select="'&#13;&#10;'"/>


    <xsl:template match="/">
        <xsl:call-template name="DetailRecords"/>
        <xsl:call-template name="TrailerRecord"/>
    </xsl:template>

    <!--Builds Detail Records -->
    <xsl:template name="DetailRecords">

        <xsl:for-each select = "wd:Report_Data/wd:Report_Entry">
            <xsl:value-of select="wd:Employee_ID"/>
            <xsl:value-of select="$separator"/>
            <xsl:value-of select="wd:First_Name"/>
            <xsl:value-of select="$separator"/>
            <xsl:value-of select="wd:Last_Name"/>
            <xsl:value-of select="$separator"/>
            <xsl:for-each select = "wd:Payroll_Result_Lines">
                <xsl:value-of select="wd:Deduction/@wd:Descriptor"/>
                <xsl:value-of select="$separator"/>
                <xsl:value-of select = "wd:Result_Line_Amount"/>
                <xsl:value-of select="$separator"/>            
           <xsl:value-of select="$linefeed"/>

           </xsl:for-each>

        </xsl:for-each>

    </xsl:template>

    <!--Builds Trailer Record -->
    <xsl:template name="TrailerRecord">

    </xsl:template>

</xsl:stylesheet>

对于这个 XML 文件。

<?xml version='1.0' encoding='UTF-8'?>
<wd:Report_Data xmlns:wd="urn:com.workday.report/CR-TestReport">
  <wd:Report_Entry>
    <wd:Employee_ID>21072</wd:Employee_ID>
    <wd:Last_Name>Bradley</wd:Last_Name>
    <wd:First_Name>Charles</wd:First_Name>
    <wd:Payroll_Result_Lines>
        <wd:Deduction wd:Descriptor="401(k) [USA]">
            <wd:ID wd:type="WID">ff68c0f1227c41a2a5221f6a7e6a67e5</wd:ID>
            <wd:ID wd:type="Deduction_Code">401K</wd:ID>
        </wd:Deduction>
        <wd:Result_Line_Amount>377.96</wd:Result_Line_Amount>
    </wd:Payroll_Result_Lines>
    <wd:Payroll_Result_Lines>
        <wd:Deduction wd:Descriptor="401(k) Roth [USA]">
            <wd:ID wd:type="WID">2d277641684a48fb861a896e4eeb3eea</wd:ID>
            <wd:ID wd:type="Deduction_Code">401K-R</wd:ID>
        </wd:Deduction>
        <wd:Result_Line_Amount>94.49</wd:Result_Line_Amount>
    </wd:Payroll_Result_Lines>
    <wd:Payroll_Result_Lines>
        <wd:Deduction wd:Descriptor="401(k) T">
            <wd:ID wd:type="WID">2d277641684a48fb861a896e4eeb3eea</wd:ID>
            <wd:ID wd:type="Deduction_Code">401K-T</wd:ID>
        </wd:Deduction>
        <wd:Result_Line_Amount>94.49</wd:Result_Line_Amount>
    </wd:Payroll_Result_Lines>
</wd:Report_Entry>
<wd:Report_Entry>
    <wd:Employee_ID>21082</wd:Employee_ID>
    <wd:Last_Name>Grossman</wd:Last_Name>
    <wd:First_Name>Alex</wd:First_Name>
    <wd:Payroll_Result_Lines>
        <wd:Deduction wd:Descriptor="401(k) [USA]">
            <wd:ID wd:type="WID">ff68c0f1227c41a2a5221f6a7e6a67e5</wd:ID>
            <wd:ID wd:type="Deduction_Code">401K</wd:ID>
        </wd:Deduction>
        <wd:Result_Line_Amount>10657.15</wd:Result_Line_Amount>
    </wd:Payroll_Result_Lines>
    <wd:Payroll_Result_Lines>
        <wd:Deduction wd:Descriptor="401(k) T">
            <wd:ID wd:type="WID">ff68c0f1227c41a2a5221f6a7e6a67e5</wd:ID>
            <wd:ID wd:type="Deduction_Code">401K-T</wd:ID>
        </wd:Deduction>
        <wd:Result_Line_Amount>10657.15</wd:Result_Line_Amount>
    </wd:Payroll_Result_Lines>
  </wd:Report_Entry>
</wd:Report_Data>

【问题讨论】:

您能用文字解释一下所需的逻辑吗?例如。我希望每个 ___ 都有一行,并且在每一行中我都希望有这些值:___。 嗨,这是要求,输出将是一个逗号分隔的文件,每条记录输出在单独的行上。 1. 详细记录应包含以下字段(按此顺序): 员工 ID 名 姓 扣除代码 结果行 金额 尾记录应包含此数据(按此顺序): 文件上的记录总数 总金额文件 2.Remove 工资单结果,扣除代码为“401K-T”。更新您的代码以排除这些记录。谢谢,斯里尼 【参考方案1】:

当您使用 XSLT 2 创建 CSV 时,我建议您利用 xsl:value-of separator 在您的 CSV 中“构建”一行或记录

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    xpath-default-namespace="urn:com.workday.report/CR-TestReport"
    xmlns:wd="urn:com.workday.report/CR-TestReport"
    exclude-result-prefixes="#all"
    version="3.0">

  <xsl:output method="text"/>

    <xsl:param name="separator" select="','"/>
    <xsl:param name="linefeed" select="'&#13;&#10;'"/>

  <xsl:template match="/">
      <xsl:variable name="rows" select="//Report_Entry/Payroll_Result_Lines[Deduction[not(ID[@wd:type = 'Deduction_Code'] = '401K-T')]]"/>
      <xsl:apply-templates select="$rows"/>
      <xsl:value-of select="count($rows), sum($rows/Result_Line_Amount)" separator="$separator"/>
  </xsl:template>

  <xsl:template match="Payroll_Result_Lines">
      <xsl:value-of select="../(Employee_ID, First_Name, Last_Name), Deduction/@wd:Descriptor, Result_Line_Amount" separator="$separator"/>
      <xsl:value-of select="$linefeed"/>
  </xsl:template>

</xsl:stylesheet>

https://xsltfiddle.liberty-development.net/jyRYYiE

【讨论】:

【参考方案2】:

我无法发表评论,但我得到的答案与@zx485 相同,只是我删除了终止分隔符。上面想要的结果没有每行的尾随逗号:

            <xsl:value-of select = "wd:Result_Line_Amount"/>
            <xsl:value-of select="$linefeed"/>
        </xsl:for-each>

我希望这会有所帮助并希望分享 - 我是 SO 新手,因此不确定提供的最佳方式,因为 zx 的答案非常接近。

【讨论】:

【参考方案3】:

要获得所需的结果,您可以将模板更改为以下内容。

<!--Builds Detail Records -->
<xsl:template name="DetailRecords">
    <xsl:for-each select = "wd:Report_Data/wd:Report_Entry">
        <xsl:for-each select = "wd:Payroll_Result_Lines[wd:Deduction/wd:ID[@wd:type='Deduction_Code'] != '401K-T']">
            <xsl:value-of select="../wd:Employee_ID"/>
            <xsl:value-of select="$separator"/>
            <xsl:value-of select="../wd:First_Name"/>
            <xsl:value-of select="$separator"/>
            <xsl:value-of select="../wd:Last_Name"/>
            <xsl:value-of select="$separator"/>
            <xsl:value-of select="wd:Deduction/wd:ID[@wd:type='Deduction_Code']"/>
            <xsl:value-of select="$separator"/>
            <xsl:value-of select = "wd:Result_Line_Amount"/>
            <xsl:value-of select="$separator"/>            
       <xsl:value-of select="$linefeed"/>
       </xsl:for-each>
    </xsl:for-each>
    <xsl:value-of select="count(wd:Report_Data/wd:Report_Entry/wd:Payroll_Result_Lines[wd:Deduction/wd:ID[@wd:type='Deduction_Code'] != '401K-T'])"/>
    <xsl:value-of select="$separator"/>            
    <xsl:value-of select="sum(wd:Report_Data/wd:Report_Entry/wd:Payroll_Result_Lines[wd:Deduction/wd:ID[@wd:type='Deduction_Code'] != '401K-T']/wd:Result_Line_Amount)"/> 
</xsl:template>

其结果是:

21072,Charles,Bradley,401K,377.96,
21072,Charles,Bradley,401K-R,94.49,
21082,Alex,Grossman,401K,10657.15,
3,11129.6

【讨论】:

非常感谢 zx485,它现在可以工作了我已经根据你的代码从我的 XSLT 中删除了预告片记录模板 很高兴我能帮上忙。

以上是关于使用 XSLT 节点值的 XML 到 CSV 未显示的主要内容,如果未能解决你的问题,请参考以下文章

XSLT 1.0:CSV 到 XML - 如何分而治之

使用 XSLT 组合具有相同 ID 及其值的节点 (XML)

使用 Xslt 将 Xml 中类似节点的数据导出到 excel

使用 Python 或 XSLT 将复杂的 XML 转换为 CSV

XML 到 CSV 使用 XSLT 帮助

使用 XSLT 将具有相同 ID 的元素 (XML) 合并到 txt 文件