Tomcat 6、7在EL表達式解析時存在的一個Bug

本文出處:http://blog.csdn.net/chaijunkun/article/details/9097229,轉載請註明。由於本人不定期會整理相關博文,會對相應內容作出完善。因此強烈建議在原始出處查看此文。

今天在做數據分頁顯示的時候遇到了一個問題,經過測試,證實是Tomcat 6的一個bug,我所用的版本爲:apache-tomcat-6.0.36,和7.0.30均能復現。下面詳細描述一下這個bug:

該bug是在JSTL<c:forEach>標籤中發現的,後來分析是EL表達式實現時產生的問題。jsp頁面中有一個list需要遍歷,這個list的類型爲ArrayList<String>,我在其中放置的數據爲(爲方便我寫成數組的形式):["1","...","4","5","6","7","8","...","10"],這是一個很常見的帶頁碼縮略的分頁導航。在展示這些數據的時候我使用了下面的代碼:

<c:forEach var="looper" items="${pageHelper.pageList}">
	<c:choose>
		<c:when test="${looper eq pageHelper.pageDot}">
		<p>分頁遊標的 點點點</p>
		</c:when>
		<c:when test="${looper eq pageHelper.pageNo}">
		<p>當前頁爲第${looper}頁面</p>
		</c:when>
		<c:otherwise>
		<p>分頁遊標:${looper}</p>
		</c:otherwise>
	</c:choose>
</c:forEach>

這裏pageHelper就是分頁組件,其中預設了pageDot爲"...",pageNo爲當前的頁碼(假設爲6),其他情況直接顯示分頁遊標。在循環遍歷中只不過使用了最基本的條件判斷語句,由於pageList在定義中已經明確指出是List<String>,按邏輯應該eq是按照字符串判斷的,但是居然出異常了:

javax.el.ELException: Cannot convert ... of type class java.lang.String to class java.lang.Long
爲什麼會出現“類型轉換錯誤”呢?通過分析代碼走向,當進入循環後,list中的第一條數據是“1”,而pageHelper.pageNo爲long型,此時tomcat的EL表達式解析器會把looper類型轉換爲Long型而不是把pageHelper.pageNo類型轉換爲String進行比較,當遍歷到下一元素時,looper="...",這時looper的類型已經確定,比較的時候tomcat還要試圖將looper轉換爲Long類型,於是就出錯了。

爲此我專門寫了一個實例代碼:

<c:forEach var="looper" items="${pageHelper.pageList}">
	<c:choose>
		<c:when test="${looper eq fn:trim(pageHelper.pageDot)}">
		<p>分頁遊標的 點點點</p>
		</c:when>
		<c:when test="${looper eq fn:trim(pageHelper.pageNo)}">
		<p>當前頁爲第${looper}頁面</p>
		</c:when>
		<c:otherwise>
		<p>分頁遊標:${looper}</p>
		</c:otherwise>
	</c:choose>
</c:forEach>

很簡單,每次比較的時候都把後者用fn:trim方法進行去除左右非可見字符。相當於強制轉換爲String類型,此時tomcat又可以正常解析代碼,並未報錯。

同樣的一套代碼,我將其部署到resin中發現無論是修改前還是修改後都能正常運行,可見,應該是tomcat的bug。

示例代碼下載地址:http://download.csdn.net/detail/chaijunkun/5581883

讓tomcat報錯的演示地址:/bug/show.do

避免此bug的方法演示地址:/bug/avoid.do

以上地址前可能需要加上項目名稱(具體取決於你如何部署該項目)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章