XSLT如果选择包含“值”

时间:2012-06-11 17:54:29

标签: xml xslt xml-parsing biztalk hl7

我想写出4个问题是否存在。到目前为止,我的方法是选择=“// NTE_NotesAndCommentsSegment_2 / NTE_3_Comment”,它会检索所有3条评论。但是我遇到了麻烦

  1. 选择NTE_3_Comment,其中包含“问题1”(字符串值)

  2. 当问题不存在时写出问题4

  3. 我还需要为SETID输出正确的数字。

  4. 注意:问题实际上并没有数字。我正在使用ID对输出进行排序。

    输入XML:

    <NTE_NotesAndCommentsSegment_2>
        <NTE_1_SetIdNotesAndComments>1</NTE_1_SetIdNotesAndComments>
        <NTE_2_SourceOfComment></NTE_2_SourceOfComment>
        <NTE_3_Comment>Question 1 ? Answer 1</NTE_3_Comment>
    </NTE_NotesAndCommentsSegment_2>
    <NTE_NotesAndCommentsSegment_2>
        <NTE_1_SetIdNotesAndComments>1</NTE_1_SetIdNotesAndComments>
        <NTE_2_SourceOfComment></NTE_2_SourceOfComment>
        <NTE_3_Comment>Question 2 ? Answer 2</NTE_3_Comment>
    </NTE_NotesAndCommentsSegment_2>
    <NTE_NotesAndCommentsSegment_2>
        <NTE_1_SetIdNotesAndComments>1</NTE_1_SetIdNotesAndComments>
        <NTE_2_SourceOfComment></NTE_2_SourceOfComment>
        <NTE_3_Comment>Question 3? Answer 3</NTE_3_Comment>
    </NTE_NotesAndCommentsSegment_2>
    

    预期输出XML:

    <NTE_NotesAndCommentsSegment_2>
        <NTE_1_SetIdNotesAndComments>1</NTE_1_SetIdNotesAndComments>
        <NTE_2_SourceOfComment></NTE_2_SourceOfComment>
        <NTE_3_Comment>Question 1 ? Answer 1</NTE_3_Comment>
    </NTE_NotesAndCommentsSegment_2>
    <NTE_NotesAndCommentsSegment_2>
        <NTE_1_SetIdNotesAndComments>2</NTE_1_SetIdNotesAndComments>
        <NTE_2_SourceOfComment></NTE_2_SourceOfComment>
        <NTE_3_Comment>Question 2 ? Answer 2</NTE_3_Comment>
    </NTE_NotesAndCommentsSegment_2>
    <NTE_NotesAndCommentsSegment_2>
        <NTE_1_SetIdNotesAndComments>3</NTE_1_SetIdNotesAndComments>
        <NTE_2_SourceOfComment></NTE_2_SourceOfComment>
        <NTE_3_Comment>Question 3 ? Answer 3</NTE_3_Comment>
    </NTE_NotesAndCommentsSegment_2>
    <NTE_NotesAndCommentsSegment_2>
        <NTE_1_SetIdNotesAndComments>4</NTE_1_SetIdNotesAndComments>
        <NTE_2_SourceOfComment></NTE_2_SourceOfComment>
        <NTE_3_Comment>Question 4 ? *Blank* </NTE_3_Comment>
    </NTE_NotesAndCommentsSegment_2>
    

    我正在寻找可以帮助我改变此解决方案方法的建议。在此先感谢。

    解决方案:感谢@ O.R.Mapper建议。无论问题是否存在,每次都会写出所有四个问题。如果源中不存在问题,答案将显示为空白。

                <NTE_NotesAndCommentsSegment_2>
                <NTE_1_SetIdNotesAndComments>1</NTE_1_SetIdNotesAndComments>
                <NTE_3_Comment>Question 1 ?<xsl:value-of select="substring-after(//NTE_NotesAndCommentsSegment_2/NTE_3_Comment[starts-with(text(),'Question 1')],'?')"/>
                </NTE_3_Comment>
                </NTE_NotesAndCommentsSegment_2>
    
                <NTE_NotesAndCommentsSegment_2>
                <NTE_1_SetIdNotesAndComments>2</NTE_1_SetIdNotesAndComments>
                <NTE_3_Comment>Question 2 ?<xsl:value-of select="substring-after(//NTE_NotesAndCommentsSegment_2/NTE_3_Comment[starts-with(text(),'Question 2')],'?')"/>
                </NTE_3_Comment>
                </NTE_NotesAndCommentsSegment_2>
    
                <NTE_NotesAndCommentsSegment_2>
                <NTE_1_SetIdNotesAndComments>3</NTE_1_SetIdNotesAndComments>
                <NTE_3_Comment>Question 3 ?<xsl:value-of select="substring-after(//NTE_NotesAndCommentsSegment_2/NTE_3_Comment[starts-with(text(),'Question 3')],'?')"/>
                </NTE_3_Comment>
                </NTE_NotesAndCommentsSegment_2>
    
                <NTE_NotesAndCommentsSegment_2>
                <NTE_1_SetIdNotesAndComments>4</NTE_1_SetIdNotesAndComments>
                <NTE_3_Comment>Question 4 ? <xsl:value-of select="substring-after(//NTE_NotesAndCommentsSegment_2/NTE_3_Comment[starts-with(text(),'Question 4')],'?')"/>
                </NTE_3_Comment>
                </NTE_NotesAndCommentsSegment_2>
    

