0

感谢这里成员的慷慨反馈,我的 XML 到 XML 项目取得了一些重大进展。

也就是说,我在项目的最终版本中遇到了两个问题,并希望更好地了解为什么会出现这些问题。

首先,我需要创建一个由多个值组成的串联元素值。我使用以下代码进行此工作:

<xsl:template match="estimate/JobParts/JobPart/description">
    <description>
        <xsl:value-of select="concat(estimate/description,'_QTY-',estimate/JobParts/JobPart/qtyOrdered,'_',estimate/JobParts/JobPart/itemTemplate)"/>
    </description>
</xsl:template>

这会找到 的所有匹配estimate/JobParts/JobPart/description项,但在所有情况下,在替换描述值时仅使用第一个匹配的兄弟元素值。

在这种情况下,我需要使用密钥来管理迭代吗?如果是这样,我应该如何创建一个?是否有另一种使用应用模板处理此问题的方法,以便可以自动处理迭代管理,如果是这样,将如何构建?

我遇到的另一个问题是我需要手动将 CDATA 指定添加到元素,并且无法将该元素添加到 cdata-section-elements 的列表中,因为在文档结构中嵌套了另一个同名的元素不需要 CDATA 指定。

这是我试图用来手动添加 CDATA 标志的代码:

<xsl:template match="estimate/description">
    <description>
        <xsl:text disable-output-escaping="yes">
         &lt;![CDATA[
       </xsl:text>
        <xsl:value-of select="estimate/description"/>
        <xsl:text disable-output-escaping="yes">
        ]]&gt;
       </xsl:text>
    </description>
</xsl:template>

有趣的是,这对目标描述元素没有任何影响。

如果有人知道为什么这些转换中的任何一个(或两个)都不能正常工作,我渴望学习和理解。

提前为大量代码道歉,但这里是原始 XML:

<?xml version="1.0"?>
<PODOrderSheet_Main>

    <estimate>
        <customer>LINFNC</customer>
        <newJob>Y</newJob>
        <incrementJobVersion>Y</incrementJobVersion>
        <description><![CDATA[409511]]></description>
        <billPartsTogether>1</billPartsTogether>
        <dateSetup><![CDATA[4/24/2013]]></dateSetup>
        <promiseDate><![CDATA[4/24/2013]]></promiseDate>
        <scheduledShipDate><![CDATA[4/24/2013]]></scheduledShipDate>
        <adminStatus>O</adminStatus>
        <shipVia>1</shipVia> 
        <jobType>5020</jobType>
        <poNum><![CDATA[409511]]></poNum>
        <itemTemplate><![CDATA[33503MN_0212]]></itemTemplate>
        <JobParts>
            <JobPart>
                <jobPart>01</jobPart>
                <contactNum/>
                <description><![CDATA[REPLACEMENT OF LIFE INSURANCE OR ANNUITIES MINNESOTA]]></description>
                <productionStatus>O</productionStatus>
                <qtyOrdered><![CDATA[3]]></qtyOrdered>
                <shipToContact/>
                <itemTemplate><![CDATA[33503MN_0212]]></itemTemplate>
                <JobNotes>
                    <JobNote>
                        <department>001</department>
                        <jobPart>01</jobPart>
                        <note><![CDATA[Rush Order]]></note>
                    </JobNote>
                </JobNotes>
            </JobPart>
        </JobParts>
    </estimate>

    <estimate>
        <customer>LINFNC</customer>
        <newJob>Y</newJob>
        <incrementJobVersion>Y</incrementJobVersion>
        <description><![CDATA[409511]]></description>
        <billPartsTogether>1</billPartsTogether>
        <dateSetup><![CDATA[4/24/2013]]></dateSetup>
        <promiseDate><![CDATA[4/24/2013]]></promiseDate>
        <scheduledShipDate><![CDATA[4/24/2013]]></scheduledShipDate>
        <adminStatus>O</adminStatus>
        <shipVia>1</shipVia> 
        <jobType>5020</jobType>
        <poNum><![CDATA[409511]]></poNum>
        <itemTemplate><![CDATA[AL-FSC-FST068_Z08]]></itemTemplate>
        <JobParts>
            <JobPart>
                <jobPart>01</jobPart>
                <contactNum/>
                <description><![CDATA[AMERICAN LEGACY III C SHARE FACT SHEET ]]></description>
                <productionStatus>O</productionStatus>
                <qtyOrdered><![CDATA[1]]></qtyOrdered>
                <shipToContact/>
                <itemTemplate><![CDATA[AL-FSC-FST068_Z08]]></itemTemplate>
                <JobNotes>
                    <JobNote>
                        <department>001</department>
                        <jobPart>01</jobPart>
                        <note><![CDATA[Rush Order]]></note>
                    </JobNote>
                </JobNotes>
            </JobPart>
        </JobParts>
    </estimate>

    <estimate>
        <customer>LINFNC</customer>
        <newJob>Y</newJob>
        <incrementJobVersion>Y</incrementJobVersion>
        <description><![CDATA[409511]]></description>
        <billPartsTogether>1</billPartsTogether>
        <dateSetup><![CDATA[4/24/2013]]></dateSetup>
        <promiseDate><![CDATA[4/24/2013]]></promiseDate>
        <scheduledShipDate><![CDATA[4/24/2013]]></scheduledShipDate>
        <adminStatus>O</adminStatus>
        <shipVia>1</shipVia> 
        <jobType>5020</jobType>
        <poNum><![CDATA[409511]]></poNum>
        <itemTemplate><![CDATA[AN06819-AL3C_1012]]></itemTemplate>
        <JobParts>
            <JobPart>
                <jobPart>01</jobPart>
                <contactNum/>
                <description><![CDATA[AMERICAN LEGACY III C SHARE APPLICATION SUPPLEMENT  - MULTI-STATE]]></description>
                <productionStatus>O</productionStatus>
                <qtyOrdered><![CDATA[1]]></qtyOrdered>
                <shipToContact/>
                <itemTemplate><![CDATA[AN06819-AL3C_1012]]></itemTemplate>
                <JobNotes>
                    <JobNote>
                        <department>001</department>
                        <jobPart>01</jobPart>
                        <note><![CDATA[Rush Order]]></note>
                    </JobNote>
                </JobNotes>
            </JobPart>
        </JobParts>
    </estimate>

