JSP
一、 介绍
JSP: Java Server Page SUN 公司提供的动态网页编程技术,是 Java Web 服
务器端的动态资源。
它相比 html 而言, html 只能为用户提供静态数据,而 Jsp 技术允许在页
面中嵌套 java 代码,为用户提供动态数据。
相比 servlet 而言, servlet 很难对数据进行排版,而 jsp 除了可以用 java
代码产生动态数据的同时,也很容易对数据进行排版。
不管是 JSP 还是 Servlet,虽然都可以用于开发动态 web 资源。但由于这 2
门技术各自的特点,在长期的软件实践中,人们逐渐把 servlet 作为 web 应用
中的控制器组件来使用, 而把 JSP 技术作为数据显示模板来使用。
其实 Jsp 就是一个 Servlet,当我们第一次访问 Jsp 的时候, Jsp 引擎都会
将这个 Jsp 翻译 成一个 Servlet,这个文件存放在 Tomcat 中的 work 目录中。

二、 准备工作
**1. 配置 Eclipse
**这一步不是必须的,当然由于 eclipse 中有写默认的配置项我们觉得不是很
完美,比如 字符编码,比如 HTML 的 DOCTYPE,我们可以在新建 JSP 页面之
前就先修改为我们需要的。


2. 新建 JSP 文件

新建完成以后的完整代码如下

三、 基础语法
1. 注释
在 JSP 中支持两种注释的语法操作,一种是显示注释,这种注释是允许客
户端看见的; 另一种是隐式注释,此种注释是客户端无法看见的
① 显示注释语法:从 HTML 风格继承而来
② 隐式注释语法:从 JAVA 风格继承; JSP 自己的注释
JSP 的三种注释方式:
1) // 注释,单行注释 /* 多行注释*/
2)
3) <%-- JSP 注释 --%>
**2. Scriptlet
**在 JSP 中最重要的部分就是 Scriptlet(脚本小程序), 所有嵌入在
HTML 代
码中的 Java 程序都必须使用 Scriptlet 标记出来,在 JSP
中一共有三种
Scriptlet 代码:
第一种: <% %>: java 脚本段,可以定义局部变量、编写语句
第二种: <%!
%>:声明,可以定义全局(成员)变量、方法、类,生成的代码在servlet类体中
第三种: <%=
%>:表达式,数据一个变量或具体内容,生成的代码在servlet的service方法体中
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<!DOCTYPE html >
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Script脚本小程序</title>
</head>
<body>
<pre>
Scriptlet(脚本小程序),所有嵌入在 HTML 代码中的 Java 程序都必须使用 Scriptlet 标记出来.
三种Scriptlet代码:
<%--
<% // java脚本段 %>
--%>
1、<% %>
java脚本段,生成的代码在servlet的service方法体中,变量是局部变量
<%--
<%! // 声明全局变量,方法、类等 %>
--%>
2、<%! %>>
声明全局变量、方法、类等,生成的代码在servlet类体中
<%--
<%=变量名、字面量 %>
--%>
3、<%= %>
表达式,输出表达式,可以输出变量名、字面量,生成的代码在servlet的service方法体中
</pre>
<!--java脚本段 -->
<%
int num = 1;
System.out.println(num);
out.print(num);
out.print(str);
request.setAttribute("abc", "ABC");
%>
<%!
int a = 10;
String str = "Hello";
%>
<%=a%> <br>
<%=100%><br>
<%=request.getAttribute("abc")%><br>
</body>
</html>
3. JSP 的指令标签
JSP 的指令标签用来控制整个 jsp 的行为,就当前 JSP 页面的某些方面对
JSP 转换器作出指示。

