饭宝 发表于 2022-9-16 17:21:55

JSP基础知识总结

目录

[*]JSP概述

[*]什么是 jsp
[*]Servlet 程序输出 html 页面
[*]如何创建一个 jsp 动态页面程序
[*]如何修改 jsp 文件的默认编码

[*]jsp 的运行原理
[*]jsp 的语法

[*]jsp 文件头部声明介绍(page 指令介绍)
[*]jsp 中的三种脚本介绍
[*]jsp 中的注释

[*]jsp 九大内置对象
[*]jsp 四大域对象
[*]jsp中out输出流和response.getwriter()输出流

[*]jsp中out 和response的writer的区别演示
[*]out 流和 writer 流的两个缓冲区如何工作

[*]jsp 的常用标签

[*]静态包含--很常用
[*]动态包含--很少用
[*]页面转发--常用

[*]Listener 监听器

[*]什么是 Listener 监听器

[*]ServletContextListener 监听器

JSP概述

什么是 jsp

JSP(全称 Java Server Pages)是由 Sun 公司专门为了解决动态生成 HTML 文档的技术。
Servlet 程序输出 html 页面

在 jsp 技术之前,如果我们要往客户端输出一个页面。我们可以使用 Servlet 程序来实现。具体的代码如下:
public class HtmlServlet extends HttpServlet {
    private static final long serialVersionUID = 1L;
    protected void doGet(HttpServletRequest request,
                         HttpServletResponse response) throws ServletException, IOException {
      // 设置返回的数据内容的数据类型和编码
      response.setContentType("text/html; charset=utf-8");
      // 获取字符输出流
      Writer writer = response.getWriter();
      //输出页面内容
      writer.write("<!DOCTYPE html PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\"\"http://www.w3.org/TR/html4/loose.dtd\">");
      writer.write("<html>");
      writer.write("<head>");
      writer.write("<meta http-equiv=\"Content-Type\" content=\"text/html; charset=UTF-8\">");
      writer.write("<title>Insert title here</title>");
      writer.write("</head>");
      writer.write("<body>");
      writer.write("这是由 Servlet 程序输出的 html 页面内容!");
      writer.write("</body></html>");
    }
    protected void doPost(HttpServletRequest request,
                        HttpServletResponse response) throws ServletException, IOException {
    }
}在浏览器中输入访问 Servlet 的程序地址得到以下结果:
https://pic.imgdb.cn/item/6308719c16f2c2beb1cfd672.jpg
从上面的代码可以发现。通过 Servlet 输出简单的 html 页面信息都非常不方便。
如果要输出一个复杂页面的时候,就更加的困难,而且不利于页面的维护和调试。
所以 sun 公司推出一种叫做 jsp 的动态页面技术来实现对页面的输出繁锁工作。
jsp 页面的访问不能像 HTML 页面一样拖到浏览器中。只能通过浏览器访问 Tomcat 服务器再访问 jsp 页面。
如何创建一个 jsp 动态页面程序


[*]选中 Web 目录,右键创建一个 jsp 文件
https://pic.imgdb.cn/item/6308728a16f2c2beb1d05f38.jpg

[*]输入 jsp 页面的文件名
https://pic.imgdb.cn/item/630872f416f2c2beb1d0937b.jpg
[*]在 body 标签中添加你想要显示的文本内容
https://pic.imgdb.cn/item/630873ab16f2c2beb1d0f7d2.jpg
[*]先启动Tomcat服务器,
然后在浏览器中输入 jsp 页面的访问地址。
jsp 页面的访问地址和 html 页面的访问路径一样 http://ip:端口号/工程名/文件名
也就是 http://127.0.0.1:8080/Test/Demo.jsp或 http://localhost:8080/Test/Demo.jsp(以自己的工程路径为准)
https://pic.imgdb.cn/item/6308747816f2c2beb1d17559.jpg
如何修改 jsp 文件的默认编码

有些idea的默认编码格式可能是GBK格式或者其他格式的,然后在JSP编码时候可能会出现乱码,修改为UTF-8格式可以避免乱码。
https://pic.imgdb.cn/item/6308765116f2c2beb1d27985.jpg
注意事项:
1、jsp 页面是一个类似于 html 的一个页面。 jsp 直接存放到 WebContent 目录下,和 html 一样
访问 jsp 的时候,也和访问 html 一样
2、jsp 的默认编码集是 iso-8859-1 修改 jsp 的默认编码为 UTF-8
jsp 的运行原理