</PODOrderSheet_Main>

这是所需的输出:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE estimate>
<estimate>
    <customer>LINFNC</customer>
    <newJob>Y</newJob>
    <incrementJobVersion>Y</incrementJobVersion>
    <description><![CDATA[409511]]></description>
    <billPartsTogether>1</billPartsTogether>
    <dateSetup><![CDATA[4/24/2013]]></dateSetup>
    <promiseDate><![CDATA[4/24/2013]]></promiseDate>
    <scheduledShipDate><![CDATA[4/24/2013]]></scheduledShipDate>
    <adminStatus>O</adminStatus>
    <shipVia>1</shipVia>
    <jobType>5020</jobType>
    <poNum><![CDATA[409511]]></poNum>
    <itemTemplate><![CDATA[33503MN_0212]]></itemTemplate>
    <JobParts>
        <JobPart>
            <jobPart>01</jobPart>
            <contactNum/>
            <description>409511_QTY-3_33503MN_0212</description>
            <priority>1</priority>
            <productionStatus>O</productionStatus>
            <qtyOrdered><![CDATA[3]]></qtyOrdered>
            <shipToContact/>
            <itemTemplate><![CDATA[33503MN_0212]]></itemTemplate>
            <JobNotes>
                <JobNote>
                    <department>001</department>
                    <jobPart>01</jobPart>
                    <note><![CDATA[Rush Order]]></note>
                </JobNote>
            </JobNotes>
        </JobPart>
        <JobPart>
            <jobPart>02</jobPart>
            <contactNum/>
            <description>409511_QTY-1_AL-FSC-FST068_Z08</description>
            <priority>1</priority>
            <productionStatus>O</productionStatus>
            <qtyOrdered><![CDATA[1]]></qtyOrdered>
            <shipToContact/>
            <itemTemplate><![CDATA[AL-FSC-FST068_Z08]]></itemTemplate>
            <JobNotes>
                <JobNote>
                    <department>001</department>
                    <jobPart>02</jobPart>
                    <note><![CDATA[Rush Order]]></note>
                </JobNote>
            </JobNotes>
        </JobPart>
        <JobPart>
            <jobPart>03</jobPart>
            <contactNum/>
            <description>409511_QTY-1_AN06819-AL3C_1012</description>
            <priority>1</priority>
            <productionStatus>O</productionStatus>
            <qtyOrdered><![CDATA[1]]></qtyOrdered>
            <shipToContact/>
            <itemTemplate><![CDATA[AN06819-AL3C_1012]]></itemTemplate>
            <JobNotes>
                <JobNote>
                    <department>001</department>
                    <jobPart>03</jobPart>
                    <note><![CDATA[Rush Order]]></note>
                </JobNote>
            </JobNotes>
        </JobPart>
    </JobParts>
</estimate>

