EL通过整数键访问映射值


85

我有一个以整数为键的地图。使用EL,如何通过其键访问值?

Map<Integer, String> map = new HashMap<Integer, String>();
map.put(1, "One");
map.put(2, "Two");
map.put(3, "Three");

我以为这可以工作,但不能(在请求的属性中已经有地图):

<c:out value="${map[1]}"/>

跟进:我找到了问题所在。显然${name[1]}是使用数字作为进行地图查找Long。当我更改HashMapTreeMap并收到错误时,我发现了这一点:

java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.Long

如果我将地图更改为:

Map<Long, String> map = new HashMap<Long, String>();
map.put(1L, "One");

然后${name[1]}返回“一个”。那是什么 为什么<c:out>将数字视为长号。对我来说似乎违反直觉(因为int比long更常用)。

所以我的新问题是,是否有一个EL表示法可通过Integer值访问地图?

Answers:


117

初步答案(EL 2.1,2009年5月)

本Java论坛主题所述

基本上,自动装箱会将Integer对象放入Map中。即:

map.put(new Integer(0), "myValue")

EL(表达式语言)将0评估为Long,因此在地图中寻找Long作为键。即它评估:

map.get(new Long(0))

由于aLong永远不等于Integer对象,因此不会在地图中找到该条目。
简而言之就是这样。


自2009年5月以来的更新(EL 2.2)

2009年12月,EL 2.2与JSP 2.2 / Java EE 6一起推出,与EL 2.1相比一些区别
似乎(“ EL表达式将整数解析为长”)是:

您可以在EL 2.2内部intValueLong对象self上调用该方法

<c:out value="${map[(1).intValue()]}"/>

这可能是一个很好的解决方法(下面在Tobias Liefke答案中也提到了)


原始答案:

EL使用以下包装器:

Terms                  Description               Type
null                   null value.               -
123                    int value.                java.lang.Long
123.00                 real value.               java.lang.Double
"string" ou 'string'   string.                   java.lang.String
true or false          boolean.                  java.lang.Boolean

JSP页面演示了这一点:

 <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>

 <%@ page import="java.util.*" %>

 <h2> Server Info</h2>
Server info = <%= application.getServerInfo() %> <br>
Servlet engine version = <%=  application.getMajorVersion() %>.<%= application.getMinorVersion() %><br>
Java version = <%= System.getProperty("java.vm.version") %><br>
<%
  Map map = new LinkedHashMap();
  map.put("2", "String(2)");
  map.put(new Integer(2), "Integer(2)");
  map.put(new Long(2), "Long(2)");
  map.put(42, "AutoBoxedNumber");

  pageContext.setAttribute("myMap", map);  
  Integer lifeInteger = new Integer(42);
  Long lifeLong = new Long(42);  
%>
  <h3>Looking up map in JSTL - integer vs long </h3>

  This page demonstrates how JSTL maps interact with different types used for keys in a map.
  Specifically the issue relates to autoboxing by java using map.put(1, "MyValue") and attempting to display it as ${myMap[1]}
  The map "myMap" consists of four entries with different keys: A String, an Integer, a Long and an entry put there by AutoBoxing Java 5 feature.       

  <table border="1">
    <tr><th>Key</th><th>value</th><th>Key Class</th></tr>
    <c:forEach var="entry" items="${myMap}" varStatus="status">
    <tr>      
      <td>${entry.key}</td>
      <td>${entry.value}</td>
      <td>${entry.key.class}</td>
    </tr>
    </c:forEach>
</table>

    <h4> Accessing the map</h4>    
    Evaluating: ${"${myMap['2']}"} = <c:out value="${myMap['2']}"/><br>
    Evaluating: ${"${myMap[2]}"}   = <c:out value="${myMap[2]}"/><br>    
    Evaluating: ${"${myMap[42]}"}   = <c:out value="${myMap[42]}"/><br>    

    <p>
    As you can see, the EL Expression for the literal number retrieves the value against the java.lang.Long entry in the map.
    Attempting to access the entry created by autoboxing fails because a Long is never equal to an Integer
    <p>

    lifeInteger = <%= lifeInteger %><br/>
    lifeLong = <%= lifeLong %><br/>
    lifeInteger.equals(lifeLong) : <%= lifeInteger.equals(lifeLong) %> <br>

因此,没有办法让EL将数字扩展为整数吗?
郭富城2009年

1
@Steve:的确,EL似乎不支持这一点。
VonC

我从Google搜索中找到了这个问题和答案。当然,当我从Map <Integer,String>切换到Map <Long,String>时,我就可以使用我在JSP页面中使用的EL从它中拉出。谢谢!!
约翰·蒙施

@Steve:有可能-见我的答案
Tobias Liefke

@SteveKuo应该确实有可能。我现在意识到,这6年的答案是在EL 2.2尚未发布时编写的。我已经编辑并更新了上述答案。
VonC'3

11

除了上面的注释之外,另一个有用的提示是,当您在某个变量(例如请求参数)中包含字符串值时。在这种情况下,传递该值也将导致JSTL将“ 1”的值键入为字符串,因此在Map哈希图中找不到匹配项。

解决这个问题的一种方法是做这样的事情。

<c:set var="longKey" value="${param.selectedIndex + 0}"/>

现在将其视为Long对象,然后将其包含在地图Map或其他对象中时有机会匹配该对象。

然后,像往常一样继续

${map[longKey]}

10

如果将数字放在“(”“)”中,则可以使用Long的所有函数。这样,您可以将long转换为int:

<c:out value="${map[(1).intValue()]}"/>

我没有立即看到您的回答。+1。我已经在答案中包括并记录了这种可能性,以提高可视性。
VonC'3

3

根据上面的帖子,我尝试了这一点,并且效果很好,我想将Map B的值用作Map A的键:

<c:if test="${not empty activityCodeMap and not empty activityDescMap}">
<c:forEach var="valueMap" items="${auditMap}">
<tr>
<td class="activity_white"><c:out value="${activityCodeMap[valueMap.value.activityCode]}"/></td>
<td class="activity_white"><c:out value="${activityDescMap[valueMap.value.activityDescCode]}"/></td>
<td class="activity_white">${valueMap.value.dateTime}</td>
</tr>
</c:forEach>
</c:if>

3

如果您碰巧拥有一个不能更改的MapwithInteger键,则可以编写一个自定义EL函数将a转换LongInteger。这将使您可以执行以下操作:

<c:out value="${map[myLib:longToInteger(1)]}"/>
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.