2 个答案:

答案 0 :(得分:0)

(1)的解决方案是以下XPath表达式:

//NTE_NotesAndCommentsSegment_2/NTE_3_Comment[starts-with(text(), 'Question 1')]

答案 1 :(得分:0)

完整的xslt解决方案可能如下所示:

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

    <xsl:template match="/">
        <xsl:apply-templates />
    </xsl:template>

    <xsl:template match="root">
        <root>
            <xsl:call-template name="for.loop">
                <xsl:with-param name="i" select="1" />
                <xsl:with-param name="count" select="4" />
                <xsl:with-param name="answer" select="substring-after(//NTE_NotesAndCommentsSegment_2[position() = 1]/NTE_3_Comment, '?')" />
            </xsl:call-template>
        </root>
    </xsl:template>

    <xsl:template name="for.loop">

        <xsl:param name="i" />
        <xsl:param name="count" />
        <xsl:param name="answer" />

        <xsl:if test="$i &lt;= $count">
            <NTE_NotesAndCommentsSegment_2>
                <NTE_1_SetIdNotesAndComments>
                    <xsl:value-of select="$i" />
                </NTE_1_SetIdNotesAndComments>
                <NTE_2_SourceOfComment></NTE_2_SourceOfComment>
                <NTE_3_Comment>
                <xsl:choose>
                    <xsl:when test="$answer = ''">
                        <xsl:value-of select="concat('Question ', $i, ' ? *Blank*')" />
                    </xsl:when>
                    <xsl:otherwise>
                        <xsl:value-of select="concat('Question ', $i, ' ? ', $comment)"/>
                    </xsl:otherwise>
               </xsl:choose>
               </NTE_3_Comment>
            </NTE_NotesAndCommentsSegment_2>
        </xsl:if>

        <!-- Repeat the loop until finished -->
        <xsl:if test="$i &lt;= $count">
            <xsl:call-template name="for.loop">
                <xsl:with-param name="i" select="$i + 1" />
                <xsl:with-param name="count" select="$count">
                <xsl:with-param name="comment" select="substring-after(//NTE_NotesAndCommentsSegment_2[position() = $i + 1]/NTE_3_Comment, '?')" />
            </xsl:call-template>
        </xsl:if>

    </xsl:template>

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

</xsl:stylesheet>