这是现在的 XSL:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
version="1.0">
<xsl:output indent="yes" encoding="UTF-8" method="xml"/>

<xsl:strip-space elements="*"/>

<xsl:output cdata-section-elements="dateSetup promiseDate scheduledShipDate poNum qtyOrdered itemTemplate note"/>

<xsl:template match="@*|node()">
    <xsl:copy>
        <xsl:apply-templates/>
    </xsl:copy>
</xsl:template>

<xsl:template match="/PODOrderSheet_Main">
    <estimate>
        <xsl:copy-of select="estimate[1]/customer"/>
        <xsl:copy-of select="estimate[1]/newJob"/>
        <xsl:copy-of select="estimate[1]/incrementJobVersion"/>
        <xsl:copy-of select="estimate[1]/description"/>
        <xsl:copy-of select="estimate[1]/billPartsTogether"/>
        <xsl:copy-of select="estimate[1]/dateSetup"/>
        <xsl:copy-of select="estimate[1]/promiseDate"/>
        <xsl:copy-of select="estimate[1]/scheduledShipDate"/>
        <xsl:copy-of select="estimate[1]/adminStatus"/>
        <xsl:copy-of select="estimate[1]/shipVia"/>
        <xsl:copy-of select="estimate[1]/jobType"/>
        <xsl:copy-of select="estimate[1]/poNum"/>
        <xsl:copy-of select="estimate[1]/itemTemplate"/>
        <JobParts>
            <xsl:apply-templates select="estimate/JobParts/JobPart"/>
        </JobParts>
    </estimate>
</xsl:template>

<xsl:template match="estimate/description">
<description>
    <xsl:text disable-output-escaping="yes">
     &lt;![CDATA[
   </xsl:text>
        <xsl:value-of select="estimate/description"/>
    <xsl:text disable-output-escaping="yes">
    ]]&gt;
   </xsl:text>
</description>
</xsl:template>

<xsl:template match="//estimate/JobParts/JobPart/description">
<description>
<xsl:value-of select="concat(//estimate/description,'_QTY-',//JobPart/qtyOrdered,'_',//JobPart/itemTemplate)"/>
</description>
</xsl:template>

<xsl:template match="jobPart">
    <xsl:copy><xsl:number count="JobPart" level="any" format="01"/></xsl:copy>
</xsl:template>

</xsl:stylesheet>

非常感谢...

4

2 回答 2

1

我遇到的另一个问题是我需要手动将 CDATA 指定添加到元素,并且无法将该元素添加到 cdata-section-elements 的列表中,因为在文档结构中嵌套了另一个同名的元素不需要 CDATA 指定。

是否使用 CDATA 部分表示字符数据的特定位对 XML 解析器没有影响,<foo>something</foo><foo><![CDATA[something]]></foo>. 实际上,使用 disable-output-escaping 手动添加 cdata 包装器不一定会产生正确的结果,因为序列化程序没有意识到它输出的文本在 CDATA 中,并且无论如何都会使用实体对其进行转义:如果你有一个description

<description><![CDATA[409 < 511]]></description>

然后尝试使用 doe CDATA 标记将其复制到输出将产生

<description><![CDATA[409 &lt; 511]]></description>

您最好不要理会 CDATA,而让序列化程序以正常方式进行实体转义

<description>409 &lt; 511</description>

(相当于原始元素)。

于 2013-05-05T23:33:39.093 回答
1

您的样式表存在一些问题:

  • 您正在estimate/description使用包含元素xsl:copy-of。因此,您将简单地获得元素的副本,并且estimate/description永远不会应用匹配的模板来转换它。

  • match和属性中的位置路径select可以是绝对的,也可以是对于上下文节点的。绝对位置路径通常是样式表设计不佳的标志,并且很难从几个具有匹配路径的节点中挑选一个。正如您所发现的,通过获取集合中第一个节点的字符串值,将一组节点转换为字符串。position()最好使用相对定位路径,而不是使用某种方式从集合中挑选单个节点。

  • 您似乎不清楚CDATAPCDATA实体的使用。这只是在 XML 的文本节点中表示特殊字符的不同方式,而 和的含义相同。由于您只有两种表示同一文本字符串的替代方法,因此您不太可能特别想要其中一种,但您可能有一个要求其 XML 数据“恰到好处”的软件。<![CDATA[use <> in Perl for reading]]>use &lt;&gt; in Perl for reading