jsp的本质其实是一个Servlet程序。
首先我们去找到我们 Tomcat 的目录下的 work\Catalina\localhost 目录。
当我们新建Demo工程。并启动 Tomcat
服务器后。我们发现
在 work\Catalina\localhost 目录下多出来一个 JSPDemo目录。
https://pic.imgdb.cn/item/6308804216f2c2beb1d8ba94.jpg
一开始目录还是空目录。
然后,我们在浏览器输入一个 jsp 文件的访问路径访问。
比如 http://127.0.0.1:8080/JSPDemo/Demo.jsp访问Demo.jsp 文件
JSPDemo目录马上会生成 org\apache\jsp 目录。
并且在会中有两个文件。
https://pic.imgdb.cn/item/630880ca16f2c2beb1d93767.jpg
Demo_jsp.class 文件很明显是 index_jsp.java 源文件编译后的字节码文件。
那么Demo_jsp.java 是个什么内容呢?
生成的 java 文件名,是以原来的文件名加上_jsp 得到。 xxxx_jsp.java 文件的名字
我们打开 Demo_jsp.java 文件查看里面的内容:
发现,生成的类继承于 HttpJspBase 类。这是一个 jsp 文件生成 Servlet 程序要继承的基类!!!
https://pic.imgdb.cn/item/630881ec16f2c2beb1da3ab7.jpg
自动生成的java文件继承于HttpJspBase类
于是,关联源代码去查看一下 HttpJspBase 类的内容。从源码的类注释说明中,发现HttpJspBase 这个类就是所有 jsp 文件生成 Servlet 程序需要去继承的基类。并且这个 HttpJspBase 类继承于 HttpServlet 类。所以 jsp 也是一个 Servlet 小程序。
https://pic.imgdb.cn/item/630884c516f2c2beb1dc27f3.jpg
我们分别在工程的 Web 目录下创建多个 jsp 文件。然后依次访问。它们都被翻译为.java 文件并编译成为.class 字节码文件
https://pic.imgdb.cn/item/6308859e16f2c2beb1dcb921.jpg
打开 Demo_jsp.java 文件查看里面的内容可以发现。jsp 中的 html 页面内容都被翻译到 Servlet 中的 service
方法中直接输出。
https://www.z4a.net/images/2022/08/26/imagef9cb549082749771.png
小结:
从生成的文件我们不难发现一个规则。
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 方法中原样输出。
jsp 的语法

jsp 文件头部声明介绍(page 指令介绍)

这是 jsp 文件的头声明。表示这是 jsp 页面。
属性含义language属性值只能是 java,表示翻译的得到的是 java 语言的contentType设置响应头 contentType 的内容pageEncoding设置当前 jsp 页面的编码import给当前 jsp 页面导入需要使用的类包autoFlush设置是否自动刷新 out 的缓冲区,默认为 truebuffer设置 out 的缓冲区大小。默认为 8KBerrorPage设置当前 jsp 发生错误后,需要跳转到哪个页面去显示错误信息isErrorPage设置当前 jsp 页面是否是错误页面。是的话,就可以使用 exception 异常对象session设置当前 jsp 页面是否获取 session 对象,默认为 trueextends给服务器厂商预留的 jsp 默认翻译的 servlet 继承于什么类jsp 中的三种脚本介绍

一、声明脚本:
声明脚本格式如下:
在声明脚本块中,我们可以干 4 件事情
1.我们可以定义全局变量。
2.定义 static 静态代码块
3.定义方法
4.定义内部类
几乎可以写在类的内部写的代码,都可以通过声明脚本来实现
二、表达式脚本
表达式脚本格式如下:
表达式脚本 用于向页面输出内容。
表达式脚本 翻译到 Servlet 程序的 service 方法中 以 out.print() 打印输出
out 是 jsp 的一个内置对象,用于生成 html 的源代码
注意:表达式不要以分号结尾,否则会报错
表达式脚本可以输出任意类型。
比如:
1.输出整型
2.输出浮点型
3.输出字符串
4.输出对象
三、代码脚本
代码脚本如下:
代码脚本里可以书写任意的 java 语句。
代码脚本的内容都会被翻译到 service 方法中。
所以 service 方法中可以写的 java 代码,都可以书写到代码脚本中
jsp 中的注释

// 单行 java 注释
/*
多行 java 代码注释
*/
单行注释和多行注释能在翻译后的 java 源代码中看见。
<%-- jsp 注释 --%>
jsp 注释在翻译的时候会直接被忽略掉

html 的注释会被翻译到 java 代码中输出到 html 页面中查看jsp 九大内置对象

我们打开翻译后的 java 文件。查看_jspService 方法。
https://pic.imgdb.cn/item/6308896a16f2c2beb1dec533.jpg
通过源码可以发现 jsp 中九大内置对象分别是:
对象含义request 对象请求对象,可以获取请求信息response 对象响应对象。可以设置响应信息pageContext 对象当前页面上下文对象。可以在当前上下文保存属性信息session 对象会话对象。可以获取会话信息exception 对象异常对象只有在 jsp 页面的 page 指令中设置 isErrorPage="true" 的时候才会存在application 对象ServletContext 对象实例,可以获取整个工程的一些信息config 对象ServletConfig 对象实例,可以获取 Servlet 的配置信息out 对象输出流page 对象表示当前 Servlet 对象实例(无用,用它不如使用 this 对象)九大内置对象 , 都是我们可以在 【 代码脚本 】 中或 【 表达式脚本 】 中直接使用的对象。
jsp 四大域对象

