JSP
1、简介
1.1、创建与访问
JSP 是 Java 的服务器页面, 主要作用是代替 Servlet 程序回传 html 页面的数据
因为 Servlet 程序回传 html 页面数据是一件非常繁锁的事情,开发成本和维护成本都极高
使用 Servlet 回传 html 页面数据
代码示例:
使用 HttpServletResponse 类往客户端回传数据
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//往客户端回传字符串数据
resp.setContentType("text/html; charset=UTF-8");
PrintWriter printWriter = response.getWriter();
printWriter.write("<!DOCTYPE html>\r\n");
printWriter.write(" <html lang=\"en\">\r\n");
printWriter.write(" <head>\r\n");
printWriter.write(" <meta charset=\"UTF-8\">\r\n");
printWriter.write(" <title>Title</title>\r\n");
printWriter.write(" </head>\r\n");
printWriter.write(" <body>\r\n");
printWriter.write(" 这是 html 页面数据 \r\n");
printWriter.write(" </body>\r\n");
printWriter.write("</html>\r\n");
printWriter.write("\r\n");
}
使用 JSP 回传 html 页面数据
代码示例:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
这是 html 页面数据
</body>
</html>
- 如何创建 jsp 的页面?
- jsp 如何访问:
jsp 页面和 html 页面一样,都是存放在 web 目录下,访问也跟访问 html 页面一样- 在 web 目录下有如下的文件:
a.html 页面,访问地址是http://ip:port/工程路径/a.html
b.jsp 页面,访问地址是http://ip:port/工程路径/b.jsp
- 在 web 目录下有如下的文件:
1.2、jsp 的本质
jsp 页面本质上是一个 Servlet 程序
当我们第一次访问 jsp 页面的时候,Tomcat 服务器会帮我们把 jsp 页面翻译成为一个 java 源文件,并且对它进行编译成 为 .class 字节码程序
先访问,然后 Tomcat 会翻译为 java 源文件
我们打开 java 源文件不难发现其里面的内容是:
IDEA 中跟踪原代码发现,HttpJspBase 类直接地继承了 HttpServlet 类。
也就是说,jsp 翻译出来的 java 类,它间接了继承了 HttpServlet 类。
也就是说,翻译出来的是一个 Servlet 程序
总结:
通过翻译的 java 源代码我们就可以得到结果:jsp 就是 Servlet 程序
观察翻译出来的 java 源文件,不难发现,其底层实现也是通过输出流,把 html 页面数据回传给客户端
在方法_jspService()
中使用输出流
2、jsp 的三种语法
2.1、jsp 头部的 page 指令
jsp 的 page 指令可以修改 jsp 页面中一些重要的属性,或者行为
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
属性 | 描述 |
---|---|
language 属性 | 表示 jsp 翻译后是什么语言文件。暂时只支持 java |
contentType 属性 | 表示 jsp 返回的数据类型是什么,也是源码中 response.setContentType()参数值 |
pageEncoding 属性 | 表示当前 jsp 页面文件本身的字符集 |
import 属性 | 跟 java 源代码中一样,用于导包,导类 |
以下两个属性是给 out 输出流使用
属性 | 描述 |
---|---|
autoFlush 属性 | 设置当 out 输出流缓冲区满了之后,是否自动刷新冲级区。默认值是 true |
buffer 属性 | 设置 out 缓冲区的大小。默认是 8kb |
当设置autoFlush="false" buffer="1kb"
的时候,若文件中有大量数据,就会报缓冲区溢出的异常,所以这两个属性一般不轻易更改
属性 | 描述 |
---|---|
errorPage 属性 | 设置当 jsp 页面运行时出错,自动跳转去的错误页面路径 |
isErrorPage 属性 | 设置当前 jsp 页面是否是错误信息页面,默认是 false。如果是 true 可以获取异常信息 |
session 属性 | 设置访问当前 jsp 页面,是否会创建 HttpSession 对象。默认是 true |
extends 属性 | 设置 jsp 翻译出来的 java 类默认继承谁 |
errorPage属性
表示错误后自动跳转去的路径,这个路径一般都是以斜杠打头,它表示请求地址为 http://ip:port/工程路径/
,映射到代码的 Web 目录
isErrorPage 属性
的值设置为 TRUE 时候,能获取异常信息,是因为当值为 TRUE 时,Java源码文件中会出现一个 exception 的对象,而后就可以通过此对象来获取异常信息|
2.2、jsp 中的常用脚本
2.2.1、声明脚本(极少使用)
格式:
<%! 声明 java 代码 %>
作用:
可以给 jsp 翻译出来的 java 源文件中的 a_jsp 类定义属性、方法、静态代码块、内部类等
<%--1、声明类属性--%>
<%!
private Integer id;
private String name;
private static Map<String,Object> map;
%>
<%--2、声明 static 静态代码块--%>
<%!
static {
map = new HashMap<String,Object>();
map.put("key1", "value1");
map.put("key2", "value2");
map.put("key3", "value3");
}
%>
<%--3、声明类方法--%>
<%!
public int abc(){
return 12;
}
%>
<%--4、声明内部类--%>
<%!
public static class A {
private Integer id = 12;
private String abc = "abc";
}
%>
定义的模块都会出现在该 jsp 文件的 java 源文件的类中
声明脚本代码翻译对照:
2.2.2、表达式脚本(常用)
格式:
<%= 表达式 %>
作用:
在 jsp 页面上输出数据,浏览器会获取到回传的数据,同时 java 源文件中也会收到数据
表达式脚本的特点:
- 所有的表达式脚本都会被翻译到 _jspService() 方法中
- 表达式脚本都会被翻译成为 out.print() 输出到页面上
- 由于表达式脚本翻译的内容都在_jspService() 方法中,所以_jspService()方法中的对象都可以直接使用
- 表达式脚本中的表达式不能以分号结束
<%=12 %> <br>
<%=12.12 %> <br>
<%="我是字符串" %> <br>
<%=map%> <br>
<%=request.getParameter("username")%>
翻译对照:
表达式不能以分号结尾,如果用了分号,则在 java 源文件中 out.print 语句就会报错,分号引起语句提前结束
out.print(request.getParameter(“username”));
2.2.3、代码脚本
格式:
<% java 语句 %>
作用:
可以在 jsp 页面中,编写我们自己需要的功能(写的是 java 语句)
代码脚本的特点是
- 代码脚本翻译之后都在 _jspService() 方法中
- 由于代码脚本翻译之后是存到 _jspService() 方法中,所以在_jspService()方法中的现有对象都可以直接使用
- 还可以由多个代码脚本块组合完成一个完整的 java 语句
- 代码脚本可以和表达式脚本一起组合使用,在 jsp 页面上输出数据
翻译之后的对比:
代码脚本翻译之后都在 _jspService()
方法中
<body>
<%
int i = 12;
if( i == 12){
System.out.println("成立!!!");
}else {
System.out.println("不成立");
}
%>
这是b.jsp页面数据
</body>
在 java 源文件中如下:
System.out.println(“成立!!!”); 语句输出是在 控制台输出:
在_jspService()
方法中的现有对象都可以直接使用
<body>
<%
String username = request.getParameter("username");
System.out.println("用户名请求的参数值为" + username);
%>
这是b.jsp页面数据
</body>
浏览器输出:
通过网址的?属性=值
键值对来设置请求的参数值
控制台输出:
代码脚本可以和表达式脚本一起组合使用,在 jsp 页面上输出数据
<body>
<%
for (int i = 0; i < 5; i++) {
%>
<%=i%><br>
<%
}
%>
这是b.jsp页面数据
</body>
浏览器输出:
<body>
<table border="1">
<%
for (int i = 1; i <= 5; i++) {
%>
<tr>
<td>
第<%=i%>行
</td>
</tr>
<%
}
%>
</table>
这是b.jsp页面数据
</body>
浏览器输出:
2.3、jsp 中的三种注释
2.3.1、html 注释
<!-- 这是 html 注释 -->
html 注释会被翻译到 java 源代码中
在_jspService() 方法里,以 out.writer 输出到客户端
2.3.2、java 注释
<%
// 单行 java 注释
/* 多行 java 注释 */
%>
java 注释会被翻译到 java 源代码中
2.3.3、jsp 注释
<%-- 这是 jsp 注释 --%>
jsp 注释可以注掉,jsp 页面中所有代码
IDEA 代码显示:
java 源文件中的数据:
3、jsp 九大内置对象
jsp 中的内置对象,是指 Tomcat 在翻译 jsp 页面成为 Servlet 源代码后,内部提供的九大对象,叫内置对象
九大内置对象 | |
---|---|
HttpServletRequest request | 请求对象 |
HttpServletResponse response | 响应对象 |
PageContext pageContext | jsp 的上下文对象 |
HttpSession session | 会话对象 |
ServletContext application | ServletContext 对象 |
ServletConfig config | ServletConfig 对象 |
JspWriter out | jsp 输出流对象 |
Object page | 指向当前的 jsp 的对象 |
exception | 异常对象 |
4、jsp 四大域对象
四个域对象分别是:
四大域对象 | |
---|---|
PageContext 类 pageContext | 是 jsp 的上下文对象,当前 jsp 页面 范围内有效 |
HttpServletRequest 类 request | 请求对象,一次请求 内有效 |
HttpSession 类 session | 会话对象, 一个会话范围 内有效(打开浏览器访问服务器,直到关闭浏览器) |
ServletContext 类 application | ServletContext 对象,整个 web 工程 范围内都有效(只要 web 工程不停止,数据都在) |
域对象是可以像 Map 一样存取数据的对象
四个域对象功能一样,不同的是它们对数据的存取范围,虽然四个域对象都可以存取数据,在使用上它们是有优先顺序的
四个域在使用的时候,优先顺序分别是,他们从小到大的范围的顺序 pageContext
====>>> request
====>>> session
====>>> application
b.jsp 数据
<body>
<%--分别向四大域对象设置值,并取出--%>
<h2>here is b.jsp!!!</h2>
<%
pageContext.setAttribute("key", "pageContext");
request.setAttribute("key", "request");
session.setAttribute("key", "session");
application.setAttribute("key", "application");
%>
pageContext域对象的值:<%= pageContext.getAttribute("key")%><br>
request域对象的值:<%=request.getAttribute("key")%><br>
session域对象的值:<%=session.getAttribute("key")%><br>
application域对象的值:<%=application.getAttribute("key")%><br>
<%
request.getRequestDispatcher("/a.jsp").forward(request,response);
%>
</body>
a.jsp 数据
<body>
<h2>a.jsp!!!</h2>
pageContext域对象的值:<%= pageContext.getAttribute("key")%><br>
request域对象的值:<%=request.getAttribute("key")%><br>
session域对象的值:<%=session.getAttribute("key")%><br>
application域对象的值:<%=application.getAttribute("key")%><br>
</body>
请求转发之后PageContext 类 pageContext 对象就没有值了,因为当前 jsp 页面 范围内有效
访问 b 页面之后,再访问 a 页面,就是二次请求,HttpServletRequest 类 request 对象就没有值了
5、jsp 中的 out 输出和 response.getWriter 输出的区别
response 中表示响应,我们经常用于设置返回给客户端的内容(输出)
out 也是给用户做输出使用的
<body>
<%
out.write("out输出 <br/>");
response.getWriter().write("response输出 <br/>");
%>
</body>
明明 out输出 代码在上面,但是输出结果再下面,具体原因需要分析底层
如下:
<body>
<%
out.write("out输出1 <br/>");
out.flush();
out.write("out输出22 <br/>");
response.getWriter().write("response输出 <br/>");
%>
</body>
由于 jsp 翻译之后,底层源代码都是使用 out 来进行输出,所以一般情况下,我们在 jsp 页面中统一使用 out 来进行输出,避免打乱页面输出内容的顺序
out.write()
输出 字符串 没有问题(底层是直接强转为字节类型,结果就是ASC 码)
out.print()
输出 任意数据 都没有问题(都转换成为字符串后调用的 write 输出)
深入源码,浅出结论:在 jsp 页面中,可以统一使用 out.print()来进行输出
6、jsp 的常用标签
6.1、jsp 静态包含
静态包含使用情况:
当有多个网页,每个网页都有提供相同的信息,为了方便维护,可以将这重复的信息单独提取出来
静态包含的特点:
- 静态包含不会翻译被包含的 jsp 页面(只生成一个java源文件)
- 静态包含其实是把被包含的 jsp 页面的代码拷贝到包含的位置执行输出
示例:
<%--
<%@ include file="" %> 就是静态包含
file 属性指定你要包含的 jsp 页面的路径
地址中第一个斜杠 / 表示为 http://ip:port/工程路径/ ,映射到代码的 web 目录
--%>
<%@ include file="/b.jsp"%>
a .jsp 页面数据
<body>
<h2>a.jsp!!!</h2>
主体页面的信息<br>
<%--斜杠表示工程路径,也就是web目录下--%>
<%@ include file="/b.jsp"%>
</body>
被包含的 b .jsp 页面数据
<body>
<h2>这里是 b .jsp!!!</h2>
附属页面信息<br>
</body>
6.2、jsp 动态包含
动态包含的特点:
- 动态包含会把包含的 jsp 页面也翻译成为 java 代码
- 动态包含底层代码使用如下代码去调用被包含的 jsp 页面执行输出。
JspRuntimeLibrary.include(request, response, "/include/footer.jsp", out, false);
所以被包含 b 页面的 request、response、out 三个对象是引用包含 a 页面的对象 - 动态包含,还可以传递参数
示例:
a .jsp 页面数据
<body>
<%--
<jsp:include page=""></jsp:include> 这是动态包含
page 属性是指定你要包含的 jsp 页面的路径
动态包含也可以像静态包含一样,把被包含的内容执行输出到包含位置
--%>
<h2>a.jsp!!!</h2>
主体页面的信息<br>
<%--斜杠表示工程路径,也就是web目录下--%>
<jsp:include page="/b.jsp">
<jsp:param name="username" value="bbj"/>
<jsp:param name="password" value="root"/>
</jsp:include>
</body>
被包含的 b .jsp 页面数据
<body>
<h2>这里是 b .jsp!!!</h2>
附属页面信息:<%=request.getParameter("username")%><br>
附属页面信息:<%=request.getParameter("password")%>
</body>
动态包含的底层原理:
为什么输出结果先是主体页面,后是被包含页面:
1、因为主体页面调用被包含页面是使用如下代码
2、所以被包含 b 页面的 request、response、out 三个对象是引用包含 a 页面的对象
3、所以两个页面共用同一个 out 缓冲区
4、在代码中,先是主体页面代码执行,所以 out 缓冲区中先有主体页面,后有被包含页面,然后 out 缓冲区刷新到 response 缓冲区,然后输出到客户端浏览器页面显示
6.3、jsp 标签-转发
示例:
<%--
<jsp:forward page=""></jsp:forward> 是请求转发标签,功能就是请求转发
page 属性设置请求转发的路径
--%>
<body>
<h2>这里是 b .jsp!!!</h2>
<jsp:forward page="/a.jsp"></jsp:forward>
</body>
和以下代码功能一样
<%
request.getRequestDispatcher("/a.jsp").forward(request,response);
%>
7、练习题
在浏览器页面中输出九九乘法口诀表
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
<style type="text/css">
table{
width: 650px;
}
</style>
</head>
<body>
<%--在 jsp 页面中输出九九乘法口诀表--%>
<h2 align="center">九九乘法口诀表</h2>
<table align="center">
<%for (int i = 1; i < 10; i++) {%>
<tr>
<%for (int j = 1; j <= i; j++) {%>
<td><%=(j + "x" + i + "=" + (i * j)+" ")%></td>
<%}%>
</tr>
<%}%>
</table>
</body>
</html>
从servlet程序请求中转到 jsp
客户端请求--------------》servlet获取请求参数------------》通过业务层service转到DAO层,通过DAO层的sql语句去访问数据库数据-----------》将获取到的数据保存到request,传递到jsp-----------》通过jsp处理数据并响应输出到浏览器
在 servlet 程序的 doGET() 方法中将获取到的数据存入 request 参数,并请求中转到 jsp 页面
servlet 程序的 doGET()
方法
// 保存查询到的结果(学生信息)到 request 域中
req.setAttribute("stuList", studentList);
// 请求转发到 showStudent.jsp 页面
req.getRequestDispatcher("/test/showStudent.jsp").forward(req,resp);
jsp 中处理数据并返回到客户端
得到 request 域中的数据
<% List<Student> studentList = (List<Student>) request.getAttribute("stuList"); %>
进行输出
<% for (Student student : studentList) { %>
<%=student.getId()%>
<%=student.getName()%>
<%=student.getAge()%>
<%=student.getPhone()%>
<% } %>
8、Listener 监听器
什么是 Listener 监听器?
- Listener 监听器它是 JavaWeb 的三大组件之一
JavaWeb 的三大组件分别是:Servlet 程序、Filter 过滤器、Listener 监 听器 - Listener 它是 JavaEE 的规范,就是接口
- 监听器的作用是,监听某种事物的变化,然后通过回调函数,反馈给客户(程序)去做一些相应的处理
8.1、ServletContextListener 监听器
ServletContextListener
它可以 监听 ServletContext 对象的创建和销毁
ServletContext 对象在 web 工程启动的时候创建,在 web 工程停止的时候销毁
监听到创建和销毁之后都会分别调用 ServletContextListener 监听器的方法反馈
两个方法分别是:
contextInitialized()
创建之后马上调用,做初始化
contextDestroyed()
销毁之后调用
public interface ServletContextListener extends EventListener {
/*** 在 ServletContext 对象创建之后马上调用,做初始化 */
public void contextInitialized(ServletContextEvent sce);
/*** 在 ServletContext 对象销毁之后调用 */
public void contextDestroyed(ServletContextEvent sce);
}
如何使用 ServletContextListener 监听器监听 ServletContext 对象?
使用步骤如下:
- 编写一个类去实现 ServletContextListener
- 实现其两个回调方法
- 到 web.xml 中去配置监听器
编写监听器实现类:
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
public class MyServletContextListenerImpl implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent servletContextEvent) {
System.out.println("ServletContext 对象被创建了");
}
@Override
public void contextDestroyed(ServletContextEvent servletContextEvent) {
System.out.println("ServletContext 对象被销毁了");
}
}
web.xml 中的配置:
<!--配置监听器-->
<listener>
<listener-class>MyServletContextListenerImpl</listener-class>
</listener>
web 工程启动的时候,或者重新部署的时候
web 工程停止的时候