一、EL、JSTL
首先我们要明白为什么要使用EL(Expression Language 表达式语言)、JSTL(JSP Standard Tag Library JSP标准标签库),jsp页面完全可以用java代码取代他们,也就是说本来是可以没有这两种技术的,但是我们实际开发中,有一个原则:页面中不应该出现java代码。因为满篇的<% %>和java代码,会使页面极度混乱,到后期根本无法维护。所以这时EL和JSTL就出现了,使用各种标签方式,可以让页面不出现一行<% %>。JSTL是sun公司定义的标准,它不依赖于任何一种框架,所以我们应该重点掌握。
1、下面举一些简单的例子说明如何使用EL:
假设在服务器端,设置了request.setAttribute("hello", "Hello World !") ;
在jsp页面可以有两种方式显示hello中的内容:
(jsp脚本方式:)<%=request.getAttribute("hello") %> -->打印结果:Hello World !
(EL表达式,使用方法,$和{ }:)${hello} -->打印结果:Hello World !
大括号里面的hello 就是scope中的变量,也就是在服务器端setAttribute中的hello 变量。这样我们就把<%%>去掉了,而且代码也简化了很多。
在EL表达式中也有内置对象(也叫隐含对象,隐式对象都是一个意思),和jsp的内置对象基本上对应着,包括:pageScope,requestScope,sessionScope,applicationScope。
那么既然有这么多个对象,上面那个例子中,EL表达式它怎么去找那个hello呢? 它会按范围从小到大的顺序依次查找hello这个变量,上个例子中,我们把hello放在了request中,那么她首先看pageScope中有这个hello吗,发现没有,那么继续requestScope,找到了,然后就显示其内容。
这时默认的查找方法,我们还可以明确的告诉它从那个scope中去找,这样可以提高性能,因为明确了嘛,不用一个个挨着去找了,上面的例子我们可以改为:${ requestScope.hello }
注意的一点:EL和jsp脚本的不同,假设在服务器端并没有设置hello这个变量,那么jsp脚本就会显示null,而是用EL则什么都不显示。
再举一个取对象的例子:
假设定义了一个User类,里面包含了:userName, userAge, group 属性,还有相应的getter ,setter 方法。
还定义了一个Group类,里面包含了:groupName 属性,还有相应的getter, setter 方法。
在服务器端,有如下的代码:
Group group = new Group() ;
group.setGroupName("培训机构") ;
User user = new User() ;
user.setUserName("张三") ;
user.setUserAge(20) ;
user.setGroup(group) ;
request.setAttribute("userObject", user) ;
那么现在要在jsp页面中显示:
姓名:${userObject.userName}
年龄:${userObject.userAge}
所属组:${userObject.group.groupName}
这样就可以用“.”(存取器)导航把相应的信息显示出来了。内部实现的机制依旧是利用反射机制,所以“.”后面的名字要和getter方法上的名字一致。
再举一个例子用于显示Map和数组:
在服务器端,有如下代码
Map map = new HashMap() ;
map.put("key1", "value1") ;
map.put("key2", "value2") ;
String[] strArry = new String[]{"abc", "def", "ghi"}
request.setAttribute("mapvalues", map) ;
request.setAttribute("strarry", strArry) ;
在jsp中显示:
mapvalue.key1: ${ mapvalues.key1 } -->value1
mapvalue.key2: ${ mapvalues.key2 } -->value2
strArry的第二个元素:${ strarry[1]} --> def
对于对象数组,以及List的显示,也是类似的,如下语法:
${ userArray[1].userName }、 ${ userList[2].userName }
对于运算符的支持,如:${ 1+ 2 } --> 打印3
对函数的支持,如:${ empty value1} -->用于判断value1的值是否为空,为空这打印true , 不存在value1 也为true
2、JSTL(只总结核心标签C:)
*<c:out >: 用于输出,本身EL表达式就可以完成输出,为什么还要用这个呢,这里面有一个default参数,就是当为空时,可以默认输出default里面的内容,这个EL表达式是不能做到的。 例如:<c:out value="${ abc }" default="不存在abc变量" /> -->如果在服务器端没有setAttribute abc 的值,那么此时就会输出:不存在abc变量 这样的结果。 另外一种写法:<c:out value="${abc}" >不存在abc变量
如果value中的值不是EL表达式,则直接打印出其中的字符串。
还可以设置out的escapeXml属性:分为true和false,true是默认属性,即把内容原样输出;false是转意输出,比如输出的如果是html代码,则可以按html效果输出。
*<c:if>: 用于条件判断,例如 :<c:if test="${ empty value1 }" var="v">value1为空 此句代码的含义是:如果value1为空的话,则打印出 -->value1为空 并把判断结果放在v变量中。
*<c:choose>,<c:when>,<c:otherwise>: 用于条件分支选择,例如:
<c:choose>
<c:when test="${ empty value1 }">
value1为空!
<c:otherwise>
value1不为空!
可以有多个<c:when>,<c:when>必须在<c:otherwise>的前面
*<c:forEach>: 用于循环打印输出,例如:
姓名 | 年龄 | 所属组 |
没有符合条件的数据! | ||
${ user.userName} | ${ user.userAge} | ${ user.group.groupName} |
现在有另外一个需求,要求偶数行的背景设置为红色,则可以这样写代码:
姓名 | 年龄 | 所属组 |
没有符合条件的数据! | ||
${ user.userName} | ${ user.userAge} | ${ user.group.groupName} |
也可以在
还可以用forEach:标签循环取得Map中的值。例如:
<c:forEach items="mapvalue" var="v">
${ v.key } = ${ v.value }
这样就循环输出Map的内容了。
*<c:forTokens>: 用于按特定的字符分隔的输出。例如:
假设在服务器端有这样的代码:request.setAttribute("strTokens", "1, 2, 3, 4, 5") ;
<c:forTokens items="strTokens" delims="," var="v" >
${ v }
*<c:url>: 用于生成一个url 地址。例如:
<c:url value="http://localhost/sysmgr/add_user.jsp" var="v">
<c:param name="userName" value="Jack" />
<c:param name=“userAge" value=30 />
*<c:set>:用于设置一个变量的值,或者变量属性。
补充:<c:forEach>标签用法
<c:foreach>的varStatus属性暂存项的相关状态属性,可由EL表达式引用。继续上例所述。
${vs.index} 此项的索引,从0开始
${ vs .count} 此项的计数序号,从1开始
${ vs .first} 此项是否是第一项,布尔值
${ vs .last} 此项是否是最后一项,布尔值
${ vs .begin} 此次迭代的起始索引,对应<c:foreach>中begin属性值
${ vs .end} 此次迭代的终止索引,对应<c:foreach>中end属性值
${ vs .step} 此次迭代的跳跃步伐,对应<c:foreach>中step属性值