天天看點

EL表達式詳解

EL表達式詳解

<%@ page isELIgnored="true"%> 表示是否禁用EL語言,TRUE表示禁止.FALSE表示不禁止.JSP2.0中預設的啟用EL語言.

JSTL 标簽 庫由标簽庫和 EL 表達式語言兩個部分組成。 EL 在 JSTL 1.0 規範中被引入,當時用來作為 Java 表達式來工作,而該表達式必須配合 JSTL 的标簽庫才能得到需要的結果。 說明:在 JSTL 1.1 規範中, JSP2.0 容器已經能夠獨立的了解任何 EL 表達式。 EL 可以獨立出現在 JSP 頁面的任何角落。

EL 全名為Expression Language

EL 文法很簡單,它最大的特點就是使用上很友善。接下來介紹EL主要的文法結構:

${sessionScope.user.sex}

所有EL都是以${為起始、以}為結尾的。上述EL範例的意思是:從Session的範圍中,取得使用者的性别。

假若依照之前JSP Scriptlet(寫在<% %>之間的程式片)的寫法如下:

User user = (User)session.getAttribute("user");

String sex = user.getSex();

兩者相比較之下,可以發現EL 的文法比傳統JSP Scriptlet 更為友善、簡潔。

.與[]運算符

EL 提供 . 和 [ ] 兩種運算符來導航資料。下列兩者所代表的意思是一樣的:

${sessionScope.user.sex}等于${sessionScope.user["sex"]}

. 和 [ ] 也可以同時混合使用,如下:

${sessionScope.shoppingCart[0].price} 結果為shoppingCart中第一項物品的價格。

不過,以下兩種情況,兩者會有差異:

(1) 當要存取的屬性名稱中包含一些特殊字元,如. 或 – 等并非字母或數字的符号,就一定要使用 [ ],例如:${user.My-Name }

上述是不正确的方式,應當改為:${user["My-Name"] }

(2) 我們來考慮下列情況:

${sessionScope.user[data]}

此時,data 是一個變量,假若data的值為"sex"時,那上述的例子等于${sessionScope.user.sex};

假若data 的值為"name"時,它就等于${sessionScope.user.name}。是以,如果要動态取值時,就可以用上述的方法來做,但. 無法做到動态取值。

EL 變量

EL 存取變量資料的方法很簡單,例如:${username}。它的意思是取出某一範圍中名稱為username的變量。因為我們并沒有指定哪一個範圍的username,

是以它的預設值會先從Page 範圍找,假如找不到,再依序到Request、Session、Application範圍。假如途中找到username,就直接回傳,不再繼續找下去,但是假如全部的範圍都沒有找到時,就回傳null,當然EL表達式還會做出優化,頁面上顯示空白,而不是列印輸出NULL。

屬性範圍(jstl名稱):EL中的名稱

Page:PageScope

Request :RequestScope

Session :SessionScope

Application :ApplicationScope

我們也可以指定要取出哪一個範圍的變量:

範例:說明

