【问题标题】:Access Enum value using EL with JSTL使用带有 JSTL 的 EL 访问枚举值
【发布时间】:2010-09-12 12:32:38
【问题描述】:

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

public enum Status { 

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

    private final String val;

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

    public String getStatus() {
        return val;
    }

}

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

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

我不确定这是否可行。

【问题讨论】:

    标签: java jsp jakarta-ee jstl


    【解决方案1】:
    <%@ page import="com.example.Status" %>
    
    1. ${dp.status eq Title.VALID.getStatus()}
    2. ${dp.status eq Title.VALID}
    3. ${dp.status eq Title.VALID.toString()}
    
    • 将导入放在顶部,在JSP 页眉中
    • 如果您想使用 getStatus 方法,请使用 #1
    • 如果您想使用 枚举元素 本身,请使用 #2 或 #3
    • 您可以使用 == 代替 eq

    【讨论】:

      【解决方案2】:

      当有很多点可以使用时,我会这样做......

      public enum Status { 
      
          VALID("valid"), OLD("old");
      
          private final String val;
      
          Status(String val) {
              this.val = val;
          }
      
          public String getStatus() {
              return val;
          }
      
          public static void setRequestAttributes(HttpServletRequest request) {
              Map<String,String> vals = new HashMap<String,String>();
              for (Status val : Status.values()) {
                  vals.put(val.name(), val.value);
              }
              request.setAttribute("Status", vals);
          }
      
      }
      

      JSP

      <%@ page import="...Status" %>
      <% Status.setRequestAttributes(request) %>
      
      <c:when test="${dp.status eq Status.VALID}">
      ...
      

      【讨论】:

        【解决方案3】:

        当使用 MVC 框架时,我将以下内容放入我的控制器中。

        request.setAttribute(RequestParameterNamesEnum.INBOX_ACTION.name(), RequestParameterNamesEnum.INBOX_ACTION.name());
        

        这允许我在我的 JSP 页面中使用以下内容。

        <script> var url = 'http://www.nowhere.com/?${INBOX_ACTION}=' + someValue;</script>
        

        它也可以用于你的比较

        <c:when test="${someModel.action == INBOX_ACTION}">
        

        我更喜欢输入字符串文字。

        【讨论】:

          【解决方案4】:

          这里还有两种可能性:

          JSP EL 3.0 常量

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

          <%@ page import="org.example.Status" %>
          <c:when test="${dp.status eq Status.VALID}">
          

          但是,一些 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;
            }
          }
          

          然后在你的 JSP 中:

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

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

          如果存储枚举的变量可能为空,也要小心。如果你的代码不能保证它不为空,你需要先检查一下:

          <c:when test="${not empty db.status and dp.status.valid}">
          

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

          【讨论】:

          • “JSP EL 3.0 常量”部分必须是公认的答案,因为它是使用内置功能实现所需结果的标准方法。附带说明一下,InteliJ IDEA(至少 Ultimate 版本 2017.2.4)确实支持开箱即用,并在您键入 ${MyEnum.} 时显示可用常量列表,将插入符号放在点后面并按 Ctrl+Space 以显示建议。
          • [UPDATE] 似乎 IntelliJ IDEA 已经解决了这个答案中提到的问题。
          【解决方案5】:

          在 Java 类中:

              public class EnumTest{
              //Other property link
              private String name;
              ....
          
                  public enum Status {
                          ACTIVE,NEWLINK, BROADCASTED, PENDING, CLICKED, VERIFIED, AWARDED, INACTIVE, EXPIRED, DELETED_BY_ADMIN;
                      }
          
                  private Status statusobj ;
          
              //Getter and Setters
          }
          

          所以现在 POJO 和枚举 obj 已创建。 现在 EnumTest 您将在 servlet 或控制器类中使用会话对象进行设置 session.setAttribute("enumTest", EnumTest );

          在 JSP 页面中

          <c:if test="${enumTest.statusobj == 'ACTIVE'}">
          
          //TRUE??? THEN PROCESS SOME LOGIC
          

          【讨论】:

            【解决方案6】:

            为此,我执行以下操作:

            <c:set var="abc">
                <%=Status.OLD.getStatus()%>
            </c:set>
            
            <c:if test="${someVariable == abc}">
                ....
            </c:if>
            

            它看起来很丑,但是很管用!

            【讨论】:

              【解决方案7】:

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

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

                &lt;c:when test="&lt;%= dp.getStatus() == Status.VALID %&gt;"&gt;

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

              2. 您可以与字符串进行比较:

                &lt;c:when test="${dp.status == 'VALID'}"&gt;

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

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

                &lt;c:set var="VALID" value="&lt;%=Status.VALID%&gt;"/&gt;

                然后你可以这样做:

                &lt;c:when test="${dp.status == VALID}"&gt;

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

              【讨论】:

              • 关于选项 2,编译器无法验证它,但在运行时字符串将使用Enum.valueOf(Class&lt;T&gt; enumType, String name) 转换为枚举,如果枚举没有具有该名称的常量,则将触发ELException .表达式不会总是错误的。
              【解决方案8】:

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

              <% pageContext.setAttribute("old", Status.OLD); %>
              

              然后我可以做到:

              <c:when test="${someModel.status == old}"/>...</c:when>
              

              按预期工作。

              【讨论】:

              • 使用小脚本是不好的风格。
              【解决方案9】:

              如果使用 Spring MVC,Spring 表达式语言 (SpEL) 会很有帮助:

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

              【讨论】:

              • 这似乎不适用于内部枚举?原因:org.springframework.expression.spel.SpelEvaluationException: EL1005E:(pos 0): Type cannot be found 'my.package.model.EngagementRequest.EngagementStatus'
              • 尝试使用'my.package.model.EngagementRequest$EngagementStatus'
              • 这个解决方案的一个好处是,如果你的表达式中有错误,你会收到一条错误消息,&lt;c:if&gt;&lt;c:when&gt; 并不总是发生这种情况(它们会悄悄地失败)。
              【解决方案10】:

              与字符串的简单比较:

              <c:when test="${someModel.status == 'OLD'}">
              

              【讨论】:

              • 对于那些需要来源的人:这在“表达式语言规范,版本 2.2”的第 1.17 节中指定(例如),这是JSR-245 的一部分。
              • JavaServer Pages™ 规范 2.0 版在 JSP.2.3.5.7 中说:“• 如果 A 或 B 是字符串,则将 A 和 B 都强制转换为字符串,按词法比较”
              • 但是你失去了拥有枚举的优势:如果有一天枚举改变,这可能会导致麻烦的误解。通常,如果我发现自己更改了一个枚举,我会觉得很安全,而且我可能不会记得那个视图中的字符串到枚举引用...
              • 这是与枚举的 toString 比较吗?因此,如果您覆盖 toString(例如,您想要一个友好的显示名称),那么您需要确保您还更改了匹配的值。
              • FWIW,今天在我的 Java 8(IBM Java for WebSphere,以防万一),这似乎不起作用。它似乎只在与字符串值比较时才有效,这里是小写的“旧”
              【解决方案11】:

              我对 Kornel 的问题没有答案,但我对其他脚本示例有意见。大多数表达式都隐式信任toString(),但Enum.valueOf() 需要一个来自/匹配Enum.name() 属性的值。所以应该使用例如:

              <% pageContext.setAttribute("Status_OLD", Status.OLD.name()); %>
              ...
              <c:when test="${someModel.status == Status_OLD}"/>...</c:when>
              

              【讨论】:

                【解决方案12】:

                向枚举添加一个方法,如:

                public String getString() {
                    return this.name();
                }
                

                例如

                public enum MyEnum {
                    VALUE_1,
                    VALUE_2;
                    public String getString() {
                        return this.name();
                    }
                }
                

                那么你可以使用:

                <c:if test="${myObject.myEnumProperty.string eq 'VALUE_2'}">...</c:if>
                

                【讨论】:

                  【解决方案13】:

                  我通常认为将 java 代码混合到 jsps/tag 文件中是不好的做法。使用 'eq' 应该可以解决问题:

                  <c:if test="${dp.Status eq 'OLD'}">
                    ...
                  </c:if>
                  

                  【讨论】:

                  • 所以使用 == 而不是 eq 是一种不好的做法?它们的作用完全相同,因此没有“技巧”。
                  • 当然,我并没有就 eq vs == 的使用发表声明。这个问题的许多答案涉及将 java 代码插入到 jsp 或标记文件中,这可能是一个拐杖。我倾向于将 Java 代码中的业务逻辑(可以轻松彻底地进行单元测试)与 JSP 中的显示逻辑分开。
                  • 对我来说,将魔术字符串插入 JSP 似乎是同样糟糕的做法,当您想要重构枚举时编译器无法检查这些字符串。似乎双方都没有好的解决方案。
                  猜你喜欢
                  • 2023-03-09
                  • 2011-01-15
                  • 2014-06-01
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 2017-06-19
                  • 1970-01-01
                  • 1970-01-01
                  相关资源
                  最近更新 更多