1) page 指令
掌握使用 page 指令常用的属性 contentType 和 pageEncoding,分别用来
指定 MIME 类型 和编码格式。import 属性,用来引入其他包,和 java 中
import语句作用相同。
language:支持的语言,目前只有java
contentType:响应类型及编码格式
pageEncoding:字符编码
import:导包,只有该属性可以同时设置多次
errorPage:当页面报错时要跳转的页面,一般与isErrorPage属性一起使用,且跳转的页面的isErrorPage属性设置为true
isErrorPage:是否是处理错误的页面,默认是false。一般与errorPage属性一起使用
注:页面的编码以charset的值为准;若charset未设置,则以pageEncoding为准;若两者都未设置,则使用默认编码ISO-8859-1。
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"
import="java.util.List" import="java.util.ArrayList" isErrorPage="false" %>
<!DOCTYPE html >
2) include 静态包含
<%@ include file="要包含的文件路径"%>相对路径
<%@include file="include.jsp"%>
或
<%@include file="include.html"%>
静态包含就是将内容进行了直接的替换,就好比程序中定义的变量一样 是
在 servlet 引擎转译时,就把此文件内容包含了进去(两个文件的源代码整合到
一起, 全部放到_jspService 方法中),所以只生成了一个
servlet,所以两个页
面不能有同名的变量。 运行效率高一点点。耦合性较高,不够灵活。
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<!DOCTYPE html >
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>include静态包含</title>
</head>
<body>
<%--
include静态包含
格式:<%@include file="文件路径" %> 相对路径
file属性支持表达式去取值
特点:
只会生成一个源码文件,相当于直接将内容拷贝进来,所以不能有同名变量。
运行效率高一点点。耦合性较高,不够灵活。
--%>
<div style="height: 800px;width:800px">
<div id="left" style="width: 200px;height:800px;background-color: pink;float: left;">
<%@include file="left.jsp" %>
</div>
<div id="right" style="width: 600px;height:800px;background-color: gray;float: left;">
<%@include file="right.jsp" %>
</div>
</div>
<%
// int a = 1;
%>
</body>
</html>
4). JSP 动作标签
动作标签,它们被编译成执行某个操作的 Java 代码,例如访问某个 Java 对
象,或者调用某个方法。 JSP
容器有自己支持的标准动作。除标准动作之外,还可以创建指定的标签,
用来执行某些操作。这些 jsp 的动作标签,与 html 提供的标签有本质的区别:
* 动作标签是由 tomcat(服务器)来解释执行!与 java
代码一样,都是在服务器
端执行
* html 由浏览器来执行!
include 动态包含
include 动作用于动态地包含另一个资源,它可以包含另一个 JSP 页面、一
个 Servlet 或 者一个静态的 HTML 页面。
使用包含操作,可以将一些重复的代码包含进来继续使用,从正常的页面组
成来看,有时可能分为几个区域。而其中的一些区域可能是一直不需要改变的,
改变的就其中的一个具体内容区域。现在有两种方法可以实现上述功能。
方法一:在每个 JSP 页面(HTML)都包含工具栏、头部信息、尾部信息、
具体内容
方法二:将工具栏、头部信息、尾部信息都分成各个独立的文件,以后使用
的时候直接 导入进来
很明显,第二种方法比第一种更好,第一种会存在很多重复的代码,并且修
改很不方便, 在 JSP 中如果要想实现包含的操作,有两种做法: 静态包含、
动
态包含,静态包含使用 include 指令即可,动态包含则需要使用 include
动作标
签。
动态包含在代码的编译阶段,包含和被包含部分是两个独立的部分,只有当
运行时,才会动态包含进来,好比方法的调用。
--
<jsp:include page="include.jsp"></jsp:include>
注意:动态包含,中间不要加任何内容,包括空格,除非确认要使用参数,否则报错!
<jsp:include page="include.html"></jsp:include>
<%
String a = "hello.jsp";
%>
<jsp:include page="<%=a%>"></jsp:include>
使用动态包含还可以通过在页面之间传参。接收参数通过
request.getParameter(name)来完成
jsp:include page="hello.jsp" flush="true">
<jsp:param name="school" value="shsxt"/>
</jsp:include>
<%=request.getParameter("school")%>
注意:有些对象在 JSP 中能够直接使用,可以通过观察.java 代码知道。目
前我们只需要知道 request,response,session,application 可以直接使用即可。
特点:
生成多个源码文件,相当于方法的调用,可以有同名变量。
耦合度低,比较灵活,效率高。
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<!DOCTYPE html >
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>include动态包含</title>
</head>
<body>
<%--
include动态包含
语法:<jsp:include page="要包含的文件路径"></jsp:include>
page属性支持表达式去取值
特点:
生成多个源码文件,相当于方法的调用,可以有同名变量。
耦合度低,比较灵活,效率高。
注:
1、<jsp:include>双标签之间不能有任何内容(除非有参数)
Expecting "jsp:param" standard action with "name" and "value" attributes
2、动态包含可以传递参数
<jsp:param name="参数名" value="参数值"/>
获取参数使用:request.getParameter(name)
value属性支持表达式,name属性不支持
The name attribute of the jsp:param standard action does not accept any expressions
--%>
<%
int a =10;
String key = "userName";
String val = "admin";
%>
<div style="height: 800px;width:800px">
<div id="left" style="width: 200px;height:800px;background-color: blue;float: left;">
<jsp:include page="left.jsp"></jsp:include>
<jsp:include page="left.jsp">
<jsp:param value="<%=val %>" name="uname"/>
</jsp:include>
</div>
<div id="right" style="width: 600px;height:800px;background-color: green;float: left;">
<jsp:include page="right.jsp"></jsp:include>
</div>
</div>
</body>
</html>
5. 九大内置对象
通过解析后的 jsp 文件查看即可
***1.request 对象
***request 对象是 javax.servlet.httpServletRequest 类型的对象。
该对象代表了客户端的请求信息,主要用于接受通过 HTTP
协议传送到服务器的数据。
(包括头信息. 系统信息. 请求方式以及请求参数等)。 request
对象的作用域为一次请求。
***2.response 对象
***response 代表的是对客户端的响应,主要是将 JSP 容器处理过的对象传回
到客户端。 response 对象也具有作用域,它只在 JSP 页面内有效。
3.session 对象\
session 对象是由服务器自动创建的与用户请求相关的对象。服务器为每个
用户都生成一个 session
对象,用于保存该用户的信息,跟踪用户的操作状态。session 对象内部使用 Map
类来保存数据,因此保存数据的格式为
"Key/value"。 session 对象的 value
可以是复杂的对象类型,而不仅仅局限于字
符串类型。
4.application 对象
application 对象可将信息保存在服务器中,直到服务器关闭,否则
application 对象中保存的信息会在整个应用中都有效。
与 session 对象相比, application 对象生命周期更长,类似于系统的"全局
变量"。
5.out 对象
out 对象用于在 Web 浏览器内输出信息,并且管理应用服务器上的输出缓
冲区。在使用 out 对象输出数据时,可以对数据缓冲区进行操作,及时清除缓冲
区中的残余数据,为其他的输出让出缓冲空间。待数据输出完毕后,要及时关闭输
出流。
6.pageContext 对象
pageContext 对象的作用是取得任何范围的参数,通过它可以获取 JSP
页面的 out. request. reponse. session. application 等对象。
pageContext 对象的创建和初始化都是由容器来完成的,在 JSP 页面中可
以直接使用 pageContext 对象。
7.config 对象
config 对象的主要作用是取得服务器的配置信息。通过 pageConext 对象
的 getServletConfig() 方法可以获取一个 config 对象。
当一个 Servlet 初始化时,容器把某些信息通过 config 对象传递给这个
Servlet。
开发者可以在 web.xml 文件中为应用程序环境中的 Servlet 程序和 JSP 页
面提供初始化参数。
8.page 对象
page 对象代表 JSP 本身,只有在 JSP 页面内才是合法的。 page 隐含对
象本质上包含当前 Servlet 接口引用的变量,类似于 Java 编程中的 this
指针。
9.exception 对象
exception 对象的作用是显示异常信息,只有在包含 isErrorPage="true"
的页面中才可以被使用,在一般的 JSP 页面中使用该对象将无法编译 JSP
文件。
excepation 对象和 Java 的所有对象一样,都具有系统提供的继承结构。
exception 对象几乎定义了所有异常情况。
如果在 JSP 页面中出现没有捕获到的异常,就会生成 exception 对象,并
把 exception 对象传送到在 page 指令中设定的错误页面中,然后在错误页面中
处理相应的 exception 对象。
四、 EL 表达式
1. EL 语法和使用
EL(Expression Language) 是为了使 JSP 写起来更加简单。表达式语言的
灵感来自于 ECMAScript 和 XPath 表达式语言,它提供了在 JSP 中简化表达式
的方法,让 Jsp 的代码更 加简化。
语法结构非常简单: ${expressio}
EL 表达式一般操作的都是域对象中的数据,操作不了局部变量。域对象的
概念在 JSP 中一共有四个: pageContext, request, session,
application;范围依次是,本页面,一次请求, 一次会话,整个应用程序。而
EL 默认的查找方式为从小到大查找,找到即可。当域对象全
找完了还未找到则返回空字符串""。
当需要指定从某个特定的域对象中查找数据时可以使用四个域对象对应的
空间对象,分别是: pageScope, requestScope, sessionScope,
applicationScope。
2. EL 运算
使用 EL 表达式可以用来判断集合中是否有元素${empty list}
<%@page import="java.util.HashMap"%>
<%@page import="java.util.Map"%>
<%@page import="java.util.ArrayList"%>
<%@page import="java.util.List"%>
<%@page import="com.shsxt.model.User"%>
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<!DOCTYPE html >
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>EL表达式</title>
</head>
<body>
<%
// 设置域对象的值
pageContext.setAttribute("uname", "zhangsan");
request.setAttribute("uname", "lisi");
session.setAttribute("uname", "wangwu");
application.setAttribute("uname", "zhaoliu");
String str = "admin";
// 将对象存到域对象中
User user = new User();
user.setUserId(1);
user.setUserName("Lisa");
user.setUserPwd("123123");
request.setAttribute("user", user);
// 将集合存到域对象中
User user2 = new User();
user2.setUserId(2);
user2.setUserName("Lily");
user2.setUserPwd("123456");
List<String> list = new ArrayList<>();
list.add("aaa");
list.add("bbb");
list.add("ccc");
List<User> userList = new ArrayList<>();
userList.add(user);
userList.add(user2);
List<String> list2 = new ArrayList<>();
List<String> list3 = null;
request.setAttribute("mylist", list);
request.setAttribute("userList", userList);
request.setAttribute("list2", list2);
request.setAttribute("list3", list3);
Map map = new HashMap();
map.put("aaa", "111");
map.put("bbb", 2222);
map.put("ccc-a", 333);
request.setAttribute("map", map);
request.setAttribute("num", 10);
request.setAttribute("num2", 2);
request.setAttribute("str1", "aa");
request.setAttribute("str2", "bb");
pageContext.setAttribute("a", "aa", PageContext.REQUEST_SCOPE);
pageContext.getAttribute("a", PageContext.REQUEST_SCOPE);
%>
<%=request.getAttribute("uname1") %>
<hr>
<%--
EL表达式
语法:${key } key代表的是存在域对象中的键
注意:el表达式一般操作的是域对象,无法操作变量
取值范围:
从小范围到大范围取,取到为止,若未取到,则一直向大返回查找;若四个范围都未获取到,返回空字符串。
获取指定范围的值:
pageScope、requestScope、sessionScope、applicationScope
pageContext, request, session, application;范围依次是,本页面,一次请求, 一次会话,整个应用程序。
而 EL 默认的查找方式为 从小到大查找,找到即可。当域对象全 找完了还未找到则返回空字符串””。
EL操作对象
获取对象中的属性:${对象.属性名}
注:属性必须提供get和set方法
--%>
<br>
<h2>EL的基本使用</h2>
uname:${uname } <br>
uname1: ${uname1 }<br>
requestScope:${requestScope.uname } <br>
str:${str }<br>
<br><br>
<h2>EL 对象取值</h2>
user对象:${user }<br>
获取user对象中的属性:${user.userName }<br>
<h2>EL 集合取值</h2>
获取集合:${mylist } ---- ${userList }<br>
获取指定下标的值: ${mylist[0] } ---- ${userList[0].userPwd }<br>
获取集合的长度:${userList.size() }
<h2>判断为空empty</h2>
判断字符串是否为空:${empty uname } --- ${empty uname1 } --- ${!empty uname }<br>
判断对象是否为空:${empty user }<br>
判断集合是否为空:${empty userList } --- ${empty list2 } --- ${empty list3 }<br>
<h2>EL Map取值</h2>
获取Map的值:${map }<br>
获取指定key的value:${map.aaa } --- ${map["aaa"] }<br>
<h2>EL 运算</h2>
${1+1 } -- ${num+1 } -- ${num + num2 } <br>
${num / num2 } ---
<%-- ${num div num2 } --%>
<br>
<h2>EL 比较</h2>
${num == 1 } -- ${num >= 10 } <br>
${str1 == str2 } -- ${str1 eq str2 }<br>
${num + 10 > 10 }
</body>
</html>
五、 JSP 内置对象总结
掌握内置对象的主要作用
掌握九个内置对象及其对应类型
在 JSP
之中,已经专门为用户的使用提供了许多的内置对象,所谓的内置对象主要是指所有的对象不用由用户自己进行对象的实例化操作,而可以直接进行使用
例如,之前的 request, 有一个接收参数的方法: public String
getParameter(String
paramName),实际上对于 request 本身就是一个内置对象
WEB 容器为用户准备好了这些个内置对象
1) JSP 中有九个内置对象

