解析Tomcat 6、7在EL表達式解析時存在的一個Bug
今天在做數(shù)據(jù)分頁顯示的時候遇到了一個問題,經(jīng)過測試,證實是Tomcat 6的一個bug,我所用的版本為:apache-tomcat-6.0.36,和7.0.30均能復現(xiàn)。下面詳細描述一下這個bug:
該bug是在JSTL<c:forEach>標簽中發(fā)現(xiàn)的,后來分析是EL表達式實現(xiàn)時產(chǎn)生的問題。jsp頁面中有一個list需要遍歷,這個list的類型為ArrayList<String>,我在其中放置的數(shù)據(jù)為(為方便我寫成數(shù)組的形式):["1","...","4","5","6","7","8","...","10"],這是一個很常見的帶頁碼縮略的分頁導航。在展示這些數(shù)據(jù)的時候我使用了下面的代碼:
<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就是分頁組件,其中預設(shè)了pageDot為"...",pageNo為當前的頁碼(假設(shè)為6),其他情況直接顯示分頁游標。在循環(huán)遍歷中只不過使用了最基本的條件判斷語句,由于pageList在定義中已經(jīng)明確指出是List<String>,按邏輯應該eq是按照字符串判斷的,但是居然出異常了:
javax.el.ELException: Cannot convert ... of type class java.lang.String to class java.lang.Long
為什么會出現(xiàn)“類型轉(zhuǎn)換錯誤”呢?通過分析代碼走向,當進入循環(huán)后,list中的第一條數(shù)據(jù)是“1”,而pageHelper.pageNo為long型,此時tomcat的EL表達式解析器會把looper類型轉(zhuǎn)換為Long型而不是把pageHelper.pageNo類型轉(zhuǎn)換為String進行比較,當遍歷到下一元素時,looper="...",這時looper的類型已經(jīng)確定,比較的時候tomcat還要試圖將looper轉(zhuǎn)換為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方法進行去除左右非可見字符。相當于強制轉(zhuǎn)換為String類型,此時tomcat又可以正常解析代碼,并未報錯。
同樣的一套代碼,我將其部署到resin中發(fā)現(xiàn)無論是修改前還是修改后都能正常運行,可見,應該是tomcat的bug。
示例代碼:點擊下載
讓tomcat報錯的演示地址:/bug/show.do
避免此bug的方法演示地址:/bug/avoid.do
以上地址前可能需要加上項目名稱(具體取決于你如何部署該項目)
- Tomcat7.0安裝配置詳細(圖文)
- window7下Tomcat7.0安裝配置方法
- IIS6.0+Tomcat7.0整合總結(jié)(推薦)
- Tomcat7中開啟gzip壓縮功能的配置方法
- tomcat6.0 /7.0安裝版內(nèi)存溢出設(shè)置方法
- eclipse3.2.2 + MyEclipse5.5 + Tomcat5.5.27 配置數(shù)據(jù)庫連接池
- Win7系統(tǒng)下tomcat7.0配置教程
- windows下tomcat7.0安裝圖文教程
- CentOS6.5下Tomcat7 Nginx Redis配置步驟教程詳解
- Tomcat 7通過設(shè)置不同的端口部署兩個項目
相關(guān)文章
基于SpringBoot和Vue3的博客平臺文章列表與分頁功能實現(xiàn)
在前面的教程中,我們已經(jīng)實現(xiàn)了基于Spring Boot和Vue3的發(fā)布、編輯、刪除文章功能。本教程將繼續(xù)引導您實現(xiàn)博客平臺的文章列表與分頁功能,需要的朋友可以參考閱讀2023-04-04詳解SpringBoot?統(tǒng)一后端返回格式的方法
今天我們來聊一聊在基于SpringBoot前后端分離開發(fā)模式下,如何友好的返回統(tǒng)一的標準格式以及如何優(yōu)雅的處理全局異常,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧2022-05-05一文詳解Java中的可變對象(Mutable)與不可變對象(Immutable)
如何在 Java 中創(chuàng)建不可變對象?我以前以為所有對象都是不可變的,因為如果你改變一個 String 實例的內(nèi)容,它總是會創(chuàng)建一個新的 String 對象并指向該對象,在本文中,我不僅將分享在 Java 中Immutable的步驟,還將討論可變對象與不可變對象及其優(yōu)缺點2023-11-11jasypt對配置文件的數(shù)據(jù)加密與解密方式
這篇文章主要介紹了jasypt對配置文件的數(shù)據(jù)加密與解密方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01springboot基于Mybatis mysql實現(xiàn)讀寫分離
這篇文章主要介紹了springboot基于Mybatis mysql實現(xiàn)讀寫分離,需要的朋友可以參考下2019-06-06EasyExcel工具讀取Excel空數(shù)據(jù)行問題的解決辦法
EasyExcel是阿里巴巴開源的一個excel處理框架,以使用簡單,節(jié)省內(nèi)存著稱,下面這篇文章主要給大家介紹了關(guān)于EasyExcel工具讀取Excel空數(shù)據(jù)行問題的解決辦法,需要的朋友可以參考下2022-08-08SpringBoot整合RabbitMQ及生產(chǎn)全場景高級特性實戰(zhàn)
本文主要介紹了SpringBoot整合RabbitMQ及生產(chǎn)全場景高級特性實戰(zhàn),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-10-10Java新特性之Nashorn_動力節(jié)點Java學院整理
這篇文章主要介紹了Java新特性之Nashorn的相關(guān)資料,需要的朋友可以參考下2017-06-06