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]}「One」を返します。それは何ですか?なぜ<c:out>数字を長いものとして扱うのですか。私には直感に反しているようです(intはlongよりも一般的に使用されているため)。

だから私の新しい質問は、Integer値で地図にアクセスするためのEL表記はありますか?

回答:


117

最初の回答(EL 2.1、2009年5月)

このJavaフォーラムのスレッドで述べたように:

基本的に、オートボクシングは整数オブジェクトをマップに配置します。すなわち:

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

EL(Expressions Languages)は、0をLongとして評価するため、マップ内のキーとしてLongを探します。つまり、以下を評価します。

map.get(new Long(0))

aLongIntegerオブジェクトと等しくなることはないため、マップ内のエントリは見つかりません。
一言で言えばそれだけです。


2009年5月以降の更新(EL 2.2)

2009年12月には、JSP 2.2 / Java EE6でEL2.2が導入されましたが、EL2.1とはいくつかの違いがあります。
( " EL Expression parsing integer as long ")のようです:

EL2.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に数値を整数として展開させる方法はありませんか?
Steve Kuo

1
@スティーブ:確かに、ELはそれをサポートしていないようです。
vonC 2009年

この質問と回答をGoogle検索で見つけました。案の定、Map <Integer、String>からMap <Long、String>に切り替えるとすぐに、JSPページにあるELを使用してそこからプルすることができました。ありがとう!
John Munsch 2011

@スティーブ:それは可能です-私の答えを参照してください
トビアスリーフケ2015年

@SteveKuoそれは確かに可能であるはずです。EL 2.2がまだリリースされていないときに、この6年前の回答が書かれたことに気づきました。私はその答えを編集して更新しました。
vonC 2015年

11

上記のコメントに加えて、もう1つの役立つヒントは、リクエストパラメータなどの変数に文字列値が含まれている場合です。この場合、これを渡すと、JSTLがたとえば「1」の値をスティングとしてキーイングするため、マップハッシュマップで一致が見つかりません。

これを回避する1つの方法は、このようなことを行うことです。

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

これはLongオブジェクトとして扱われるようになり、マップMapなどに含まれているオブジェクトと一致する可能性があります。

次に、通常どおり次のようなものを続けます

${map[longKey]}

10

数字を「(」「)」に入れると、Longのすべての関数を使用できます。そうすれば、longをintにキャストできます。

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

私はあなたの答えをすぐには見ませんでした。+1。わかりやすくするために、その可能性を回答に含めて文書化しました。
vonC 2015年

3

上記の投稿に基づいて、私はこれを試しましたが、これはうまくいきました。マップBの値をマップ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

あなただけ持って起こる場合MapInteger、変更することはできませんキーを、あなたが書くことができ、カスタムEL機能を変換するためLongInteger。これにより、次のようなことができます。

<c:out value="${map[myLib:longToInteger(1)]}"/>
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.