EL отримує доступ до значення карти за допомогою цілочисельного ключа


85

У мене є карта, яку вводить Integer. За допомогою 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. Я зрозумів це, коли я змінив HashMapдо TreeMapі отримав повідомлення про помилку:

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значенням?

Відповіді:


117

Початкова відповідь (EL 2.1, травень 2009 р.)

Як згадувалося в цій темі форуму Java :

В основному автобокс ставить цілий об'єкт на карту. тобто:

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

EL (мови виразів) оцінює 0 як Long і, таким чином, шукає Long як ключ на карті. тобто він оцінює:

map.get(new Long(0))

Оскільки a Longніколи не дорівнює Integerоб’єкту, він не знаходить запис на карті.
Ось у двох словах.


Оновлення з травня 2009 р. (EL 2.2)

У грудні 2009 року було представлено EL 2.2 з JSP 2.2 / Java EE 6 , з деякими відмінностями порівняно з EL 2.1 .
Здається (" Вираз EL розбирає ціле число як довго "), що:

ви можете викликати метод intValueна Longоб'єкті self усередині EL 2.2 :

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

Це може бути хорошим рішенням тут (також згадується нижче в Тобіас Liefke «s відповідь )


Оригінальна відповідь:

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 розширити число як ціле число?
Стів Куо,

1
@ Steve: справді, EL, схоже, не підтримує цього.
VonC

Я знайшов це запитання та відповідь із пошуку Google. Звичайно, як тільки я перейшов з Map <Integer, String> на Map <Long, String>, я зміг витягти з нього, використовуючи EL, який я мав на своїй сторінці JSP. Дякую!!
Джон Мунш,

@ Стів: Це можливо - див. Мою відповідь
Тобіас Лієфке

@SteveKuo Це повинно бути справді можливим. Тепер я усвідомлюю, що ця 6-річна відповідь була написана, коли EL 2.2 ще не вийшов. Я відредагував та оновив зазначену відповідь.
VonC 02.03.15

11

Ще однією корисною підказкою на додаток до вищезазначеного коментаря буде, коли у вас є значення рядка, яке міститься у якійсь змінній, такій як параметр запиту. У цьому випадку передача цього також призведе до того, що JSTL введе значення "say" 1 як жало і як таке не знайде збігу в хеш-карті карти.

Один із способів обійти це - зробити щось подібне.

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

Тепер це буде розглядатися як довгий об’єкт, а потім матиме шанс зрівняти об’єкт, коли він міститься на карті карти чи будь-якому іншому.

Потім продовжуйте, як зазвичай, щось подібне

${map[longKey]}

10

Ви можете використовувати всі функції Long, якщо ви введете число в "(" ")". Таким чином, ви можете передати long в int:

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

Я не відразу побачив вашу відповідь. +1. Я включив та задокументував цю можливість у своїй відповіді для більшої наочності.
VonC 02.03.15

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

Якщо у вас просто трапляється a Mapз Integerключами, які ви не можете змінити, ви можете написати власну функцію EL для перетворення a Longв Integer. Це дозволить вам зробити щось на зразок:

<c:out value="${map[myLib:longToInteger(1)]}"/>
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.