EL表達(dá)式
EL?全名為Expression Language
EL?語法很簡單,它最大的特點(diǎn)就是使用上很方便。接下來介紹EL主要的語法結(jié)構(gòu):
${sessionScope.user.sex}
所有EL都是以${為起始、以}為結(jié)尾的。上述EL范例的意思是:從Session的范圍中,取得
用戶的性別。假若依照之前JSP Scriptlet的寫法如下:
User user = (User)session.getAttribute("user");
String sex = user.getSex( );
兩者相比較之下,可以發(fā)現(xiàn)EL?的語法比傳統(tǒng)JSP Scriptlet?更為方便、簡潔。
?
.與?[ ]?運(yùn)算符
EL?提供?.?和?[ ]?兩種運(yùn)算符來導(dǎo)航數(shù)據(jù)。下列兩者所代表的意思是一樣的:
${sessionScope.user.sex}等于${sessionScope.user["sex"]}
.?和?[ ]?也可以同時(shí)混合使用,如下:
${sessionScope.shoppingCart[0].price}
回傳結(jié)果為shoppingCart中第一項(xiàng)物品的價(jià)格。
不過,以下兩種情況,兩者會有差異:
(1)?當(dāng)要存取的屬性名稱中包含一些特殊字符,如.?或 – 等并非字母或數(shù)字的符號,就一定要使用?[ ],例如:${user.My-Name }
上述是不正確的方式,應(yīng)當(dāng)改為:${user["My-Name"] }
(2)?我們來考慮下列情況:
${sessionScope.user[data]}
此時(shí),data?是一個(gè)變量,假若data的值為"sex"時(shí),那上述的例子等于${sessionScope.user.sex};
假若data?的值為"name"時(shí),它就等于${sessionScope.user.name}。因此,如果要動態(tài)取值時(shí),就可以用上述的方法來做,但.?無法做到動態(tài)取值。
?
EL?變量
EL?存取變量數(shù)據(jù)的方法很簡單,例如:${username}。它的意思是取出某一范圍中名稱為username的變量。因?yàn)槲覀儾]有指定哪一個(gè)范圍的username,所以它的默認(rèn)值會先從Page?范圍找,假如找不到,再依序到Request、Session、Application范圍。假如途中找到username,就直接回傳,不再繼續(xù)找下去,但是假如全部的范圍都沒有找到時(shí),就回傳null,當(dāng)然EL表達(dá)式還會做出優(yōu)化,頁面上顯示空白,而不是打印輸出NULL。
屬性范圍(jstl名稱)
EL中的名稱
Page
PageScope
Request
RequestScope
Session
SessionScope
Application
ApplicationScope
我們也可以指定要取出哪一個(gè)范圍的變量:
范例
說明
${pageScope.username}
取出Page范圍的username變量
${requestScope.username}
取出Request范圍的username變量
${sessionScope.username}
取出Session范圍的username變量
${applicationScope.username}
取出Application范圍的username變量
其中,pageScope、requestScope、sessionScope和applicationScope都是EL?的隱含對象,由它們的名稱可以很容易猜出它們所代表的意思,例如:${sessionScope.username}是取出Session范圍的username?變量。這種寫法是不是比之前JSP?的寫法:
String username = (String) session.getAttribute("username");容易、簡潔許多.
自動轉(zhuǎn)變類型
EL?除了提供方便存取變量的語法之外,它另外一個(gè)方便的功能就是:自動轉(zhuǎn)變類型,我們來看下面這個(gè)范例:
${param.count + 20}
假若窗體傳來count的值為10時(shí),那么上面的結(jié)果為30。之前沒接觸過JSP?的讀者可能會認(rèn)為上面的例子是理所當(dāng)然的,但是在JSP 1.2?之中不能這樣做,原因是從窗體所傳來的值,它們的類型一律是String,所以當(dāng)你接收之后,必須再將它轉(zhuǎn)為其他類型,如:int、float?等等,然后才能執(zhí)行一些數(shù)學(xué)運(yùn)算,下面是之前的做法:
String str_count = request.getParameter("count");
int count = Integer.parseInt(str_count);
count = count + 20;
所以,注意不要和java的語法(當(dāng)字符串和數(shù)字用“+”鏈接時(shí)會把數(shù)字轉(zhuǎn)換為字符串)搞混淆嘍。
EL?隱含對象
JSP有9個(gè)隱含對象,而EL也有自己的隱含對象。EL隱含對象總共有11?個(gè)
隱含對象
類型
說明
PageContext
javax.servlet.ServletContext
表示此JSP的PageContext
PageScope
java.util.Map
取得Page范圍的屬性名稱所對應(yīng)的值
RequestScope
java.util.Map
取得Request范圍的屬性名稱所對應(yīng)的值
sessionScope
java.util.Map
取得Session范圍的屬性名稱所對應(yīng)的值
applicationScope
java.util.Map
取得Application范圍的屬性名稱所對應(yīng)的值
param
java.util.Map
如同ServletRequest.getParameter(String name)?;貍鱏tring類型的值
paramValues
java.util.Map
如同ServletRequest.getParameterValues(String name)?;貍鱏tring[]類型的值
header
java.util.Map
如同ServletRequest.getHeader(String name)?;貍鱏tring類型的值
headerValues
java.util.Map
如同ServletRequest.getHeaders(String name)?;貍鱏tring[]類型的值
cookie
java.util.Map
如同HttpServletRequest.getCookies()
initParam
java.util.Map
如同ServletContext.getInitParameter(String name)?;貍鱏tring類型的值
?
不過有一點(diǎn)要注意的是如果你要用EL輸出一個(gè)常量的話,字符串要加雙引號,不然的話EL會默認(rèn)把你認(rèn)為的常量當(dāng)做一個(gè)變量來處理,這時(shí)如果這個(gè)變量在4個(gè)聲明范圍不存在的話會輸出空,如果存在則輸出該變量的值。
屬性(Attribute)與范圍(Scope)
與范圍有關(guān)的EL?隱含對象包含以下四個(gè):pageScope、requestScope、sessionScope?和
applicationScope,它們基本上就和JSP的pageContext、request、session和application一樣,所以筆者在這里只稍略說明。不過必須注意的是,這四個(gè)隱含對象只能用來取得范圍屬性值,即JSP中的getAttribute(String name),卻不能取得其他相關(guān)信息,例如:JSP中的request對象除可以存取屬性之外,還可以取得用戶的請求參數(shù)或表頭信息等等。但是在EL中,它就只能單純用來取得對應(yīng)范圍的屬性值,例如:我們要在session?中儲存一個(gè)屬性,它的名稱為username,在JSP?中使用session.getAttribute("username")?來取得username?的值, 但是在EL中,則是使用${sessionScope.username}來取得其值的。
cookie
所謂的cookie是一個(gè)小小的文本文件,它是以key、value的方式將Session Tracking的內(nèi)容記錄在這個(gè)文本文件內(nèi),這個(gè)文本文件通常存在于瀏覽器的暫存區(qū)內(nèi)。JSTL并沒有提供設(shè)定cookie的動作,因?yàn)檫@個(gè)動作通常都是后端開發(fā)者必須去做的事情,而不是交給前端的開發(fā)者。假若我們在cookie?中設(shè)定一個(gè)名稱為userCountry的值,那么可以使用${cookie.userCountry}來取得它。
?
header?和headerValues
header?儲存用戶瀏覽器和服務(wù)端用來溝通的數(shù)據(jù),當(dāng)用戶要求服務(wù)端的網(wǎng)頁時(shí),會送出一個(gè)記載要求信息的標(biāo)頭文件,例如:用戶瀏覽器的版本、用戶計(jì)算機(jī)所設(shè)定的區(qū)域等其他相關(guān)數(shù)據(jù)。假若要取得用戶瀏覽器的版本,即${header["User-Agent"]}。另外在鮮少機(jī)會下,有可能同一標(biāo)頭名稱擁有不同的值,此時(shí)必須改為使用headerValues?來取得這些值。
?
注意:因?yàn)閁ser-Agent?中包含“-”這個(gè)特殊字符,所以必須使用“[]”,而不能寫成
$(header.User-Agent)。
?
initParam
就像其他屬性一樣,我們可以自行設(shè)定web?站臺的環(huán)境參數(shù)(Context),當(dāng)我們想取得這些參數(shù)initParam就像其他屬性一樣,我們可以自行設(shè)定web?站臺的環(huán)境參數(shù)(Context),當(dāng)我們想取得這些參數(shù)
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"
version="2.4">:
:
那么我們就可以直接使用?${initParam.userid}來取得名稱為userid,其值為mike?的參數(shù)。下面是之前的做法:String userid = (String)application.getInitParameter("userid");
param和paramValues
在取得用戶參數(shù)時(shí)通常使用一下方法:
request.getParameterValues(String?name)
?
在 EL中則可以使用param和paramValues兩者來取得數(shù)據(jù)。
${param.name}
${paramValues.name}
這里param 的功能和request.getParameter(String name)相同,而paramValues和
request.getParameterValues(String name)相同。如果用戶填了一個(gè)表格,表格名稱為username,則我們就可以使用${param.username}來取得用戶填入的值。
?
看到這里,大家應(yīng)該很明確EL表達(dá)式只能通過內(nèi)置對象取值,也就是只讀操作,如果想進(jìn)行寫操作的話就讓后臺代碼去完成,畢竟EL表達(dá)式僅僅是視圖上的輸出標(biāo)簽罷了。
pageContext
我們可以使用?${pageContext}來取得其他有關(guān)用戶要求或頁面的詳細(xì)信息。下表列出了幾個(gè)比較常用的部分
Expression
說明
${pageContext.request.queryString}
取得請求的參數(shù)字符串
${pageContext.request.requestURL}
取得請求的URL,但不包括請求之參數(shù)字符串,即servlet的HTTP地址。
${pageContext.request.contextPath}
服務(wù)的webapplication的名稱
${pageContext.request.method}
取得HTTP的方法(GET、POST)
${pageContext.request.protocol}
取得使用的協(xié)議(HTTP/1.1、HTTP/1.0)
${pageContext.request.remoteUser}
取得用戶名稱
${pageContext.request.remoteAddr}
取得用戶的IP地址
${pageContext.session.new}
判斷session是否為新的,所謂新的session,表示剛由server產(chǎn)生而client尚未使用
${pageContext.session.id}
取得session的ID
${pageContext.servletContext.serverInfo}
取得主機(jī)端的服務(wù)信息
?這個(gè)對象可有效地改善代碼的硬編碼問題,如頁面中有一A標(biāo)簽鏈接訪問一個(gè)SERVLET,如果寫死了該SERVLET的HTTP地址那么如果當(dāng)該SERVLET的SERVLET-MAPPING改變的時(shí)候必須要修改源代碼,這樣維護(hù)性會大打折扣。
EL算術(shù)運(yùn)算
表達(dá)式語言支持的算術(shù)運(yùn)算符和邏輯運(yùn)算符非常多,所有在Java語言里支持的算術(shù)運(yùn)算符,表達(dá)式語言都可以使用;甚至Java語言不支持的一些算術(shù)運(yùn)算符和邏輯運(yùn)算符,表達(dá)式語言也支持。
?
Code?
上面頁面中示范了表達(dá)式語言所支持的加、減、乘、除、求余等算術(shù)運(yùn)算符的功能,讀者可能也發(fā)現(xiàn)了表達(dá)式語言還支持div、mod等運(yùn)算符。而且表達(dá)式語言把所有數(shù)值都當(dāng)成浮點(diǎn)數(shù)處理,所以3/0的實(shí)質(zhì)是3.0/0.0,得到結(jié)果應(yīng)該是Infinity。
?
如果需要在支持表達(dá)式語言的頁面中正常輸出“$”符號,則在“$”符號前加轉(zhuǎn)義字符“”,否則系統(tǒng)以為“$”是表達(dá)式語言的特殊標(biāo)記。
EL關(guān)系運(yùn)算符
關(guān)系運(yùn)算符
說明
范例
結(jié)果
==?或?eq
等于
${5==5}或${5eq5}
true
!=?或?ne
不等于
${5!=5}或${5ne5}
false
小于
${3<5}或${3lt5}
true
>?或?gt
大于
${3>5}或{3gt5}
false
<=?或?le
小于等于
${3<=5}或${3le5}
true
>=?或?ge
大于等于
5}或${3ge5}
false
表達(dá)式語言不僅可在數(shù)字與數(shù)字之間比較,還可在字符與字符之間比較,字符串的比較是根據(jù)其對應(yīng)UNICODE值來比較大小的。
注意:在使用EL 關(guān)系運(yùn)算符時(shí),不能夠?qū)懗桑?br />
${param.password1} = = ${param.password2}
或者
${ ${param.password1 } = = ${ param.password2 } }
而應(yīng)寫成
${ param.password1 = = param.password2 }
EL邏輯運(yùn)算符
邏輯運(yùn)算符
范例
結(jié)果
&&或and
交集${A && B}或${A and B}
true/false
||或or
并集${A || B}或${A or B}
true/false
!或not
非${! A }或${not A}
true/false
?
Empty?運(yùn)算符
Empty?運(yùn)算符主要用來判斷值是否為空(NULL,空字符串,空集合)。
?
條件運(yùn)算符
${ A ? B : C}