使用带有 JSTL 的 EL 访问枚举值

Posted

技术标签:

【中文标题】使用带有 JSTL 的 EL 访问枚举值【英文标题】:Access Enum value using EL with JSTL 【发布时间】: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">

我不确定这是否可行。

【问题讨论】:

【参考方案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 个选择,没有一个是完美的:

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

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

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

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

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

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

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

    &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 似乎是同样糟糕的做法,当您想要重构枚举时编译器无法检查这些字符串。似乎双方都没有好的解决方案。

以上是关于使用带有 JSTL 的 EL 访问枚举值的主要内容,如果未能解决你的问题,请参考以下文章

在 JavaScript 中访问 Java / Servlet / JSP / JSTL / EL 变量

在 JavaScript 中访问 Java / Servlet / JSP / JSTL / EL 变量

EL和JSTL

在 JSP/JSTL/EL 中访问集合的大小 [重复]

EL表达式和JSTL标签库

9.EL表达式 和 JSTL核心标签库