2) JSP 内置对象四种属性范围(重点)
四种属性范围
在
JSP中提供了四种属性的保存范围,所谓的属性保存范围,指的就是一个设置的对象,可以再多少个页面中保存并可以继续使用
page 范围 pageContext : 只在一个页面中保存属性,跳转之后无效
request 范围 request : 只在一次请求中保存,服务器跳转后依然有效
session 范围 session :
在一次会话范围中,无论何种跳转都可以使用,但是新开浏览器无法使用
application 范围 application : 在整个服务器上保存,所有用户都可以用
属性都有的操作方法

验证各个属性范围的特点
page: 本页面取得服务器端跳转(<jsp :forward>)后无效
request : 服务器跳转(<jsp :forward>)有效,客户端跳转(超链接)无效
如果是客户端跳转,则相当于发出了两次请求,那么第一次的请求将不存在了
如果希望不管是客户端还是服务器跳转,都可以保存的话,就需要继续扩大范围。
session :
无论客户端还是服务器端都可以取得,但是现在重新开启一个新的浏览器,则
无法取得之前设置的 session 了,因为每一个 session
只保存在当前的浏览器当中,并在相关的页面取得。
对于服务器而言,每一个连接到它的客户端都是一个 session
如果想要让属性设置一次之后,不管是否是新的浏览器打开都能取得则可以使用
application
application : 所有的 application
属性直接保存在服务器上,所有的用户(每一个 session)
都可以直接访问取得
只要是通过 application 设置的属性,则所有的 session
都可以取得,表示公共的内容,
但是如果此时服务器重启了,则无法取得了,因为关闭服务器后,所有的属性都消失了,所以需要重新设置。
使用哪个范围呢?在合理范围尽可能小
page 范围
page 范围是通过 pageContext 对象设置的,之前研究过的 page
属性范围中使用的是
pageContext 进行属性设置的,但是从 javax.servlet.jsp.PageContext
类中可以发现,有以下的
一种设置属性的方法:
public void setAttribute(String name, Object value, int scope)
在 PageContext 类中存在四个表示属性范围的常量,可以直接通过这些常量指定
scope:
pageContext.PAGE_SCOPE
pageContext.REQUEST_SCOPE
pageContext.SESSION_SCOPE
pageContext.APPLICATION_SCOPE
在其它页面取值时,需要指定对应的作用域,或者通过 EL 表达式
<%=pageContext.getAttribute("a",pageContext.SESSION_SCOPE)
%>
