使用EL和JSTL访问枚举值

IaC*_*der 103 java jsp jstl java-ee

我有一个名为Status的枚举定义如下:

public enum Status { 

    VALID("valid"), OLD("old");

    private final String val;

    Status(String val) {
        this.val = val;
    }

    public String getStatus() {
        return val;
    }

}
Run Code Online (Sandbox Code Playgroud)

我想VALID从JSTL标签访问值.特别test<c:when>标签的属性.例如

<c:when test="${dp.status eq Status.VALID">
Run Code Online (Sandbox Code Playgroud)

我不确定这是否可行.

Ale*_*jev 111

对字符串的简单比较:

<c:when test="${someModel.status == 'OLD'}">
Run Code Online (Sandbox Code Playgroud)

  • 对于那些需要源的人:这是在(表达式语言规范,版本2.2)的1.17节中指定的(例如),它是[JSR-245]的一部分(http://www.jcp.org/en/jsr) /细节?ID = 245). (11认同)
  • 但是你放弃了枚举的优势:如果枚举有一天改变,这可能会导致麻烦的误解.通常情况下,如果我发现自己更改枚举,我觉得很安全,可能我不记得那个视图中的字符串到枚举引用... (10认同)
  • JavaServer Pages™规范2.0版在JSP.2.3.5.7中说:"•如果A或B是字符串,则将A和B强制转换为字符串,从词汇上进行比较" (4认同)

Jam*_*mes 54

如果使用Spring MVC,Spring Expression Language(SpEL)可能会有所帮助:

<spring:eval expression="dp.status == T(com.example.Status).VALID" var="isValid" />
<c:if test="${isValid}">
   isValid
</c:if>
Run Code Online (Sandbox Code Playgroud)

  • 尝试使用'my.package.model.EngagementRequest $ EngagementStatus' (4认同)

Mat*_*att 40

这里有3个选择,其中没有一个是完美的:

  1. 您可以在test属性中使用scriptlet :

    <c:when test="<%= dp.getStatus() == Status.VALID %>">

    这使用了枚举,但它也使用了一个scriptlet,它不是JSP 2.0中的"正确方法".但最重要的是,当您想要when使用相同的条件添加另一个条件时,这不起作用${}.这意味着您要测试的所有变量都必须在scriptlet中声明,或者保存在请求或会话中(pageContext变量在.tag文件中不可用).

  2. 你可以比较字符串:

    <c:when test="${dp.status == 'VALID'}">

    这看起来很干净,但是您引入了一个复制枚举值的字符串,并且无法由编译器验证.因此,如果您从枚举中删除该值或重命名该值,您将不会再看到这部分代码不可访问.您基本上每次都必须通过代码进行搜索/替换.

  3. 您可以将您使用的每个枚举值添加到页面上下文中:

    <c:set var="VALID" value="<%=Status.VALID%>"/>

    然后你可以这样做:

    <c:when test="${dp.status == VALID}">

我更喜欢最后一个选项(3),即使它也使用了一个scriptlet.这是因为它仅在您设置值时使用它.稍后,您可以在更复杂的EL表达式中使用它,以及其他EL条件.在选项(1)中,您不能在test单个when标记的属性中使用scriptlet和EL表达式.


IaC*_*der 22

因此,要完全解决我的问题,我需要执行以下操作:

<% pageContext.setAttribute("old", Status.OLD); %>
Run Code Online (Sandbox Code Playgroud)

然后我就能做到:

<c:when test="${someModel.status == old}"/>...</c:when>
Run Code Online (Sandbox Code Playgroud)

这按预期工作.

  • 使用scriptlet是不好的风格. (11认同)

Rup*_*ott 12

这里有两种可能性:

JSP EL 3.0常量

只要您使用至少3.0版本的EL,就可以将常量导入页面,如下所示:

<%@ page import="org.example.Status" %>
<c:when test="${dp.status eq Status.VALID}">
Run Code Online (Sandbox Code Playgroud)

但是,有些IDE还没有理解这一点(例如IntelliJ),所以如果你输入错误,你就不会收到任何警告,直到运行时.

一旦获得适当的IDE支持,这将是我的首选方法.

助手方法

你可以在你的枚举中添加getter.

public enum Status { 
  VALID("valid"), OLD("old");

  private final String val;

  Status(String val) {
    this.val = val;
  }

  public String getStatus() {
    return val;
  }

  public boolean isValid() {
    return this == VALID;
  }

  public boolean isOld() {
    return this == OLD;
  }
}
Run Code Online (Sandbox Code Playgroud)

然后在你的JSP中:

<c:when test="${dp.status.valid}">
Run Code Online (Sandbox Code Playgroud)

这在所有IDE中都受支持,如果您还不能使用EL 3.0,也可以使用它.这就是我现在所做的,因为它将所有逻辑都包含在我的枚举中.

如果存储枚举的变量可能为null,也要小心.如果您的代码不保证它不为null,您需要先检查它:

<c:when test="${not empty db.status and dp.status.valid}">
Run Code Online (Sandbox Code Playgroud)

我认为这种方法优于那些在JSP中设置中间值的方法,因为您必须在需要使用枚举的每个页面上执行此操作.但是,使用此解决方案,您只需要声明一次getter.

  • "JSP EL 3.0常量"部分必须是可接受的答案,因为它是使用内置功能实现所需结果的标准方法.在旁注中,InteliJ IDEA(至少是Ultimate版本2017.2.4)确实支持它开箱即用,并在键入`$ {MyEnum.}`时显示可用常量列表,在点后面放入插入符并按`Ctrl +空间`显示建议. (2认同)

小智 10

为此,我执行以下操作:

<c:set var="abc">
    <%=Status.OLD.getStatus()%>
</c:set>

<c:if test="${someVariable == abc}">
    ....
</c:if>
Run Code Online (Sandbox Code Playgroud)

它看起来很难看,但很有效!