1、EL表达式
1.1 概述
① EL 表达式的全称是:Expression Language。是表达式语言。
② EL 表达式的什么作用:EL 表达式主要是代替jsp 页面中的表达式脚本在jsp 页面中进行数据的输出。
③ 因为EL 表达式在输出数据的时候,要比jsp 的表达式脚本要简洁很多。
语法:
${表达式} //注意区别JSP表达式
说明:EL 表达式在输出null 值的时候,输出的是空串。jsp 表达式脚本输出null 值的时候,输出的是null 字符串。
exp
<body>
先在主页设置一个值
<%
request.setAttribute("key","值");
%>
JSP 表达式脚本输出key 的值是:
<%=request.getAttribute("key1")==null?"":request.getAttribute("key1")%><br/>
EL 表达式输出key 的值是:${key1}
</body>
1.2 EL 表达式搜索域数据的顺序
① EL 表达式主要是在jsp 页面中输出数据,主要是输出域对象中的数据。
② 当四个域中都有相同的key 的数据的时候,EL 表达式会按照四个域的从小到大的顺序去进行搜索,找到就输出。
<body>
<%
//往四个域中都保存了相同的key 的数据。
request.setAttribute("key", "request");
session.setAttribute("key", "session");
application.setAttribute("key", "application");
pageContext.setAttribute("key", "pageContext");
%>
${ key } //按照四个域的从小到大的顺序去进行搜索,找到就输出
</body>
1.3 EL 表达式输出Bean 的普通属性,数组属性。List 集合属性,map 集合属性
需求——输出Person 类中普通属性,数组属性。list 集合属性和map 集合属性。
//Person类
public class Person{
// 需求——输出Person 类中普通属性,数组属性。list 集合属性和map 集合属性。
private String name;
private String[] phones;
private List<String> cities;
private Map<String,Object> map;
//Bean组件,需要空参构造器,带参构造器,toString,get,set方法
public int getAge() {
return 18;
}
//等等
}
输出页面 Jsp页面
<body>
<%
Person person = new Person();
person.setName("国哥好帅!");
person.setPhones(new String[]{"18610541354","18688886666","18699998888"});
List<String> cities = new ArrayList<String>();
cities.add("北京");
cities.add("上海");
cities.add("深圳");
person.setCities(cities);
Map<String,Object>map = new HashMap<>();
map.put("key1","value1");
map.put("key2","value2");
map.put("key3","value3");
person.setMap(map);
//使用一个域对象来装person,"p"可以任意指定
pageContext.setAttribute("p", person);
%>
输出Person:${ p }<br/>
输出Person 的name 属性:${p.name} <br>
输出Person 的pnones 数组属性值:${p.phones[2]} <br>
输出Person 的cities 集合中的元素值:${p.cities} <br>
输出Person 的List 集合中个别元素值:${p.cities[2]} <br>
输出Person 的Map 集合: ${p.map} <br>
输出Person 的Map 集合中某个key 的值: ${p.map.key3} <br>
输出Person 的age 属性:${p.age} <br>
</body>
说明:
① 输入数组名,会输出数组的地址,想输出数组元素的值,加[ ]
② 直接输出List集合的名字,输出所有元素,想输出某个元素,加[ ]
③ 输出Map集合名字,输出整个Map集合,想输出某个key的值,p.map.key
注意:EL表达式底层源码中,不是直接读Bean组件的属性,而是找Bean组件的get()方法。所以,如果对应的属性没有get方法,则会报错。
1.4 EL表达式——运算
语法:
${ 运算表达式}
EL表达式支持如下运算符
(1)关系运算
关系运算符 | 说明 | 范例 | 结果 |
== 或eq | 等于 | ${ 5 == 5 }或${ 5 eq 5 } | true |
!= 或ne | 不等于 | ${ 5 !=5 } 或${ 5 ne 5 } | false |
< 或lt | 小于 | ${ 3 < 5 } 或${ 3 lt 5 } | true |
> 或gt | 大于 | ${ 2 > 10 } 或${ 2 gt 10 } | false |
<= 或le | 小于等于 | ${ 5 <= 12 } 或${ 5 le 12 } | true |
>= 或ge | 大于等于 | ${ 3 >= 5 } 或${ 3 ge 5 } | false |
(2)逻辑运算
逻辑运算符 | 说明 | 范例 | 结果 |
&& 或and | 与运算 | ${ 12 == 12 && 12 < 11 } 或${ 12 == 12 and 12 < 11 } | false |
|| 或or | 或运算 | ${ 12 == 12 || 12 < 11 } 或${ 12 == 12 or 12 < 11 } | true |
! 或not | 取反运算 | ${ !true } 或${not true } | false |
(3)算数运算
算数运算符 | 说明 | 范例 | 结果 |
+ | 加法 | ${ 12 + 18 } | 30 |
- | 减法 | ${ 18 - 8 } | 10 |
* | 乘法 | ${ 12 * 12 } | 144 |
/ 或div | 除法 | ${ 144 / 12 } 或${ 144 div 12 } | 12 |
% 或mod | 取模 | ${ 144 % 10 } 或${ 144 mod 10 } | 4 |
(4)empty运算
empty 运算可以判断一个数据是否为空,如果为空,则输出true,不为空输出false。
以下几种情况为空:
① 值为null 值的时候,为空
② 值为空串的时候,为空
③ 值是Object 类型数组,长度为零的时候
④ list 集合,元素个数为零
⑤ map 集合,元素个数为零
<body>
<%
// 1、值为null 值的时候,为空
request.setAttribute("emptyNull", null);
// 2、值为空串的时候,为空
request.setAttribute("emptyStr", "");
// 3、值是Object 类型数组,长度为零的时候
request.setAttribute("emptyArr", new Object[]{});
// 4、list 集合,元素个数为零
List<String> list = new ArrayList<>();
// list.add("abc"); 添加以后,就false了
request.setAttribute("emptyList", list);
// 5、map 集合,元素个数为零
Map<String,Object> map = new HashMap<String, Object>();
// map.put("key1", "value1"); 添加以后就false了
request.setAttribute("emptyMap", map);
%>//以下输出都为true
${ empty emptyNull } <br/>
${ empty emptyStr } <br/>
${ empty emptyArr } <br/>
${ empty emptyList } <br/>
${ empty emptyMap } <br/>
</body>
(5)三元运算
语法:
表达式1?表达式2:表达式3
说明:如果表达式1 的值为真,返回表达式2 的值,如果表达式1 的值为假,返回表达式3 的值。
示例:
${ 12 != 12 ? "加拿大炮王好帅":"加拿大炮王又骗人啦" }
(6)“.”点运算和 [ ] 中括号运算符
① .点运算,可以输出Bean 对象中某个属性的值。
② [ ]中括号运算,可以输出有序集合中某个元素的值,并且[]中括号运算,还可以输出map 集合中key 里含有特殊字符的key 的值。
说明:如果Key里面有类似 . + - 等特殊字符,点运算符输出属性,就会产生歧义,电脑不懂。因此用[]输出这种属性值。
<body>
<%
Map<String,Object> map = new HashMap<String, Object>();//向map中添加这种 带有 . + - 特殊字符的,后面用.运算没法输出
map.put("a.a.a", "aaaValue");
map.put("b+b+b", "bbbValue");
map.put("c-c-c", "cccValue");
request.setAttribute("map", map);
%>//[ ] 运算符,里面用单引号或者双引号,可以输出值
${ map['a.a.a'] } <br>
${ map["b+b+b"] } <br>
${ map['c-c-c'] } <br>
</body>
1.5 EL表达式的11个隐含对象
EL 个达式中11 个隐含对象,是EL 表达式中自己定义的,可以直接使用。
变量 | 类型 | 作用 | 说明 |
pageContext | PageContextImpl | 它可以获取jsp 中的九大内置对象 | 比写jsp(java)代码简洁 |
pageScope | Map<String,Object> | 它可以获取pageContext 域中的数据 | pageContext 域 |
requestScope | Map<String,Object> | 它可以获取Request 域中的数据 | Request 域 |
sessionScope | Map<String,Object> | 它可以获取Session 域中的数据 | Session 域 |
applicationScope | Map<String,Object> | 它可以获取ServletContext 域中的数据 | ServletContext 域 |
param | Map<String,String> | 它可以获取请求参数的值 | 客户端(浏览器)发来的参数 |
paramValues | Map<String,String[]> | 它也可以获取请求参数的值,获取多个值的时候使用。 | |
header | Map<String,String> | 它可以获取请求头的信息 | |
headerValues | Map<String,String[]> | 它可以获取请求头的信息,它可以获取多个值的情况 | |
cookie | Map<String,Cookie> | 它可以获取当前请求的Cookie 信息 | |
initParam | Map<String,String> | 它可以获取在web.xml 中配置的<context-param>上下文参数 |
(1)EL表达式获取4个特定域中的属性
上图蓝色文字代表的就是四个域
<body>
<%
pageContext.setAttribute("key1", "pageContext1");
pageContext.setAttribute("key2", "pageContext2");
request.setAttribute("key2", "request");
session.setAttribute("key2", "session");
application.setAttribute("key2", "application");
%>
${ applicationScope.key2 } //如果直接$(key2),就会按照从小到大范围获取属性值
</body>
(2)pageContext对象的使用
EL表达式只是让程序员写的更爽,JSP表达式也可以写的。
pageContext对象可以获取以下数据:
序号 | 参数 | 说明 |
1 | 协议 | |
2 | 服务器ip | |
3 | 服务器端口 | |
4 | 获取工程路径 | |
5 | 获取请求方法 | |
6 | 获取客户端ip 地址 | |
7 | 获取会话的id 编号 |
<body>
<%--//以下是java代码获取这些的写法
request.getScheme() 它可以获取请求的协议
request.getServerName() 获取请求的服务器ip 或域名
request.getServerPort() 获取请求的服务器端口号
getContextPath() 获取当前工程路径
request.getMethod() 获取请求的方式(GET 或POST)
request.getRemoteHost() 获取客户端的ip 地址
session.getId() 获取会话的唯一标识
--%>
<%//先将requst对象设置成req,就可以在EL表达式中开启简便写法
pageContext.setAttribute("req", request);
%>
<%=request.getScheme() %> <br>//特别注意:虽然${}表达式里面后面直接跟的是要获取的属性,实际上,底层还是在调用获取该属性的get方法
1.协议: ${ req.scheme }<br> //这句开启了简单模式,其他还是完整写的。
2.服务器ip:${ pageContext.request.serverName }<br>
3.服务器端口:${ pageContext.request.serverPort }<br>
4.获取工程路径:${ pageContext.request.contextPath }<br>
5.获取请求方法:${ pageContext.request.method }<br>
6.获取客户端ip 地址:${ pageContext.request.remoteHost }<br>
7.获取会话的id 编号:${ pageContext.session.id }<br>
</body>
(3)param对象和paramValues对象
param | Map<String,String> | 它可以获取请求参数的值 | 客户端(浏览器)发来的参数 |
paramValues | Map<String,String[]> | 它也可以获取请求参数的值,获取多个值的时候使用。 |
exp:
浏览器输入清流地址:
http://localhost:8080/09_EL_JSTL/other_el_obj.jsp?username=wzg168&password=666666&hobby=java&hobby=cpp(?后面是请求资源路径)
用EL表达式内置对象param和paramValues获取该数据
输出请求参数username 的值:${ param.username } <br>
输出请求参数password 的值:${ param.password } <br>
输出请求参数username 的值:${ paramValues.username[0] } <br>
输出请求参数hobby 的值:${ paramValues.hobby[0] } <br>
输出请求参数hobby 的值:${ paramValues.hobby[1] } <br>
(4)header 和 handerValues 用于获取请求头的信息
header | Map<String,String> | 它可以获取请求头的信息 | |
headerValues | Map<String,String[]> | 它可以获取请求头的信息,它可以获取多个值的情况 |
请求头有一堆信息,那些信息的属性名可以进浏览器,进入F12调试模式,用浏览器来看
//注意:由于请求头的名字,很多有特殊字符,所以用[ ] ,不用.
输出请求头【User-Agent】的值:${ header['User-Agent'] } <br>
输出请求头【Connection】的值:${ header.Connection } <br>
输出请求头【User-Agent】的值:${ headerValues['User-Agent'][0] } <br>
(5)cookie,获取cookie
cookie还没学,暂时还不懂,但是他的键只有JSESSIONID
cookie | Map<String,Cookie> | 它可以获取当前请求的Cookie 信息 |
获取Cookie 的名称:${ cookie.JSESSIONID.name } <br>
获取Cookie 的值:${ cookie.JSESSIONID.value } <br>
(6)initParam,获取web.xml中配置的参数
initParam | Map<String,String> | 它可以获取在web.xml 中配置的<context-param>上下文参数 |
首先在web.xml中配置<context-param>,然后用initParam来获取
<context-param>
<param-name>username</param-name>
<param-value>root</param-value>
</context-param>
<context-param>
<param-name>url</param-name>
<param-value>jdbc:mysql:///test</param-value>
</context-param>
获取代码:
输出<Context-param>username 的值:${ initParam.username } <br>
输出<Context-param>url 的值:${ initParam.url } <br>
2、JSTL标签库(简单的说,就是用标签写代码)
2.1 简介
JSTL 标签库全称是指JSP Standard Tag Library JSP 标准标签库。是一个不断完善的开放源代码的JSP 标签库。
EL 表达式主要是为了替换jsp中的表达式脚本,而标签库则是为了替换代码脚本。这样使得整个jsp 页面变得更佳简洁。
JSTL由五个不同功能的标签库组成
功能范围 | URI | 前缀 |
核心标签库——重点 | http://java.sun.com/jsp/jstl/core | c |
格式化 | http://java.sun.com/jsp/jstl/fmt | fmt |
函数 | http://java.sun.com/jsp/jstl/functions | fn |
数据库(不使用) | http://java.sun.com/jsp/jstl/sql | sql |
XML(不使用) | http://java.sun.com/jsp/jstl/xml | x |
2.2 标签库的使用
step1 先导入jstl 标签库的jar 包。
taglibs-standard-impl-1.2.1.jar
taglibs-standard-spec-1.2.1.jar
step2 使用taglib 指令引入标签库。
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
扩展:标签库大全
1、CORE 标签库
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
2、XML 标签库
<%@ taglib prefix="x" uri="http://java.sun.com/jsp/jstl/xml" %>
3、FMT 标签库
<%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %>
4、SQL 标签库
<%@ taglib prefix="sql" uri="http://java.sun.com/jsp/jstl/sql" %>
5、FUNCTIONS 标签库
<%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions" %>
2.3 core核心库
2.3.1 <c:set />(使用很少)
作用:set 标签可以往域中保存数据(回顾四大域)
<%--
<c:set />
//java代码版的写法
域对象.setAttribute(key,value);
scope 属性设置保存到哪个域
page 表示PageContext 域(默认值)
request 表示Request 域
session 表示Session 域
application 表示ServletContext 域
var 属性设置key 是多少
value 属性设置值
--%>
保存之前:${ sessionScope.abc } <br> //啥也没有
<c:set scope="session" var="abc" value="abcValue"/>
保存之后:${ sessionScope.abc } <br> //有了avcValue
2.3.2 <c:if />
if 标签用来做if 判断。
<%--
ii.<c:if />
if 标签用来做if 判断。
test 属性表示判断的条件(属性值使用EL 表达式输出)如果EL表达式为true,则输出标签的内容
--%>
<c:if test="${ 12 == 12 }">
<h1>12 等于12</h1> //此时h1标签会输出
</c:if>
<c:if test="${ 12 != 12 }">
<h1>12 不等于12</h1> //此时h1标签不会输出
</c:if>
2.3.3 <c:choose> <c:when> <c:otherwise>标签
作用:多路判断。
说明:跟if...else if....else 非常接近
用法:标签间的父子关系看空格
choose 标签开始选择判断
when 标签表示每一种判断情况
test 属性表示当前这种判断情况的值
otherwise 标签表示剩下的情况
注意事项:
(1)标签里不能使用html 注释,要使用jsp 注释
(2)when 标签的父标签一定要是choose 标签(嵌套多留判断时需要特别注意)
exp:
<%
request.setAttribute("height", 180);
%>
<c:choose>//条件1
<c:when test="${ requestScope.height > 190 }">
<h2>小巨人</h2>
</c:when>//条件2
<c:when test="${ requestScope.height > 180 }">
<h2>很高</h2>
</c:when>//条件3
<c:when test="${ requestScope.height > 170 }">
<h2>还可以</h2>
</c:when>//如果都不符合,就otherwise
<c:otherwise>
<c:choose> //继续嵌套多重判断//嵌套条件1
<c:when test="${requestScope.height > 160}">
<h3>大于160</h3>
</c:when>//嵌套条件2
<c:when test="${requestScope.height > 150}">
<h3>大于150</h3>
</c:when>
//嵌套的否则
<c:otherwise>
其他小于140
</c:otherwise> //结束嵌套否则
</c:choose>//结束嵌套多重判断
</c:otherwise>//结束最外部否则
</c:choose>
2.3.4 <c:forEach />
作用:遍历输出使用。
(1)遍历1到10
<%--说明:遍历1 到10,输出
begin 属性设置开始的索引
end 属性设置结束的索引
var 属性表示循环的变量(也是当前正在遍历到的数据)
想象:for (int i = 1; i < 10; i++)
--%>
<table border="1">
<c:forEach begin="1" end="10" var="i">
<tr>
<td>第${i}行</td>
</tr>
</c:forEach>
</table>
(2)遍历Object 数组
<%-- 说明:遍历Object 数组
想象:for (Object item: arr)
items 表示遍历的数据源(遍历的集合)
var 表示当前遍历到的数据给var变量,名字随便取
--%>
<% //注意代码中request对象,和 EL表达式中内置的requestScope对象长得不一样
request.setAttribute("arr", new String[]{"18610541354","18688886666","18699998888"});
%>
<c:forEach items="${ requestScope.arr }" var="item">
${ item } <br>
</c:forEach>
(3)遍历Map集合
<%-- 说明:遍历Object 数组
想象:for (Object item: arr)
items 表示遍历的数据源(遍历的集合)
var 表示当前遍历到的数据给var变量,名字随便取
--%>
<% //先创建一个map对象,并装入数据Map<String,Object> map = new HashMap<String, Object>();
map.put("key1", "value1");
map.put("key2", "value2");
map.put("key3", "value3");//注意代码中request对象,和 EL表达式中内置的requestScope对象长得不一样
request.setAttribute("map", map);
%>
<c:forEach items="${ requestScope.map }" var="entry">
${ entry } <br> //此时遍历按顺序输出key1=value1${entry.key} = ${entry.value} //也是输出key1=value1 ,只是说明,也可以单独分别取出key 和 value
</c:forEach>
(4)遍历List 集合---list 中存放Student 类,有属性:编号,用户名,密码,年龄,电话信息
Student类
public class Student{
//需要编号,用户名,密码,年龄,电话信息
private Integer id;
private String username;
private String password;
private Integer age;
private String phone;
//Bean组件标配,空参构造器,带参构造器,toString,get,set方法
public Student(Integer id, String username,String password,Integer phone){
this.id=id;
this.username = username;
//等等等等
}
public Student(){
}
}
<%--遍历List 集合---list 中存放Student 类,有属性:编号,用户名,密码,年龄,电话信息--%>
<%
List<Student> studentList = new ArrayList<Student>();
for (int i = 1; i <= 10; i++) {
studentList.add(new Student(i,"username"+i ,"pass"+i,18+i,"phone"+i)); //循环创建10个Student对象,并将其添加进集合
}
request.setAttribute("stus", studentList); //添加到request域中去
%>//创建一个表格,用于存放遍历到的数据
<table>
<tr>
<th>编号</th>
<th>用户名</th>
<th>密码</th>
<th>年龄</th>
<th>电话</th>
<th>操作</th>
</tr><%--说明:
items 表示遍历的集合
var 表示遍历到的数据
begin 表示遍历的开始索引值(注意索引值的意思)
end 表示结束的索引值
step 属性表示遍历的步长值
varStatus 属性表示当前遍历到的数据的状态
想象:for(int i = 1; i < 10; i+=2)
--%>
<c:forEach begin="2" end="7" step="2" varStatus="status" items="${requestScope.stus}" var="stu">
<tr>
<td>${stu.id}</td>
<td>${stu.username}</td>
<td>${stu.password}</td>
<td>${stu.age}</td>
<td>${stu.phone}</td>
<td>${status.step}</td>
</tr>
</c:forEach>
</table>
扩展:关于varStatus,该属性的值的Status类继承了LoopTagStatus接口,可以获取以下状态(了解即可)