Jsp表達式語(yǔ)言(EL)
Sun Microsystems 在 1997 年下半年推出了Servlet API,將它定位為 CGI 開(kāi)發(fā)人員使用的一種功能強大的工具,這些開(kāi)發(fā)人員正在尋找比CGI(通用網(wǎng)關(guān)接口)編程更高效和輕便的優(yōu)秀解決方案。但是,開(kāi)發(fā)人員很快就發(fā)現 Servlet API有其自身的缺點(diǎn),從代碼的可維護性和可擴展性方面來(lái)看,該解決方案難以實(shí)施。在某種程度上,這種缺點(diǎn)促使團隊開(kāi)發(fā)一種允許在 HTML 中嵌入Java 代碼的解決方案 — JavaServer Pages (JSP) 因此而出現。
不久以后,開(kāi)發(fā)人員意識到將表達與商務(wù)邏輯混合在一起的復雜 JSP 頁(yè)不易于理解和維護。不能編寫(xiě)scriplet 的頁(yè)面制作人員所面臨的另一個(gè)問(wèn)題是由于標準標記集而帶來(lái)的 JSP 限制。這些限制使得難點(diǎn)變成利用 JSP實(shí)施自定義標記的機制來(lái)創(chuàng )建 JSP 自定義標記。
JSP 標準標記庫 (JSTL) 是自定義標記庫的集合,它將許多 JSP 應用程序通用的核心功能封裝為簡(jiǎn)單的標記。它不再需要使用 JSP scriptlet
和表達式,而使用表達式的更高級語(yǔ)法。它還實(shí)現了通用目的的功能,如迭代和條件化、數據管理格式化、XML操作、數據庫訪(fǎng)問(wèn)、國際化和對本地化信息敏感的格式化標記以及 SQL 標記。JSTL 1.0 推出了 EL 的概念,但只限于 JSTL 標記。在JSP 2.0 中,您可以使用帶模板文本的 EL,甚至可以通過(guò) javax.servlet.jsp.el 獲得編程方式的訪(fǎng)問(wèn)。
在我們了解 JSTL 如何適應環(huán)境以及與 JSTL 表達式語(yǔ)言相關(guān)的限制以后,我們來(lái)看 JSP 2.0 的重要優(yōu)點(diǎn)之一 — JSP 表達式語(yǔ)言 (EL)。我們將特別涉及到以下內容:
JSP 表達式語(yǔ)言定義在無(wú)腳本的 JSP 頁(yè)面中支持 EL 的機制表達式語(yǔ)言語(yǔ)法JSP EL 中有效的表達式使用 EL 表達式表達式語(yǔ)言的靈感來(lái)自于 ECMAScript 和 XPath 表達式語(yǔ)言,它提供了在 JSP中簡(jiǎn)化表達式的方法。它是一種簡(jiǎn)單的語(yǔ)言,基于可用的命名空間(PageContext屬性)、嵌套屬性和對集合、操作符(算術(shù)型、關(guān)系型和邏輯型)的訪(fǎng)問(wèn)符、映射到 Java 類(lèi)中靜態(tài)方法的可擴展函數以及一組隱式對象。
EL 提供了在 JSP 腳本編制元素范圍外使用運行時(shí)表達式的功能。腳本編制元素是指頁(yè)面中能夠用于在 JSP 文件中嵌入 Java 代碼的元素。它們通常用于對象操作以及執行那些影響所生成內容的計算。JSP 2.0 將 EL 表達式添加為一種腳本編制元素。
腳本編制元素具有三個(gè)從屬形式:
聲明
Scriptlet
表達式。
讓我們來(lái)看代碼中的這三種從屬形式:
table.getColumn( )
在將 EL 添加到 JSP 工具包以后,可以使用更簡(jiǎn)單的語(yǔ)法來(lái)編寫(xiě)以上的代碼,而獲得與以上 JSP元素相同的結果。EL 表達式的另一個(gè)優(yōu)勢是,可以在不允許使用上述任何腳本編制元素從屬形式的無(wú)腳本的 JSP頁(yè)中使用它。但是必須注意,可以不使用三種腳本編制元素從屬形式中的任一種來(lái)編寫(xiě) JSP 頁(yè),而對 JSP頁(yè)是否應該為無(wú)腳本形式的選擇則完全基于應用程序的需求。如果您希望明確分開(kāi)表達與商務(wù)邏輯,則還可以選擇將頁(yè)面強制為無(wú)腳本形式。通過(guò)強制成為無(wú)腳本頁(yè)面,必須通過(guò)其他元素(如 JavaBean、EL 表達式、自定義操作和標準標記庫)提供 JSP 頁(yè)的動(dòng)態(tài)行為。
有兩種機制可以確保頁(yè)面不包含任何腳本編制元素。每種機制還提供在無(wú)腳本的 JSP 頁(yè)中支持 EL 的方法。
使用頁(yè)面指示:
在使用頁(yè)面指示時(shí),您可以通過(guò)將 isELEnabled 指示的值相應地設為“true”或“false”,指定是否支持 EL,如下所示:
使用部署描述符的元素:
當使用部署描述符的元素時(shí),您可以通過(guò)在 標記間包含布爾值“true”或“false”,指定是否支持 EL,如下所示:
...
*.jsp
true
true
....
JSP 表達式語(yǔ)言允許頁(yè)面制作人員使用簡(jiǎn)單語(yǔ)法訪(fǎng)問(wèn)組件,如:
${expr}在以上的語(yǔ)法中,expr 代表有效的表達式。必須注意,該表達式可以與靜態(tài)文本混合,還可以與其他表達式結合成更大的表達式。
有效表達式可以包含文字、操作符、變量(對象引用)和函數調用。我們將分別了解這些有效表達式中的每一種:
文字
JSP 表達式語(yǔ)言定義可在表達式中使用的以下文字:
文字 文字的值
Boolean
true 和 false
Integer
與 Java 類(lèi)似??梢园魏握龜祷蜇摂?,例如 24、-45、567
Floating Point
與 Java 類(lèi)似??梢园魏握幕蜇摰母↑c(diǎn)數,例如 -1.8E-45、4.567
String
任何由單引號或雙引號限定的字符串。對于單引號、雙引號和反斜杠,使用反斜杠字符作為轉義序列。必須注意,如果在字符串兩端使用雙引號,則單引號不需要轉義。
Null null
讓我們來(lái)看一些使用文字作為有效表達式的示例:
${false}
${3*8)操作符
JSP 表達式語(yǔ)言提供以下操作符,其中大部分是 Java 中常用的操作符:
術(shù)語(yǔ) 定義
算術(shù)型
+、-(二元)、*、/、div、%、mod、-(一元)
邏輯型
and、&&、or、||、!、not
關(guān)系型
==、eq、!=、ne、、gt、<=、le、>=、ge??梢耘c其他值進(jìn)行比較,或與布爾型、字符串型、整型或浮點(diǎn)型文字進(jìn)行比較。
空
空操作符是前綴操作,可用于確定值是否為空。
條件型 A ?B :C。根據 A 賦值的結果來(lái)賦值 B 或 C。
讓我們來(lái)看一些使用操作符作為有效表達式的示例:
${ (6 * 5) + 5 }
${empty name}隱式對象
JSP 表達式語(yǔ)言定義了一組隱式對象,其中許多對象在 JSP scriplet 和表達式中可用:
術(shù)語(yǔ) 定義
pageContext
JSP 頁(yè)的上下文。它可以用于訪(fǎng)問(wèn) JSP 隱式對象,如請求、響應、會(huì )話(huà)、輸出、servletContext 等。例如,${pageContext.response} 為頁(yè)面的響應對象賦值。
此外,還提供幾個(gè)隱式對象,允許對以下對象進(jìn)行簡(jiǎn)易訪(fǎng)問(wèn):
術(shù)語(yǔ) 定義
param
將 請求參數名稱(chēng)映射到單個(gè)字符串參數值(通過(guò)調用 ServletRequest.getParameter (String name) 獲得)。getParameter (String) 方法返回帶有特定名稱(chēng)的參數。表達式 $(param.name) 相當于 request.getParameter (name)。
paramValues
將 請求參數名稱(chēng)映射到一個(gè)數值數組(通過(guò)調用 ServletRequest.getParameter (String name) 獲得)。它與 param 隱式對象非常類(lèi)似,但它檢索一個(gè)字符串數組而不是單個(gè)值。表達式 ${paramvalues.name) 相當于 request.getParamterValues(name)。
header
將請求頭名稱(chēng)映射到單個(gè)字符串頭值(通過(guò)調用 ServletRequest.getHeader(String name) 獲得)。表達式 ${header.name} 相當于 request.getHeader(name)。
headerValues
將 請求頭名稱(chēng)映射到一個(gè)數值數組(通過(guò)調用 ServletRequest.getHeaders(String) 獲得)。它與頭隱式對象非常類(lèi)似。表達式 ${headerValues.name} 相當于 request.getHeaderValues(name)。
cookie 將 cookie 名稱(chēng)映射到單個(gè) cookie 對象。向服務(wù)器發(fā)出的客戶(hù)端請求可以獲得一個(gè)或多個(gè) cookie。表達式 ${cookie.name.value} 返回帶有特定名稱(chēng)的第一個(gè) cookie 值。如果請求包含多個(gè)同名的 cookie,則應該使用 ${headerValues.name} 表達式。
initParam 將上下文初始化參數名稱(chēng)映射到單個(gè)值(通過(guò)調用 ServletContext.getInitparameter(String name) 獲得)。
除了上述兩種類(lèi)型的隱式對象之外,還有些對象允許訪(fǎng)問(wèn)多種范圍的變量,如 Web 上下文、會(huì )話(huà)、請求、頁(yè)面:
術(shù)語(yǔ) 定義
pageScope
將頁(yè)面范圍的變量名稱(chēng)映射到其值。例如,EL 表達式可以使用 ${pageScope.objectName} 訪(fǎng)問(wèn)一個(gè) JSP 中頁(yè)面范圍的對象,還可以使用 ${pageScope.objectName.attributeName} 訪(fǎng)問(wèn)對象的屬性。
requestScope
將 請求范圍的變量名稱(chēng)映射到其值。該對象允許訪(fǎng)問(wèn)請求對象的屬性。例如,EL 表達式可以使用 ${requestScope.objectName} 訪(fǎng)問(wèn)一個(gè) JSP 請求范圍的對象,還可以使用 ${requestScope.objectName.attributeName} 訪(fǎng)問(wèn)對象的屬性。
sessionScope
將會(huì )話(huà)范圍的變量名稱(chēng)映射到其值。該對象允許訪(fǎng)問(wèn)會(huì )話(huà)對象的屬性。例如:
$sessionScope.name}
applicationScope
將應用程序范圍的變量名稱(chēng)映射到其值。該隱式對象允許訪(fǎng)問(wèn)應用程序范圍的對象。
必須注意,當表達式根據名稱(chēng)引用這些對象之一時(shí),返回的是相應的對象而不是相應的屬性。例如:即使現有的 pageContext 屬性包含某些其他值,${pageContext} 也返回 PageContext 對象。
EL 表達式可用于兩種情況:
作為標準操作和自定義操作中的屬性值
在 JSP 文件的模板文本中,如 HTML 或非 JSP 元素 — 在這種情況下,模板文本中的表達式被賦值并插入到當前的輸出中。但是,必須注意,如果標記的主體被聲明為與標記相關(guān),則不會(huì )對表達式賦值。