1.EL表达式
1.1 概念和作用
- 全称是
Expression Language
,是表达式语言- 代替 jsp 页面中的表达式脚本在 jsp 页面中进行数据的输出(主要是输出域对象中的数据)
- 格式:
${}
<body> <% request.setAttribute("key","值"); %> 表达式脚本输出 key 的值是: <%=request.getAttribute("key")%><br/> // <%=request.getAttribute("key")==null?"":request.getAttribute("key")%><br/> EL 表达式输出 key 的值是:${key} </body>
tips:
- 如果查询的key不存在,则传统方式会在页面输出null,而EL表达式不会在页面输出内容
1.2 搜索域数据的顺序
- 当四个域中都有相同的 key 的数据的时候,EL 表达式会按照四个域的从小到大的顺序去进行搜索,找到就输出:
<% request.setAttribute("key", "request"); session.setAttribute("key", "session"); application.setAttribute("key", "application"); pageContext.setAttribute("key", "pageContext"); %> ${ key } </body> // 页面输出pageContext
tips:
- 四个域对象在使用时的优先顺序按从小到大的范围查找:
pageContext->request->session->application
1.3 输出不同属性的方式
- 假设Person类如下:
public class Person { private String name; private String[] phones; private List<String> cities; private Map<String,Object> map; ... }
在jsp页面中创建了Person类的对象p并且赋予了值,此时要是有EL表达式调用对象的属性:
输出 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.key1} <br>
tips:
- 尽管Person类的属性全是私有,但还是可以直接通过上面形式直接使用属性,因为底层还是调用了Person类的get方法,所以如果某个属性没有创建get方法是无法在EL表达式中使用的
1.4 运算
<body> // 1.关系运算 ${12 == 11} ${12 eq 12} // 2.逻辑运算 ${12 == 11 && 10 eq 10} // 3.算数运算 ${ 12 + 18 } // 4.empty运算:以判断一个数据是否为空 <% request.setAttribute("emptyStr", ""); %> ${ empty emptyStr } <br/> // 5.三元运算 ${ 12 != 12 ? "yes":"no" } // 6.[]运算符() ${ map['key1'] } // 7.点运算 ${p.map} </body>
tips:
- empty运算时值为 null 值的时候是视为空的
1.5 隐含对象
- EL 表达式中自己定义的,可以直接使用:
pageContext
:它可以获取 jsp 中的九大内置对象pageScope
:它可以获取 pageContext 域中的数据requestScope
: 它可以获取 Request 域中的数据sessionScope
: 它可以获取 Session 域中的数据applicationScope
: 它可以获取 ServletContext 域中的数据param
: 它可以获取请求参数的值paramValues
: 它也可以获取请求参数的值,获取多个值的时候使用。header
: 它可以获取请求头的信息headerValues
: 它可以获取请求头的信息,它可以获取多个值的情况cookie
: 它可以获取当前请求的 Cookie 信息initParam
:它可以获取在 web.xml 中配置的上下文参数
1.5.1 获取四个特定域中的属性
<body> <% pageContext.setAttribute("key1", "pageContext1"); pageContext.setAttribute("key2", "pageContext2"); request.setAttribute("key2", "request"); session.setAttribute("key2", "session"); application.setAttribute("key2", "application"); %> # 输出application,如果直接使用${key2}就不知道是哪个域的对象了,默认输出第一个出现的,即pageContext2 ${ applicationScope.key2 } ${ sessionScope.key2 } # 输出session </body>
1.5.2 pageContext对象的使用
<body> 1.协议: ${ pageContext.request.scheme }<br> # 等价于<%=request.getScheme() %>,该方法获取请求的协议 2.服务器 ip:${ pageContext.request.serverName }<br> 3.服务器端口:${ pageContext.request.serverPort }<br> 4.获取工程路径:${ pageContext.request.contextPath }<br> # 输出/test 5.获取请求方法:${ pageContext.request.method }<br> 6.获取客户端 ip 地址:${ pageContext.request.remoteHost }<br> 7.获取会话的 id 编号:${ pageContext.session.id }<br> </body>
1.5.3 其他隐含对象的使用
# 输入网址http://localhost:8080/test/test.jsp?username=psj&password=666666&hobby=java&hobby=cpp # 在web.xml中配置: #<context-param> # <param-name>username</param-name> # <param-value>root</param-value> #</context-param> <body> 输出请求参数 username 的值:${ param.username } <br> 输出请求参数 password 的值:${ param.password } <br> 输出请求参数 hobby 的值:${ paramValues.hobby[0] } <br> 输出请求参数 hobby 的值:${ paramValues.hobby[1] } <br> 输出请求头【User-Agent】的值:${ header['User-Agent'] } <br> # 如果使用header.User-Agent不会显示内容 输出请求头【Connection】的值:${ header.Connection } <br> 输出请求头【User-Agent】的值:${ headerValues['User-Agent'][0] } <br> 获取 Cookie 的名称:${ cookie.JSESSIONID.name } <br> 获取 Cookie 的值:${ cookie.JSESSIONID.value } <br> 输出<Context-param>username 的值:${ initParam.username } <br> </body>
2.JSTL 标签库
- 全称是
JSP Standard Tag Library
- EL 表达式主要是为了替换jsp中的表达式脚本,而标签库则是为了替换代码脚本,使得整个jsp页面变得更佳简洁
- JSTL 由五个不同功能的标签库组成:
2.1 标签库的使用步骤
- 先导入 jstl 标签库的 jar 包
- 使用 taglib 指令引入标签库
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
2.2 core 核心库使用
<c:set />
:往域中保存数据# scope:属性设置保存到哪个域 # page 表示 PageContext 域(默认值) # request 表示 Request 域 # session 表示 Session 域 # application 表示 ServletContext 域 #var:属性设置 key 是多少 #value:属性设置值 保存之前:${ sessionScope.abc } <br> <c:set scope="session" var="abc" value="abcValue"/> # 相当于[域对象.setAttribute(key,value);] 保存之后:${ sessionScope.abc } <br> # 输出abcValue
<c:if />
:做 if 判断# if:标签用来做 if 判断。 # test:属性表示判断的条件(使用 EL 表达式输出) <c:if test="${ 12 == 12 }"> <h1>12 等于 12</h1> </c:if> <c:if test="${ 12 != 12 }"> <h1>12 不等于 12</h1> </c:if>
<c:choose> <c:when> <c:otherwise>
:多路判断,跟 switch … case … default 非常接近# choose 标签开始选择判断 # when 标签表示每一种判断情况 # test 属性表示当前这种判断情况的值 # otherwise 标签表示剩下的情况 <% request.setAttribute("height", 180); %> <c:choose> <c:when test="${ requestScope.height > 190 }"> <h2>小巨人</h2> </c:when> <c:when test="${ requestScope.height > 180 }"> <h2>很高</h2> </c:when> <c:when test="${ requestScope.height > 170 }"> <h2>还可以</h2> </c:when> <c:otherwise> <c:choose> # when标签的父标签一定要是choose 标签,不能不写choose标签直接将when标签写在choose标签后 <c:when test="${requestScope.height > 160}"> <h3>大于 160</h3> </c:when> <c:when test="${requestScope.height > 150}"> <h3>大于 150</h3> </c:when> <c:when test="${requestScope.height > 140}"> <h3>大于 140</h3> </c:when> <c:otherwise> 其他小于 140 </c:otherwise> </c:choose> </c:otherwise> </c:choose>
<c:forEach />
:遍历输出# begin:属性设置开始的索引 # end:属性设置结束的索引 # var:属性表示循环的变量(也是当前正在遍历到的数据) # items:表示遍历的数据源(遍历的集合) # step:属性表示遍历的步长值 # varStatus:属性表示当前遍历到的数据的状态 # 1.遍历数字 <table border="1"> <c:forEach begin="1" end="10" var="i"> <tr> <td>第${i}行</td> </tr> </c:forEach> </table> # 2.遍历对象 <% request.setAttribute("arr", new String[]{"18610541354","18688886666","18699998888"}); %> <c:forEach items="${ requestScope.arr }" var="item"> ${ item } <br> </c:forEach> # 3.遍历集合 <% Map<String,Object> map = new HashMap<String, Object>(); map.put("key1", "value1"); map.put("key2", "value2"); map.put("key3", "value3"); request.setAttribute("mapTest", map); %> <c:forEach items="${ requestScope.mapTest }" var="entry" varStatus="status"> <h1>${entry.key} = ${entry.value}</h1> <h1>${status.first}</h1> # 表示当前遍历的元素是否为第一个(status还有其他的API) </c:forEach>
tips:
- 在使用多路判断,标签里不能使用 html 注释,要使用 jsp 注释