如何用XSL检查一个值是空还是空?

例如,如果categoryName为空?我使用的是“当选择”结构。

例如:

<xsl:choose>
    <xsl:when test="categoryName !=null">
        <xsl:value-of select="categoryName " />
    </xsl:when>
    <xsl:otherwise>
        <xsl:value-of select="other" />
    </xsl:otherwise>
</xsl:choose>

当前回答

我知道这个问题已经很老了,但是在所有的答案中,我忽略了XSLT开发中这个用例的常用方法。

我想象OP中缺失的代码看起来像这样:

<xsl:template match="category">
    <xsl:choose>
        <xsl:when test="categoryName !=null">
            <xsl:value-of select="categoryName " />
        </xsl:when>
        <xsl:otherwise>
            <xsl:value-of select="other" />
        </xsl:otherwise>
    </xsl:choose>
</category>

输入是这样的:

<categories>
    <category>
       <categoryName>Books</categoryName>
    </category>
    <category>
       <categoryName>Magazines</categoryName>
       <categoryName>Periodicals</categoryName>
       <categoryName>Journals</categoryName>
    </category>
    <category>
        <categoryName><!-- please fill in category --></categoryName>
    </category>
    <category>
        <categoryName />
    </category>
    <category />
</categories>

I.e., I assume there can be zero, empty, single or multiple categoryName elements. To deal with all these cases using xsl:choose-style constructs, or in other words, imperatively, is quickly getting messy (even more so if elements can be at different levels!). A typical programming idiom in XSLT is using templates (hence the T in XSLT), which is declarative programming, not imperative (you don't tell the processor what to do, you just tell what you want output if certain conditions are met). For this use-case, that can look something like the following:

<!-- positive test, any category with a valid categoryName -->
<xsl:template match="category[categoryName[text()]]">
    <xsl:apply-templates />
</xsl:template>

<!-- any other category (without categoryName, "null", with comments etc) -->
<xsl:template match="category">
    <xsl:text>Category: Other</xsl:text>
</xsl:template>

<!-- matching the categoryName itself for easy handling of multiple names -->
<xsl:template match="categoryName">
    <xsl:text>Category: </xsl:text>
    <xsl:value-of select="." />
</xsl:template>

这是可行的(对于任何XSLT版本),因为上面的第一个具有更高的优先级(它有一个谓词)。“fall-through”匹配模板(第二个)捕获任何无效的内容。第三个函数负责以适当的方式输出categoryName值。

注意,在此场景中,不需要特别匹配类别或类别,因为处理器将自动处理所有子类,除非我们另有说明(在本例中,第二个和第三个模板不会进一步处理子类,因为它们中没有xsl:apply-templates)。

这种方法比命令式方法更容易扩展,因为它自动处理多个类别,并且可以通过添加另一个匹配模板来扩展其他元素或异常。没有if分支的编程。

注意:在XML中不存在空值。存在xsi:nil,但很少使用,特别是在没有某种模式的无类型场景中。

其他回答

如何用XSL检查一个值是空还是空? 例如,如果categoryName为空?

这可能是最简单的XPath表达式(accept answer中的表达式提供了相反的测试,如果否定,则会更长):

not(string(categoryName))

解释:

上面的not()函数的参数是false(),恰好当上下文项没有categoryName子项(“null”),或者(单个这样的)categoryName子项有字符串值——空字符串时。

我使用的是“当选择”结构。 例如: < xsl: choose > <xsl:when test="categoryName !=null"> <xsl:value-of select="categoryName " /> . < / xsl: when > < xsl:否则> <xsl:value-of select="other" /> . < / xsl:否则> < / xsl: choose >

在XSLT 2.0的使用中:

<xsl:copy-of select="concat(categoryName,  $vOther[not(string(current()/categoryName))])"/>

下面是一个完整的例子:

<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
     <xsl:output omit-xml-declaration="yes" indent="yes"/>

 <xsl:variable name="vOther" select="'Other'"/>

 <xsl:template match="/">
  <xsl:copy-of select="concat(categoryName,$vOther[not(string(current()/categoryName))])"/>
 </xsl:template>
</xsl:stylesheet>

当此转换应用于以下XML文档时:

<categoryName>X</categoryName>

生成所需的正确结果:

X

当应用于这个XML文档时:

<categoryName></categoryName>

或者关于这个:

<categoryName/>

或者在这个上面

<somethingElse>Y</somethingElse>

正确的结果是:

Other

类似地,使用以下XSLT 1.0转换:

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
 <xsl:output omit-xml-declaration="yes" indent="yes"/>

 <xsl:variable name="vOther" select="'Other'"/>

  <xsl:template match="/">
    <xsl:copy-of select=
    "concat(categoryName,  substring($vOther, 1 div not(string(categoryName))))"/>
  </xsl:template>
</xsl:stylesheet>

注意:完全没有使用条件句。在这个漂亮的Pluralsight课程中了解更多关于避免条件结构的重要性:

.NET中的战术设计模式:控制流

前两个处理空值,后两个处理空字符串。

<xsl:if test="USER/FIRSTNAME">
    USERNAME is not null
</xsl:if>
<xsl:if test="not(USER/FIRSTNAME)">
    USERNAME is null
 </xsl:if>
 <xsl:if test="USER/FIRSTNAME=''">
     USERNAME is empty string
 </xsl:if>
 <xsl:if test="USER/FIRSTNAME!=''">
     USERNAME is not empty string
 </xsl:if>

我知道这个问题已经很老了,但是在所有的答案中,我忽略了XSLT开发中这个用例的常用方法。

我想象OP中缺失的代码看起来像这样:

<xsl:template match="category">
    <xsl:choose>
        <xsl:when test="categoryName !=null">
            <xsl:value-of select="categoryName " />
        </xsl:when>
        <xsl:otherwise>
            <xsl:value-of select="other" />
        </xsl:otherwise>
    </xsl:choose>
</category>

输入是这样的:

<categories>
    <category>
       <categoryName>Books</categoryName>
    </category>
    <category>
       <categoryName>Magazines</categoryName>
       <categoryName>Periodicals</categoryName>
       <categoryName>Journals</categoryName>
    </category>
    <category>
        <categoryName><!-- please fill in category --></categoryName>
    </category>
    <category>
        <categoryName />
    </category>
    <category />
</categories>

I.e., I assume there can be zero, empty, single or multiple categoryName elements. To deal with all these cases using xsl:choose-style constructs, or in other words, imperatively, is quickly getting messy (even more so if elements can be at different levels!). A typical programming idiom in XSLT is using templates (hence the T in XSLT), which is declarative programming, not imperative (you don't tell the processor what to do, you just tell what you want output if certain conditions are met). For this use-case, that can look something like the following:

<!-- positive test, any category with a valid categoryName -->
<xsl:template match="category[categoryName[text()]]">
    <xsl:apply-templates />
</xsl:template>

<!-- any other category (without categoryName, "null", with comments etc) -->
<xsl:template match="category">
    <xsl:text>Category: Other</xsl:text>
</xsl:template>

<!-- matching the categoryName itself for easy handling of multiple names -->
<xsl:template match="categoryName">
    <xsl:text>Category: </xsl:text>
    <xsl:value-of select="." />
</xsl:template>

这是可行的(对于任何XSLT版本),因为上面的第一个具有更高的优先级(它有一个谓词)。“fall-through”匹配模板(第二个)捕获任何无效的内容。第三个函数负责以适当的方式输出categoryName值。

注意,在此场景中,不需要特别匹配类别或类别,因为处理器将自动处理所有子类,除非我们另有说明(在本例中,第二个和第三个模板不会进一步处理子类,因为它们中没有xsl:apply-templates)。

这种方法比命令式方法更容易扩展,因为它自动处理多个类别,并且可以通过添加另一个匹配模板来扩展其他元素或异常。没有if分支的编程。

注意:在XML中不存在空值。存在xsi:nil,但很少使用,特别是在没有某种模式的无类型场景中。

test="categoryName != ''"

编辑:在我看来,这涵盖了从问题中推断出的对“[非]空或空”最可能的解释,包括它是伪代码和我自己早期使用XSLT的经验。例如,“下面的Java的等价物是什么?”:

// Equivalent Java, NOT XSLT
!(categoryName == null || categoryName.equals(""))

有关更多细节,例如,明确识别null和empty,请参阅下面johnvey的回答和/或我从该回答改编的XSLT“小提琴”,其中包括Michael Kay评论中的选项以及第六种可能的解释。

在某些情况下,您可能想知道值什么时候具体为null,这在使用从. net对象序列化的XML时尤其必要。虽然接受的答案适用于此,但当字符串为空白或空时,它也返回相同的结果。,所以你无法区分。

<group>
    <item>
        <id>item 1</id>
        <CategoryName xsi:nil="true" />
    </item>
</group>

因此,您可以简单地测试属性。

<xsl:if test="CategoryName/@xsi:nil='true'">
   Hello World.
</xsl:if>

有时需要知道确切的状态,而不能简单地检查CategoryName是否已实例化,因为与Javascript不同

<xsl:if test="CategoryName">
   Hello World.
</xsl:if>

对于空元素将返回true。