这种转换可以满足您的需要。我已将初始xsl:copy列表更改为仅xsl:apply-templates包含第一个estimate元素本身没有子元素的所有子元素。这样estimate/description模板就会生效并CDATA应用标签。该JobPart/description模板将复合描述字符串所需的三个值提取到三个XSL变量中,以使事情更整洁。

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    version="1.0">

    <xsl:output indent="yes" encoding="UTF-8" method="xml"/>
    <xsl:strip-space elements="*"/>

    <xsl:output cdata-section-elements="dateSetup promiseDate scheduledShipDate poNum itemTemplate qtyOrdered note"/>

    <xsl:template match="node()|@*">
        <xsl:copy>
            <xsl:apply-templates/>
        </xsl:copy>
    </xsl:template>

    <xsl:template match="/PODOrderSheet_Main">
        <estimate>
            <xsl:apply-templates select="estimate[1]/*[not(*)]"/>
            <JobParts>
                <xsl:apply-templates select="estimate/JobParts/JobPart"/>
            </JobParts>
        </estimate>
    </xsl:template>

    <xsl:template match="estimate/description">
        <description>
            <xsl:text disable-output-escaping="yes">&lt;![CDATA[</xsl:text>
            <xsl:value-of select="."/>
            <xsl:text disable-output-escaping="yes">]]&gt;</xsl:text>
        </description>
    </xsl:template>

    <xsl:template match="JobPart/description">
        <xsl:variable name="estimate-description" select="ancestor::estimate/description"/>
        <xsl:variable name="qty-ordered" select="parent::JobPart/qtyOrdered"/>
        <xsl:variable name="item-template" select="parent::JobPart/itemTemplate"/>
        <xsl:copy>
            <xsl:value-of select="concat($estimate-description, '_QTY-', $qty-ordered, '_', $item-template)"/>
        </xsl:copy>
    </xsl:template>

    <xsl:template match="jobPart">
        <xsl:copy><xsl:number count="JobPart" level="any" format="01"/></xsl:copy>
    </xsl:template>

</xsl:stylesheet>

输出

<?xml version="1.0" encoding="UTF-8"?>
<estimate>
   <customer>LINFNC</customer>
   <newJob>Y</newJob>
   <incrementJobVersion>Y</incrementJobVersion>
   <description><![CDATA[409511]]></description>
   <billPartsTogether>1</billPartsTogether>
   <dateSetup><![CDATA[4/24/2013]]></dateSetup>
   <promiseDate><![CDATA[4/24/2013]]></promiseDate>
   <scheduledShipDate><![CDATA[4/24/2013]]></scheduledShipDate>
   <adminStatus>O</adminStatus>
   <shipVia>1</shipVia>
   <jobType>5020</jobType>
   <poNum><![CDATA[409511]]></poNum>
   <itemTemplate><![CDATA[33503MN_0212]]></itemTemplate>
   <JobParts>
      <JobPart>
         <jobPart>01</jobPart>
         <contactNum/>
         <description>409511_QTY-3_33503MN_0212</description>
         <productionStatus>O</productionStatus>
         <qtyOrdered><![CDATA[3]]></qtyOrdered>
         <shipToContact/>
         <itemTemplate><![CDATA[33503MN_0212]]></itemTemplate>
         <JobNotes>
            <JobNote>
               <department>001</department>
               <jobPart>01</jobPart>
               <note><![CDATA[Rush Order]]></note>
            </JobNote>
         </JobNotes>
      </JobPart>
      <JobPart>
         <jobPart>02</jobPart>
         <contactNum/>
         <description>409511_QTY-1_AL-FSC-FST068_Z08</description>
         <productionStatus>O</productionStatus>
         <qtyOrdered><![CDATA[1]]></qtyOrdered>
         <shipToContact/>
         <itemTemplate><![CDATA[AL-FSC-FST068_Z08]]></itemTemplate>
         <JobNotes>
            <JobNote>
               <department>001</department>
               <jobPart>02</jobPart>
               <note><![CDATA[Rush Order]]></note>
            </JobNote>
         </JobNotes>
      </JobPart>
      <JobPart>
         <jobPart>03</jobPart>
         <contactNum/>
         <description>409511_QTY-1_AN06819-AL3C_1012</description>
         <productionStatus>O</productionStatus>
         <qtyOrdered><![CDATA[1]]></qtyOrdered>
         <shipToContact/>
         <itemTemplate><![CDATA[AN06819-AL3C_1012]]></itemTemplate>
         <JobNotes>
            <JobNote>
               <department>001</department>
               <jobPart>03</jobPart>
               <note><![CDATA[Rush Order]]></note>
            </JobNote>
         </JobNotes>
      </JobPart>
   </JobParts>
</estimate>
于 2013-05-05T23:37:48.913 回答