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