四大域对象经常用来保存数据信息。
域对象含义pageContext可以保存数据在同一个 jsp 页面中使用request可以保存数据在同一个 request 对象中使用。经常用于在转发的时候传递数据session可以保存在一个会话中使用application(ServletContext)就是 ServletContext 对象四个作用域的测试代码:
新建两个 jsp 页面。分别取名叫:context1.jsp,context2.jsp
context1.jsp 的页面代码如下:
<%--
Created by IntelliJ IDEA.
User: Kohler
Date: 2022/8/26
Time: 20:00
To change this template use File | Settings | File Templates.
--%>
<%@ page language="java" contentType="text/html; charset=UTF-8"
         pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
    <head>
      <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
      <title>context1</title>
    </head>
    <body>
      这是 context1 页面<br/>
      <%
            //设置 page 域的数据
            pageContext.setAttribute("key", "pageContext-value");
      //设置 request 域的数据
            request.setAttribute("key", "request-value");
      //设置 session 域的数据
            session.setAttribute("key", "session-value");
      //设置 application 域的数据
            application.setAttribute("key", "application-value");
      %>
      <%-- 测试当前页面作用域 --%>
      <%=pageContext.getAttribute("key") %><br/>
      <%=request.getAttribute("key") %><br/>
      <%=session.getAttribute("key") %><br/>
      <%=application.getAttribute("key") %><br/>
      <%
            // 测试 request 作用域
      // request.getRequestDispatcher("/context2.jsp").forward(request, response);
      %>
    </body>
</html>context2.jsp 的页面代码如下:
<%--
Created by IntelliJ IDEA.
User: Kohler
Date: 2022/8/26
Time: 20:03
To change this template use File | Settings | File Templates.
--%>
<%@ page language="java" contentType="text/html; charset=UTF-8"
         pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>context2</title>
</head>
<body>
    这是 context2 页面 <br/>
    <%=pageContext.getAttribute("key") %><br/>
    <%=request.getAttribute("key") %><br/>
    <%=session.getAttribute("key") %><br/>
    <%=application.getAttribute("key") %><br/>
</body>
</html>测试 pageContext 作用域:
直接访问 context1.jsp 文件,结果:
https://pic.imgdb.cn/item/6308ba3316f2c2beb1fe2514.jpg
测试 request 作用域:
1.在 context1.jsp 文件中添加转发到 context2.jsp(有数据)
2.直接访问 context2.jsp 文件 (没有数据)
测试 session 作用域:
1.访问完 context1.jsp 文件
2.关闭浏览器。但是要保持服务器一直开着
3.打开浏览器,直接访问 context2.jsp 文件
测试 application 作用域:
1.访问完 context1.jsp 文件,然后关闭浏览器
2.停止服务器。再启动服务器。
3.打开浏览器访问 context2.jsp 文件
https://pic.imgdb.cn/item/6308bc6d16f2c2beb1ff6a67.jpg
context1.jsp的页面https://pic.imgdb.cn/item/6308bd6916f2c2beb1ffeb94.jpg
context2.jsp的页面context1.jsp页面中:
context1页面的pageContent的值成为null了,因为添加转发跳转到了另一个jsp页面
request还有值是因为,虽然跳转到了另一个页面但是这还属于一次请求。
context1和context2中session还有值是因为当前都是在一个浏览器中访问,换一个浏览器的结果:
https://pic.imgdb.cn/item/6308be6d16f2c2beb1006556.jpg
application一直有值是因为当前tomcat一直在一次运行中,如果重新部署或者重启服务器就会变为null
重启服务器:
https://pic.imgdb.cn/item/6308bef516f2c2beb100a18b.jpg
jsp中out输出流和response.getwriter()输出流

jsp中out 和response的writer的区别演示

<%--
Created by IntelliJ IDEA.
User: Kohler
Date: 2022/8/26
Time: 20:41
To change this template use File | Settings | File Templates.
--%>
<%@ page language="java" contentType="text/html; charset=UTF-8"
         pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
    <head>
      <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
      <title>output</title>
    </head>
    <body>
      <%
            // out 输出
            out.write("这是 out 的第一次输出<br/>");
      // out flush 之后。会把输出的内容写入 writer 的缓冲区中
            out.flush();
      // 最后一次的输出,由于没有手动 flush,会在整个页面输出到客户端的时候,自动写入到 writer 缓冲区
            out.write("这是 out 的第二次输出<br/>");
      // writer 的输出
            response.getWriter().write("这是 writer 的第一次输出<br/>");
            response.getWriter().write("这是 writer 的第二次输出<br/>");
      %>
    </body>
</html>在浏览器里输入 http://localhost:8080/JSPDemo/output.jsp运行查看的结果:
https://pic.imgdb.cn/item/6308c08b16f2c2beb1015bd5.jpg
out 流和 writer 流的两个缓冲区如何工作

https://pic.imgdb.cn/item/6308c5db16f2c2beb104ba34.png
jsp 的常用标签

静态包含--很常用

静态包含是把包含的页面内容原封装不动的输出到包含的位置。
动态包含--很少用

动态包含会把包含的 jsp 页面单独翻译成 servlet 文件,然后在执行到时候再调用翻译的 servlet 程序。并把
计算的结果返回。
动态包含是在执行的时候,才会加载。所以叫动态包含。
页面转发--常用

页: [1]
查看完整版本: JSP基础知识总结