ChatGPT解决这个技术问题 Extra ChatGPT

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

我有一个名为 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">

我不确定这是否可能。


A
Aritz

与字符串的简单比较有效:

<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,以防万一),这似乎不起作用。它似乎仅在与字符串值进行比较时才有效,这里将是小写的“旧”
A
Aritz

如果使用 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):找不到类型'my.package.model.EngagementRequest.EngagementStatus'
尝试使用'my.package.model.EngagementRequest$EngagementStatus'
这个解决方案的一个好处是,如果您的表达式中有错误,您会收到一条错误消息,而 <c:if><c:when> 并不总是发生这种情况(它们会悄悄地失败)。
A
Aritz

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

您可以在 test 属性中使用 scriptlet: 这使用枚举,但它也使用 scriptlet,它不是JSP 2.0 中的“正确方式”。但最重要的是,当您想在使用 ${} 时添加另一个条件时,这不起作用。这意味着您要测试的所有变量都必须在 scriptlet 中声明,或者保留在请求或会话中(pageContext 变量在 .tag 文件中不可用)。您可以与字符串进行比较: 这看起来很干净,但是您正在引入一个重复枚举值并且无法由编译器验证的字符串。因此,如果您从枚举中删除该值或重命名它,您将不会看到这部分代码不再可访问。您基本上每次都必须通过代码进行搜索/替换。您可以将您使用的每个枚举值添加到页面上下文中: 然后您可以这样做:

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


关于选项 2,编译器无法验证它,但在运行时字符串将使用 Enum.valueOf(Class<T> enumType, String name) 转换为枚举,如果枚举没有具有该名称的常量,则会触发 ELException。表达式不会总是错误的。
A
Aritz

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

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

然后我能够做到:

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

按预期工作。


使用小脚本是不好的风格。
C
Community

这里还有两种可能性:

JSP EL 3.0 常量

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

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

但是,一些 IDE 还不理解这一点(例如 IntelliJ),因此如果您输入错误,直到运行时才会收到任何警告。

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

辅助方法

您可以将吸气剂添加到您的枚举中。

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 显示建议。
[更新] 似乎 IntelliJ IDEA 已经解决了这个答案中提到的问题。
我很感激这是几年后的事,但我想我会添加评论,因为它可能会帮助其他人。如果 Java 枚举存储在另一个类中,EL 3.0 不会获取它。例如,这可行:public enum TestEnum { AWAITING, FILE_PRESENT, FILE_NOT_FOUND; } 但这不可行:public class TestClass { public enum TestEnum { AWAITING, FILE_PRESENT, FILE_NOT_FOUND; } } 这非常令人沮丧,因为我从未将整个类文件专用于枚举!
A
Aritz

为此,我执行以下操作:

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

<c:if test="${someVariable == abc}">
    ....
</c:if>

它看起来很丑,但有效!


e
eremmel

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

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

D
Dean

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

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>

E
ElectronicBlacksmith

使用 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}">

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


M
Mehdi
<%@ 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


E
Eclatante

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

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

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

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

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}">
...

P
Pavan

在 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 和 enum obj 被创建了。现在 EnumTest 您将在 servlet 或控制器类 session.setAttribute("enumTest", EnumTest ); 中设置会话对象。

在 JSP 页面中

<c:if test="${enumTest.statusobj == 'ACTIVE'}">

//TRUE??? THEN PROCESS SOME LOGIC