一、EL概述与基本语法
el(Expression Language)叫做表达式语言,他并不是jsp中一开始就有的,而是jsp2.0新增的技术规范。因此,首先要明确的是,el表达式是jsp的一部分,它并不是某个框架里面的,想当初我是学框架的时候才知道的这个,以为这个是框架里的呢!所以呢,你可以很自然的使用它在jsp页面中。
el的设计主要目的是简化页面输出,它能够完美替代传统jsp中的输出表达式<%=%>,并且还提供了额外的好处,例如以更自然的方式输出JavaBean的属性信息等。
语法:${有值得表达式}
二、EL输出常量、变量的值
通过el输出常量或者变量的值,用法与输出表达式<%=%>基本相同,不一样的是,当变量是null时,使用<%=%>可能会抛出空指针异常,而el不会抛出异常,此外,el还做了一部分变量类型自动转换的工作。
- 输出常量
- 输出变量
- 写在标签属性中
具体看下面代码:
<%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8"%>
<!DOCTYPE html>
<html lang="en">
<head>
<meta http-equiv="Content-Type" content="text/html;charset=utf-8">
<meta name="keywords" content="">
<meta name="description" content="">
<link rel="stylesheet" type="text/css" href="">
<title>test</title>
<script src=""></script>
</head>
<body>
<!-- 1.输出常量-->
${"hello world"}
<!-- 2.输出变量-->
<%
//一般request对象的值都是从别的action传过来的,
//但我为了简洁就直接在本jsp页面操作了
String s="hello world";
request.setAttribute("msg",s);
%>
${msg}
<!-- 3.在标签的属性中使用 -->
<%
String addr="www.baidu.com";
request.setAttribute("url",addr);
%>
<a href="${url}">这是超链接</a>
</body>
</html>
效果:
三、EL的运算符
在EL中可以进行算术运算、逻辑运算、关系运算。特别地,还有空运算符。
- 算术运算符(+、-、*、/、%等).
- 逻辑运算符(&&或and、||或者or、not)。结果true或false
- 关系运算符(<、>、==、<=、!=).结果true或false
- 空运算符,${empty null},结果为true,判断参数是不是null、空字符串、空数组、空Map或者空集合。
四、EL中与作用域相关的隐含对象
前面展示了EL与JSP输出表达式<%=%>类似功能的部分,如果EL的功能只尽于此,那么也就没有使用的必要了。除了一般的输出任务,EL还提供了更为简洁的语法形式,以便访问、输出某个** 域中对象的属性。**
首先要知道JSP中隐含的对象:
- request对象,属于HttpServletRequest,范围是request,用来处理请求。
- response对象,属于HttpServletResponse,范围是page,用来处理响应。
- session对象,属于HttpSession,范围是session,用来处理会话。
- page对象,转译为this,范围为page,相当于java中的this。
- application对象,属于ServletContext,范围application,用来处理上下文。
- exception对象,属于Throwable,范围是page,只出现于isErrorPage设定为true的jsp页面。
- config对象,属于ServletConfig,范围是page,处理转译后的Servlet配置信息。
- pageContext对象,属于PageContext,范围是page,可利用其对页面信息进行封装。
- out对象,属于JspWriter,范围是page,处理输出。
注意:这里的out对象与Servlet中的那个out对象不相同,Servlet中使用的是out变量类型是PrintWriter,而这里的out对象类型是JspWriter。
在这里,我们使用的是page、request、session、application四个对象,这四个对象分别对应着el表达式中的pageScope、requestScope、sessionScope、applicationScope。这四个对象都是映射类型Map,可以通过key-value访问,其实说白了就是"变量名=值",就是平常访问变量的形式。
使用场景:
一般的使用情况是,在一个jsp页面中,把一些的基本变量或复杂的变量(即对象),存到这四个之中的任意一个对象中(具体存哪个对象得看你想让他存留多久,然后找对应的对象生命周期)。然后在页面进行跳转时,这个隐含的对象就会进行传送,等到了一个新的页面,你就可以利用对象.变量名或** 对象[变量名] **来取到你之前存进去的值。
注意:这里说的jsp页面,不仅仅指网页,也可以从servlet传到jsp页面,说到底,jsp也是个servlet!
说了这么多,还是写个例子更加明白。为了图方便,我没有在页面之间进行传递,我只是举例子说明能把对象存到隐含对象中,再从中取值这一过程,我存到request对象,一般都是存到request对象中,毕竟一个页面到另一个页面就是一个请求与响应的过程。
<%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8"%>
<%@ page import= "java.util.HashMap" %>
<%@ page import= "java.util.List" %>
<%@ page import= "java.util.ArrayList" %>
<!DOCTYPE html>
<html lang="en">
<head>
<meta http-equiv="Content-Type" content="text/html;charset=utf-8">
<meta name="keywords" content="">
<meta name="description" content="">
<link rel="stylesheet" type="text/css" href="">
<title>test</title>
<script src=""></script>
</head>
<body>
<!-- 1.输出常量-->
<%
//存基本数据类型到request对象中。
int a=2;
request.setAttribute("aa", a);
//存Map类型的对象
HashMap<String,String> fruitMap = new HashMap<String,String>();
fruitMap.put("no1", "香蕉");
fruitMap.put("no2", "苹果");
request.setAttribute("fmap", fruitMap);
//存数组类型的对象
String[] fruits = {"apple","banana"};
request.setAttribute("farray", fruits);
//存list
List<String> fruitList = new ArrayList<String>();
fruitList.add("西瓜");
fruitList.add("哈密瓜");
request.setAttribute("flist",fruitList);
%>
基本类型:${requestScope.aa} <br/>
Map类型:${requestScope.fmap.no1} ${requestScope.fmap[no2] }<br/>
数组类型:${requestScope.farray[0] } ${requestScope.farray[1] }<br/>
list类型:${requestScope.flist[0] } ${requestScope.flist[1] }<br/>
<hr/>
基本类型:${aa} <br/>
Map类型:${fmap.no1} ${fmap[no2] }<br/>
数组类型:${farray[0] } ${farray[1] }<br/>
list类型:${flist[0] } ${flist[1] }
</body>
</html>
结果:
注意看第二次输出,我把requestScope省略掉了,照样可以找到值,那么就有一个问题,我没有指定从哪个隐含对象中取值,那么系统怎么判断呢?其实系统是挨个扫描,看看这些对象中哪个有指定变量的值,但是问题又来了,如果这四个隐含对象都存取了同名的对象,那么从哪个里面取呢,这就涉及优先级了。其实,当没有明确指定对象的作用域时,el将按照作用域范围从小到大的顺序搜索满足条件的key值,即pageScope–>requestScope–>sessionScope–>applicationScope这个顺序输出,找到一个就输出。