Java Server Pages

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>

在这里插入图片描述

  1. 如何创建 jsp 的页面?
    在这里插入图片描述
  2. jsp 如何访问:
    jsp 页面和 html 页面一样,都是存放在 web 目录下,访问也跟访问 html 页面一样
    1. 在 web 目录下有如下的文件:
      a.html 页面,访问地址是http://ip:port/工程路径/a.html
      b.jsp 页面,访问地址是http://ip:port/工程路径/b.jsp

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 源文件中也会收到数据
在这里插入图片描述
表达式脚本的特点:

  1. 所有的表达式脚本都会被翻译到 _jspService() 方法中
  2. 表达式脚本都会被翻译成为 out.print() 输出到页面上
  3. 由于表达式脚本翻译的内容都在_jspService() 方法中,所以_jspService()方法中的对象都可以直接使用
  4. 表达式脚本中的表达式不能以分号结束
<%=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 语句)

代码脚本的特点是

  1. 代码脚本翻译之后都在 _jspService() 方法中
  2. 由于代码脚本翻译之后是存到 _jspService() 方法中,所以在_jspService()方法中的现有对象都可以直接使用
  3. 还可以由多个代码脚本块组合完成一个完整的 java 语句
  4. 代码脚本可以和表达式脚本一起组合使用,在 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 pageContextjsp 的上下文对象
HttpSession session会话对象
ServletContext applicationServletContext 对象
ServletConfig configServletConfig 对象
JspWriter outjsp 输出流对象
Object page指向当前的 jsp 的对象
exception异常对象

4、jsp 四大域对象

四个域对象分别是:

四大域对象
PageContext 类 pageContext是 jsp 的上下文对象,当前 jsp 页面 范围内有效
HttpServletRequest 类 request请求对象,一次请求 内有效
HttpSession 类 session会话对象, 一个会话范围 内有效(打开浏览器访问服务器,直到关闭浏览器)
ServletContext 类 applicationServletContext 对象,整个 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 静态包含

静态包含使用情况:
当有多个网页,每个网页都有提供相同的信息,为了方便维护,可以将这重复的信息单独提取出来

静态包含的特点:

  1. 静态包含不会翻译被包含的 jsp 页面(只生成一个java源文件)
  2. 静态包含其实是把被包含的 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 动态包含

动态包含的特点:

  1. 动态包含会把包含的 jsp 页面也翻译成为 java 代码
  2. 动态包含底层代码使用如下代码去调用被包含的 jsp 页面执行输出。
    JspRuntimeLibrary.include(request, response, "/include/footer.jsp", out, false);
    所以被包含 b 页面的 request、response、out 三个对象是引用包含 a 页面的对象
  3. 动态包含,还可以传递参数

示例:

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 监听器?

  1. Listener 监听器它是 JavaWeb 的三大组件之一
    JavaWeb 的三大组件分别是:Servlet 程序、Filter 过滤器、Listener 监 听器
  2. Listener 它是 JavaEE 的规范,就是接口
  3. 监听器的作用是,监听某种事物的变化,然后通过回调函数,反馈给客户(程序)去做一些相应的处理

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 对象?

使用步骤如下:

  1. 编写一个类去实现 ServletContextListener
  2. 实现其两个回调方法
  3. 到 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 工程停止的时候
在这里插入图片描述

在这里插入图片描述

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值