${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");容易、簡潔許多.

自動轉變類型

EL 除了提供友善存取變量的文法之外,它另外一個友善的功能就是:自動轉變類型,我們來看下面這個範例:

${param.count + 20}

假若窗體傳來count的值為10時,那麼上面的結果為30。之前沒接觸過JSP 的讀者可能會認為上面的例子是理所當然的,但是在JSP 1.2 之中不能這樣做,原因是從窗體所傳來的值,它們的類型一律是String,是以當你接收之後,必須再将它轉為其他類型,如:int、float 等等,然後才能執行一些數學運算,下面是之前的做法:

String str_count = request.getParameter("count");

int count = Integer.parseInt(str_count);

count = count + 20;

是以,注意不要和java的文法(當字元串和數字用“+”連結時會把數字轉換為字元串)搞混淆喽。

EL 隐含對象

JSP有9個隐含對象,而EL也有自己的隐含對象。EL隐含對象總共有11 個

 隐含對象:類型:說明

 PageContext:javax.servlet.ServletContext:表示此JSP的PageContext

 PageScope:java.util.Map:取得Page範圍的屬性名稱所對應的值

 RequestScope:java.util.Map:取得Request範圍的屬性名稱所對應的值

 sessionScope:java.util.Map:取得Session範圍的屬性名稱所對應的值

 applicationScope:java.util.Map:取得Application範圍的屬性名稱所對應的值

 param:java.util.Map:如同ServletRequest.getParameter(String name)。回傳String類型的值

 paramValues:java.util.Map:如同ServletRequest.getParameterValues(String name)。回傳String[]類型的值

 header:java.util.Map:如同ServletRequest.getHeader(String name)。回傳String類型的值

 headerValues:java.util.Map:如同ServletRequest.getHeaders(String name)。回傳String[]類型的值

 cookie:java.util.Map:如同HttpServletRequest.getCookies()

 initParam:java.util.Map:如同ServletContext.getInitParameter(String name)。回傳String類型的值

不過有一點要注意的是如果你要用EL輸出一個常量的話,字元串要加雙引号,不然的話EL會預設把你認為的常量當做一個變量來處理,

這時如果這個變量在4個聲明範圍不存在的話會輸出空,如果存在則輸出該變量的值。

屬性(Attribute)與範圍(Scope)

與範圍有關的EL 隐含對象包含以下四個:pageScope、requestScope、sessionScope 和applicationScope,它們基本上就和JSP的pageContext、request、session和application一樣。

不過必須注意的是,這四個隐含對象隻能用來取得範圍屬性值,即JSP中的getAttribute(String name),卻不能取得其他相關資訊,

例如:JSP中的request對象除可以存取屬性之外,還可以取得使用者的請求參數或表頭資訊等等。但是在EL中,它就隻能單純用來取得對應範圍的屬性值,

例如:我們要在session 中儲存一個屬性,它的名稱為username,在JSP 中使用session.getAttribute("username") 來取得username 的值,

但是在EL中,則是使用${sessionScope.username}來取得其值的。

cookie

所謂的cookie是一個小小的文本檔案,它是以key、value的方式将Session Tracking的内容記錄在這個文本檔案内,這個文本檔案通常存在于浏覽器的暫存區内。

JSTL并沒有提供設定cookie的動作,因為這個動作通常都是後端開發者必須去做的事情,而不是交給前端的開發者。假若我們在cookie 中設定一個名稱為userCountry的值,

那麼可以使用${cookie.userCountry}來取得它。

header 和headerValues

header 儲存使用者浏覽器和服務端用來溝通的資料,當使用者要求服務端的網頁時,會送出一個記載要求資訊的标頭檔案,

例如:使用者浏覽器的版本、使用者計算機所設定的區域等其他相關資料。假若要取得使用者浏覽器的版本,即${header["User-Agent"]}。

另外在鮮少機會下,有可能同一标頭名稱擁有不同的值,此時必須改為使用headerValues 來取得這些值。

注意:因為User-Agent 中包含“-”這個特殊字元,是以必須使用“[]”,而不能寫成$(header.User-Agent)。

initParam

就像其他屬性一樣,我們可以自行設定web 站台的環境參數(Context),當我們想取得這些參數initParam就像其他屬性一樣,我們可以自行設定web 站台的環境參數(Context),

當我們想取得這些參數

<?xml version="1.0" encoding="ISO-8859-1"?>

<web-app xmlns="​​http://java.sun.com/xml/ns/j2ee​​" 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">:

<context-param>

<param-name>userid</param-name>

<param-value>mike</param-value>

</context-param>:

</web-app>

那麼我們就可以直接使用 ${initParam.userid}來取得名稱為userid,其值為mike 的參數。下面是之前的做法:

String userid = (String)application.getInitParameter("userid");

param和paramValues

在取得使用者參數時通常使用一下方法:

request.getParameter(String name)

request.getParameterValues(String name)

在 EL中則可以使用param和paramValues兩者來取得資料。

${param.name}

${paramValues.name}

這裡param 的功能和request.getParameter(String name)相同,而paramValues和

request.getParameterValues(String name)相同。如果使用者填了一個表格,表格名稱為username,則我們就可以使用${param.username}來取得使用者填入的值。

看到這裡,大家應該很明确EL表達式隻能通過内置對象取值,也就是隻讀操作,如果想進行寫操作的話就讓背景代碼去完成,畢竟EL表達式僅僅是視圖上的輸出标簽罷了。

pageContext

我們可以使用 ${pageContext}來取得其他有關使用者要求或頁面的詳細資訊。下表列出了幾個比較常用的部分

Expression:說明

${pageContext.request.queryString}:取得請求的參數字元串

${pageContext.request.requestURL}:取得請求的URL,但不包括請求之參數字元串,即servlet的HTTP位址。

${pageContext.request.contextPath}:服務的webapplication的名稱

${pageContext.request.method}:取得HTTP的方法(GET、POST)

${pageContext.request.protocol}:取得使用的協定(HTTP/1.1、HTTP/1.0)

${pageContext.request.remoteUser}:取得使用者名稱

${pageContext.request.remoteAddr}:取得使用者的IP位址

${pageContext.session.new}:判斷session是否為新的,所謂新的session,表示剛由server産生而client尚未使用

${pageContext.session.id}:取得session的ID

${pageContext.servletContext.serverInfo}:取得主機端的服務資訊

這個對象可有效地改善代碼的寫死問題,如頁面中有一A标簽連結通路一個SERVLET,如果寫死了該SERVLET的HTTP位址

那麼如果當該SERVLET的SERVLET-MAPPING改變的時候必須要修改源代碼,這樣維護性會大打折扣。

EL算術運算

表達式語言支援的算術運算符和邏輯運算符非常多,所有在Java語言裡支援的算術運算符,表達式語言都可以使用;

甚至Java語言不支援的一些算術運算符和邏輯運算符,表達式語言也支援。

表達式語言所支援的加、減、乘、除、求餘等算術運算符的功能,讀者可能也發現了表達式語言還支援div、mod等運算符。

而且表達式語言把所有數值都當成浮點數處理,是以3/0的實質是3.0/0.0,得到結果應該是Infinity。

如果需要在支援表達式語言的頁面中正常輸出“$”符号,則在“$”符号前加轉義字元“\”,否則系統以為“$”是表達式語言的特殊标記。

EL關系運算符

關系運算符:說明:範例:結果

== 或 eq:等于:${5==5}或${5eq5}:true

!= 或 ne:不等于:${5!=5}或${5ne5}:false

< 或 lt:小于:${3<5}或${3lt5}:true

> 或 gt:大于:${3>5}或{3gt5}:false

<= 或 le:小于等于:${3<=5}或${3le5}:true

>= 或 ge:大于等于:3>=5}或${3ge5}:false

 表達式語言不僅可在數字與數字之間比較,還可在字元與字元之間比較,字元串的比較是根據其對應UNICODE值來比較大小的。

注意:在使用EL 關系運算符時,不能夠寫成:${param.password1} = = ${param.password2}或者${ ${param.password1 } = = ${ param.password2 } }

而應寫成:${ param.password1 = = param.password2 }

EL邏輯運算符

邏輯運算符: 範例: 結果

&&或and: 交集${A && B}或${A and B}: true/false

||或or: 并集${A || B}或${A or B}: true/false

!或not: 非${! A }或${not A}:true/false

Empty 運算符

Empty 運算符主要用來判斷值是否為空(NULL,空字元串,空集合)。