useBean>操作,以及由ServletAPI中的几个类定义的setAttribute()方法。
此外,四个JSTL库中定义的许多定制标记本身就能够设置作为限制了作用域的变量使用的属性值。
隐式对象
表1中列出了11个EL隐式对象的标识符。
不要将这些对象与JSP隐式对象(一共只有九个)混淆,其中只有一个对象是它们所共有的。
表1.EL隐式对象
类别
标识符
描述
JSP
pageContext
PageContext实例对应于当前页面的处理
作用域
pageScope
与页面作用域属性的名称和值相关联的Map类
requestScope
与请求作用域属性的名称和值相关联的Map类
sessionScope
与会话作用域属性的名称和值相关联的Map类
applicationScope
与应用程序作用域属性的名称和值相关联的Map类
请求参数
param
按名称存储请求参数的主要值的Map类
paramValues
将请求参数的所有值作为String数组存储的Map类
请求头
header
按名称存储请求头主要值的Map类
headerValues
将请求头的所有值作为String数组存储的Map类
Cookie
cookie
按名称存储请求附带的cookie的Map类
初始化参数
initParam
按名称存储Web应用程序上下文初始化参数的Map类
尽管JSP和EL隐式对象中只有一个公共对象(pageContext),但通过EL也可以访问其它JSP隐式对象。
原因是pageContext拥有访问所有其它八个JSP隐式对象的特性。
实际上,这是将它包括在EL隐式对象中的主要理由。
其余所有EL隐式对象都是映射,可以用来查找对应于名称的对象。
前四个映射表示先前讨论的各种属性作用域。
可以用它们来查找特定作用域中的标识符,而不用依赖于EL在缺省情况下使用的顺序查找过程。
接下来的四个映射用来获取请求参数和请求头的值。
因为HTTP协议允许请求参数和请求头具有多个值,所以它们各有一对映射。
每对中的第一个映射返回请求参数或头的主要值,通常是恰巧在实际请求中首先指定的那个值。
每对中第二个映射允许检索参数或头的所有值。
这些映射中的键是参数或头的名称,但这些值是String对象的数组,其中的每个元素都是单一参数值或头值。
cookie隐式对象提供了对由请求设置的cookie名称的访问。
这个对象将所有与请求相关联的cookie名称映射到表示那些cookie特性的Cookie对象。
最后一个EL隐式对象initParam是一个映射,它储存与Web应用程序相关联的所有上下文的初始化参数的名称和值。
初始化参数是通过web.xml部署描述符文件指定的,该文件位于应用程序的WEB-INF目录中。
存取器
因为EL标识符是作为隐式对象或限制了作用域的变量(通过属性来实现)解析的,因此有必要将它们转换成Java对象。
EL可以自动包装和解包其相应的Java类中的基本类型(例如,可以在后台将int强制转换成Integer类,反之亦可),但大多数的标识符将成为指向完整的Java对象的指针。
结果是,对这些对象的特性或(在对象是数组和集合的情况下)对其元素的访问通常是令人满意的。
就为了实现这种用途,EL提供了两种不同的存取器(点运算符(.)和方括号运算符([])),也支持通过EL操作特性和元素。
点运算符通常用于访问对象的特性。
例如,在表达式${user.firstName}中,使用点运算符来访问user标识符所引用对象的名为firstName的特性。
EL使用Javabean约定访问对象特性,因此必须定义这个特性的getter方法(通常是名为getFirstName()的方法),以便表达式正确求值。
当被访问的特性本身是对象时,可以递归地应用点运算符。
例如,如果我们虚构的user对象有一个实现为Java对象的address特性,那么也可以用点运算符来访问这个对象的特性。
例如,表达式${user.address.city}将会返回这个地址对象嵌套的city特性。
方括号运算符用来检索数组和集合的元素。
在数组和有序集合(也即,实现了java.util.List接口的集合)的情况下,把要检索的元素的下标放在方括号中。
例如,表达式${urls[3]}返回urls标识符所引用的数组或集合的第四个元素(和Java语言以及JavaScript中一样,EL中的下标是从零开始的)。
对于实现java.util.Map接口的集合,方括号运算符使用关联的键查找存储在映射中的值。
在方括号中指定键,并将相应的值作为表达式的值返回。
例如,表达式${commands["dir"]}返回与commands标识符所引用的Map中的"dir"键相关联的值。
对于上述两种情况,都可允许表达式出现在方括号中。
对嵌套表达式求值的结果将被作为下标或键,用来检索集合或数组的适当元素。
和点运算符一样,方括号运算符也可以递归应用。
这使得EL能够从多维数组、嵌套集合或两者的任意组合中检索元素。
此外,点运算符和方括号运算符还可以互操作。
例如,如果数组的元素本身是对象,则可以使用方括号运算符来检索该数组的元素,并结合点运算符来检索该元素的一个特性(例如${urls[3].protocol})。
假定EL充当指定动态属性值的简化语言,EL存取器有一个有趣的功能(与Java语言的存取器不同),那就是它们在应用于null时不抛出异常。
如果应用EL存取器的对象(例如,${foo.bar}和${foo["bar"]}中的foo标识符)是null,那么应用存取器的结果也是null。
事实证明,在大多数情况下,这是一个相当有用的行为,不久您就会了解这一点。
最后,点运算符和方括号运算符可能实现某种程度的互换。
例如,也可以使用${user["firstName"]}来检索user对象的firstName特性,正如可以用${commands.dir}获取与commands映射中的"dir"键相关联的值一样。
运算符
EL还可以通过使用标识符和存取器,遍历包含应用程序数据(通过限制了作用域的变量公开)或关于环境的信息(通过EL隐式对象)的对象层次结构。
但是,只是访问这些数据,通常不足以实现许多JSP应用程序所需的表示逻辑。
最终,EL还包括了几个用来操作和比较EL表达式所访问数据的运算符。
表2中汇总了这些运算符。
表2.EL运算符
类别
运算符
算术运算符
+、-、*、/(或div)和%(或mod)
关系运算符
==(或eq)、!
=(或ne)、<(或lt)、>(或gt)、<=(或le)和>=(或ge)
逻辑运算符
&&(或and)、||(或or)和!
(或not)
验证运算符
empty
算术运算符支持数值的加法、减法、乘法和除法。
还提供了一个求余运算符。
注:
除法和求余运算符都有替代的、非符号的名称(为的是与XPath保持一致)。
清单5中显示了一个演示算术运算符用法的示例表达式。
对几个EL表达式应用算术运算符的结果是将该算术运算符应用于这些表达式返回的数值所得的结果。
清单5.利用算术运算符的EL表达式
${item.price*(1+taxRate[user.address.zipcode])}
关系运算符允许比较数字或文本数据。
比较的结果作为布尔值返回。
逻辑运算符允许合并布尔值,返回新的布尔值。
因此,可以将EL逻辑运算符应用于嵌套的关系或逻辑运算符的结果,如清单6所示。
清单6.利用关系和逻辑运算符的EL表达式
${(x>=min)&&(x<=max)}
最后一种EL运算符是empty,它对于验证数据特别有用。
empty运算符采用单个表达式作为其变量(也即,${emptyinput}),并返回一个布尔值,该布尔值表示对表达式求值的结果是不是“空”值。
求值结果为null的表达式被认为是空,即无元素的集合或数组。
如果参数是对长度为零的String求值所得的结果,则empty运算符也将返回true。
表3显示了EL运算符的优先级。
正如清单5和6所示,可以用圆括号对表达式分组,高于普通的优先级规则。
表3.EL运算符优先级(自顶到底,从左到右)
[],.
()
unary-、not、!
、empty
*、/、div、%、mod
+、binary-
()<、>、<=、>=、lt、gt、le、ge
==、!
=、eq、ne
&&、and
||、or
回页首
文字
在EL表达式中,数字、字符串、布尔值和null都可以被指定为文字值。
字符串可以用单引号或双引号定界。
布尔值被指定为true和false。
Taglib伪指令
正如我们先前讨论的,JSTL1.0包括四个定制标记库。
为了演示JSTL标记和表达式语言的交互,我们将研究几个来自JSTLcore库的标记。
和使用任何JSP定制标记库一样,必须在您想要使用这个库标记的任何页面中包括taglib伪指令。
清单7显示了用于这个特定库的伪指令。
清单7.用于JSTLcore库EL版本的taglib伪指令
<%@tagliburi="prefix="c"%>
实际上,对应于JSTLcore库的taglib伪指令有两种,因为在JSTL1.0中,EL是可选的。
所有四个JSTL1.0定制标记库都有使用JSP表达式(而不是EL)指定动态属性值的备用版本。
因为这些备用库依赖于JSP的更传统的请求时属性值,所以它们被称为RT库,而那些使用表达式语言的则被称为EL库。
开发人员用不同的taglib伪指令来区分每个库的这两个版本。
清单8显示了使用core库的RT版本的伪指令。
但是,由于现在我们讨论的重点是EL,所以首先需要这些伪指令。
清单8.用于JSTLcore库RT版本的taglib伪指令
<%@tagliburi="prefix="c_rt"%>
回页首
变量标记
我们首先要考虑的JSTL定制标记是set>操作。
正如已经说明的,限制了作用域的变量在JSTL中起关键作用,set>操作提供基于标记的机制来创建和设置限制了作用域的变量。
清单9中显示了该操作的语法,其中var属性指定了限制了作用域的变量的名称,scope属性表明了该变量驻留在哪个作用域中,value属性指定了分配给该变量的值。
如果指定变量已经存在,则简单地将所指明的值赋给它。
如果不存在,则创建新的限制了作用域的变量,并用该值初始化这个变量。
清单9.set>操作的语法
setvar="name"scope="scope"value="expression"/>
scope属性是可选的,其缺省值是page。
清单10中显示了set>的两个示例。
在第一个示例中,将会话作用域变量设置成String值。
在第二个示例中,用表达式来设置数值:
将页面作用域内名为square的变量赋值为名为x的请求参数的值的平方。
清单10.set>操作示例
setvar="timezone"scope="session"value="CST"/>
setvar="square"value="${param['x']*param['x']}"/>
您还可以将限制了作用域的变量的值指定为set>操作的主体内容,而不是使用属性。
使用这种方法,您可以重新编写清单10中的第一个示例,如清单11所示。
此外,正如我们马上可以看到的,set>标记的主体内容本身也可以使用定制标记。
set>主体内生成的所有内容都将作为一个String值赋给指定变量。
清单11.通过主体内容指定set>操作的值
setvar="timezone"scope="session">CST
set>
JSTLcore库包含第二个用于管理限制了作用域的变量的标记―remove>。
顾名思义,remove>操作是用来删除限制了作用域的变量的,它获取两个属性。
var属性指定待删除变量的名称,scope属性是可选的,它表示待删除变量