JSP
1、什么是 jsp?
- (1)jsp 的全称是 java server page,java 的服务器页面。
- 是 sun 公司专门为了解决动态生成 html 页面文档的技术
- (2)jsp 的主要作用是代替 Servlet 程序回传 html 页面的数据。
- (3)因为 Servlet 程序回传 html 页面数据是一件非常繁琐的事情,开发成本和维护成本都极高。
Servlet 回传 html 页面的代码:
public class ResponseHtml extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException,IOException {
// 通过响应的回传流回传 html 页面数据
resp.setContentType("text/html; charset=UTF-8");
PrintWriter writer = resp.getWriter();
writer.write("<!DOCTYPE html>\r\n");
writer.write(" <html lang=\"en\">\r\n");
writer.write(" <head>\r\n");
writer.write("<meta charset=\"UTF-8\">\r\n");
writer.write("<title>Title</title>\r\n");
writer.write(" </head>\r\n");
writer.write(" <body>\r\n");
writer.write("这是 html 页面数据 \r\n");
writer.write(" </body>\r\n");
writer.write("</html>\r\n");
writer.write("\r\n");
}
}
jsp 回传一个简单页面的代码:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
这是 html 页面数据
</body>
</html>
jsp 小结:
①如何创建一个 jsp 的页面?
web > New > JSP/JSPX


②jsp 如何访问?
jsp 页面和 html 页面一样,都是存放在 web 目录下,访问也跟 html 页面一样。
比如:web 目录下有如下文件:
a.html 访问地址是:http://ip:port/工程名/a.html
b.jsp 访问地址是:http://ip:port/工程名/b.jsp
2、jsp 的本质是什么?
jsp 的本质上是一个 Servlet 程序。
当我们第一次访问 jsp 页面的时候,Tomcat 服务器会帮我们把 jsp 页面翻译成一个 java 源文件,并且对它进行编程成为 .class 字节码文件。我们打开 java 源文件不难发现其里面的内容为:

我们跟踪源码发现了 HttpJapBase 类,它直接继承了 HttpServlet 类,也就是说,jsp 翻译出来的 java 类,它间接地继承了 HttpServlet 类,所以本质上 jsp 就是翻译出来的一个 Servlet 程序。

总结:通过翻译的 java 源码我们就可以得到结果:jsp 就是 Servlet 程序。
a.jsp 翻译成 java 文件后的全名是 a_jsp.java 文件
b.jsp 翻译成 java 文件后的全名是 b_jsp.java 文件
那么 当我们访问一个 xxx.jsp 文件后 翻译成 java 文件的全名是 xxx_jsp.java 文件
xxx_jsp.java 文件是一个 Servlet 程序。原来 jsp 中的 html 内容都被翻译到 Servlet 类的 service 方法中原样输出。
大家也可以去观察翻译出来的 Servlet 程序的源代码,不难发现,其底层实现,也是通过输出流,把 html 页面数据回传给客户端。
3、jsp 的三种用法
3.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
缓冲区溢出错误:

=上面这两个属性是给out输出流使用的==
⑦errorPage 属性:设置当 jsp 页面运行出错时,自动跳转去的错误页面路径
这个路径一般都是以斜杆打头,它表示请求地址为:http://ip:port/工程名/ ,映射到代码的 web 目录
⑧isErrorPage 属性:设置当前页面是否是错误信息页面,默认是 false,如果是 true 可以获取异常信息
⑨session 属性:设置访问当前 jsp 页面,是否会创建 HttpSessin 对象,默认是 true
⑩extends 属性:设置 jsp 翻译出来的 java 类默认继承谁
3.2 jsp 中的常用脚本
①声明脚本(极少使用)
声明脚本的格式是:
<%! 声明 java 代码%>
作用:可以给 jsp 翻译出来的 java 类定义属性和方法甚至是静态代码块、内部类等。
示例代码:
<%--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 页面上输出数据
表达式脚本的特点:
- 所有的表达式脚本会被翻译到
_jspServlet()
方法中 - 表达式脚本都会被翻译成为
out.print()
输出在页面上 - 由于表达式脚本翻译的内容都在
_jspServlet()
方法中,所以 _jspServlet() 方法中的对象都可以直接使用 - 表达式脚本中的表达式不能以分号结束
示例代码:
<%=12 %> <br>
<%=12.12 %> <br>
<%="我是字符串" %> <br>
<%=map%> <br>
<%=request.getParameter("username")%>
翻译对照:

③代码脚本
代码脚本的格式:
<%
java 语句
%>
代码脚本的作用:
1、代码脚本翻译之后都在 _jspService
方法中
2、代码脚本由于翻译到_jspService()
方法中,所以在 _jspService()
方法中的现有对象都可以直接使用
3、还可以由多个代码脚本块组合完成一个完整的 java 语句
4、代码脚本还可以和表达式脚本一起组合使用,在 jsp 页面上输出数据
示例代码:
<%--练习:--%>
<%--1.代码脚本----if 语句--%>
<%
int i = 13 ;
if (i == 12) {
%>
<h1>国哥好帅</h1>
<%
} else {
%>
<h1>国哥又骗人了!</h1>
<%
}
%>
<br>
<%--2.代码脚本----for 循环语句--%>
<table border="1" cellspacing="0">
<%
for (int j = 0; j < 10; j++) {
%>
<tr>
<td>第 <%=j + 1%>行</td>
</tr>
<%
}
%>
</table>
<%--3.翻译后 java 文件中_jspService 方法内的代码都可以写--%>
<%
String username = request.getParameter("username");
System.out.println("用户名的请求参数值是:" + username);
%>
翻译对照:

3.3 jsp 中的三种注释
①html 注释
<!-- 这是 html 注释 -->
html 注释会被翻译到 java 源码中,在 _jspService() 方法里,以 out.writer 输出到客户端
②java 注释
<%
// 单行 java 注释
/* 多行 java 注释 */
%>
java 注释会被翻译到 java 源代码中
③jsp 注释
<%-- 这是 jsp 注释 --%>
jsp 注释可以注掉,jsp 页面中所有代码(jsp 中真正的注释)
4、jsp 内置的九大对象
jsp 中的内置对象,是指 Tomcat 在翻译 jsp 页面成为 Servlet 源代码后,内部提供的九大对象
内置对象 | 说明 |
---|---|
request | 请求对象,可以获取请求信息 |
response | 响应对象,可以设置响应信息 |
pageContext | 当前页面的上下文对象,可以在当前页面上下文保存属性信息 |
session | 会话对象,可以获取会话信息 |
application | ServletContext 对象,可以获取整个工程的一些信息 |
config | ServletConfig 对象,可以获取 Servlet 的配置信息 |
out | jsp 的输出流对象 |
page | 表示当前 jsp 对象(无用,用它不如使用 this 对象) |
exception | 异常对象,只有在 jsp 页面的 page 指令中设置 isErrorPage="true" 时才会存在 |

**注:**九大内置对象,都是我们可以在【代码脚本】中或【表达式脚本】中直接使用的对象。
5、jsp 四大域对象
(1)四个域对象
pageContext (PageContextImpl类) 当前jsp页面范围有效
request (HttpServletRequest类) 一次请求内有效(经常用于转发时传递数据)
session (HttpSession类) 一个会话范围内有效(打开浏览器访问服务器,直到关闭浏览器)
application (ServletContext类) 整个web工程范围内都有效(只要 web 工程不停止,数据都在)
(2)域对象是可以向 Map 一样存取数据的对象
- ①四个域对象的功能一样,不同的是它们对数据的存取范围
- ②虽然四个域对象都可以存取数据,在使用上它们是有优先顺序的
- ③四个域对象在使用的时候,优先顺序分别是,他们从小到大范围的顺序是:
- pageContext ====>>> request =====>>> session ====>>> application
scope1.jsp 页面:
<body>
<h1>scope1.jsp 页面</h1>
<%
// 往四个域中都分别保存了数据
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("/scope2.jsp").forward(request,response);
%>
</body>
scope2.jsp 页面:
<body>
<h1>scope2.jsp 页面</h1>
pageContext 域是否有值:<%=pageContext.getAttribute("key")%> <br>
request 域是否有值:<%=request.getAttribute("key")%> <br>
session 域是否有值:<%=session.getAttribute("key")%> <br>
application 域是否有值:<%=application.getAttribute("key")%> <br>
</body>
6、jsp 中的 out 输出和 response.getWriter 输出的区别
response 表示响应,我们经常用于设置返回给客户端的内容(输出)
out 也是给用户做输出使用的

由于 jsp 翻译之后,底层源码都是使用 out 来进行输出的,所以一般情况下,我们在 jsp 页面中统一使用 out 来进行输出,避免打乱页面输出内容的顺序
out.write():输出字符串没有问题
out.print():输出任何数据都没有问题(都转换称为字符串后调用的 write 输出)
深入源码,浅出结论:在 jsp 页面中,可以统一使用 out.print() 来进行输出
7、jsp 的常用标签
7.1 jsp 静态包含
示例说明:
<%--
<%@ include file=""%> 就是静态包含
file 属性指定你要包含的 jsp 页面的路径
地址中第一个斜杠 / 表示为 http://ip:port/工程路径/ 映射到代码的 web 目录
静态包含的特点:
1、静态包含不会翻译被包含的 jsp 页面。
2、静态包含其实是把被包含的 jsp 页面的代码拷贝到包含的位置执行输出
--%>
<%@ include file="/include/footer.jsp"%>
7.2 jsp 动态包含
示例说明:
<%--
<jsp:include page=""></jsp:include>
这是动态包含
page 属性是指定你要包含的 jsp 页面的路径
动态包含也可以像静态包含一样。把被包含的内容执行输出到包含位置
动态包含的特点:
1、动态包含会把包含的 jsp 页面也翻译成为 java 代码
2、动态包含底层代码使用如下代码去调用被包含的 jsp 页面执行输出。
JspRuntimeLibrary.include(request, response, "/include/footer.jsp", out, false);
3、动态包含,还可以传递参数
--%>
<jsp:include page="/include/footer.jsp">
<jsp:param name="username" value="bbj"/>
<jsp:param name="password" value="root"/>
</jsp:include>
动态包含的底层原理:

7.3 jsp 标签-转发
示例说明:
<%--
<jsp:forward page=""></jsp:forward> 是请求转发标签,它的功能就是请求转发
page 属性设置请求转发的路径
--%>
<jsp:forward page="/scope2.jsp"></jsp:forward>
8、jsp 的练习题
(1)练习一:在 jsp 页面中输出九九乘法口诀表
<%@ 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 页面中输出九九乘法口诀表 --%>
<h1 align="center">九九乘法口诀表</h1>
<table align="center">
<%-- 外层循环遍历行 --%>
<% for (int i = 1; i <= 9; i++) { %>
<tr>
<%-- 内层循环遍历单元格 --%>
<% for (int j = 1; j <= i ; j++) { %>
<td><%=j + "x" + i + "=" + (i*j)%></td>
<% } %>
</tr>
<% } %>
</table>
</body>
</html>
(2)练习二:jsp 输出一个表格,里面有10个学生信息

Student 类:
public class Student {
private Integer id;
private String name;
private Integer age;
private String phone;
}
SearchStudentServlet 程序:
public class SearchStudentServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException,IOException {
// 获取请求的参数
// 发 sql 语句查询学生的信息
// 使用 for 循环生成查询到的数据做模拟
List<Student> studentList = new ArrayList<Student>();
for (int i = 0; i < 10; i++) {
int t = i + 1;
studentList.add(new Student(t,"name"+t, 18+t,"phone"+t));
}
// 保存查询到的结果(学生信息)到 request 域中
req.setAttribute("stuList", studentList);
// 请求转发到 showStudent.jsp 页面
req.getRequestDispatcher("/test/showStudent.jsp").forward(req,resp);
}
}
showStudent.jsp 页面:
<%@ page import="java.util.List" %>
<%@ page import="com.atguigu.pojo.Student" %>
<%@ page import="java.util.ArrayList" %>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
<style>
table{
border: 1px blue solid;
width: 600px;
border-collapse: collapse;
}
td,th{
border: 1px blue solid;
}
</style>
</head>
<body>
<%--练习二:jsp 输出一个表格,里面有 10 个学生信息。--%>
<%
List<Student> studentList = (List<Student>) request.getAttribute("stuList");
%>
<table>
<tr>
<td>编号</td>
<td>姓名</td>
<td>年龄</td>
<td>电话</td>
<td>操作</td>
</tr>
<% for (Student student : studentList) { %>
<tr>
<td><%=student.getId()%></td>
<td><%=student.getName()%></td>
<td><%=student.getAge()%></td>
<td><%=student.getPhone()%></td>
<td>删除、修改</td>
</tr>
<% } %>
</table>
</body>
</html>
9、什么是 Listener 监听器?
- Listener 监听器它是 JavaWeb 的三大组件之一。JavaWeb 的三大组件分别是:Servlet 程序、Filter 过滤器、Listener 监听器。
- Listener 是 JavaEE 的规范,就是接口。
- 监听器的作用是,监听某种事物的变化,然后通过回调函数,反馈给客户(程序)去做一些响应的处理。
ServletContextListener 监听器
(1)ServletContextListener 它可以监听 ServletContext 对象的创建和销毁
(2)ServletContext 对象在 web 工程启动的时候创建,在 web 工程停止的时候销毁
(3)监听到创建和销毁之后都会分别调用 ServletContextListener 监听器的方法反馈
两个方法分别是:
public interface ServletContextListener extends EventListener {
/**
* 在 ServletContext 对象创建之后马上调用,做初始化
*/
public void contextInitialized(ServletContextEvent sce);
/**
* 在 ServletContext 对象销毁之后调用
*/
public void contextDestroyed(ServletContextEvent sce);
}
(4)如何使用 ServletContextListener 监听器监听 ServletContext 对象
使用步骤如下:
- ①编写一个类去实现 ServletContextListener 接口
- ②实现两个回调方法
- ③到 web.xml 中去配置监听器
监听器实现类:
public class MyServletContextListenerImpl implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent sce) {
System.out.println("ServletContext 对象被创建了");
}
@Override
public void contextDestroyed(ServletContextEvent sce) {
System.out.println("ServletContext 对象被销毁了");
}
}
web.xml 中的配置:
<!--配置监听器-->
<listener>
<listener-class>com.atguigu.listener.MyServletContextListenerImpl</listener-class>
</listener>