JSP

什么是 JSP,它有什么用?

jsp 的全换是 java server pages。Java 的服务器页面。 jsp 的主要作用是代替 Servlet 程序回传 html 页面的数据。因为 Servlet 程序回传 html 页面数据是一件非常繁锁的事情。开发成本和维护成本都极高。

Servlet 回传 html 页面数据的代码:

public class PringHtml 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 回传一个简单 html 页面的代码:

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
这是jsp的html页面数据
</body>
</html>

如何创建JSP页面

在这里插入图片描述

JSP如何访问

jsp 页面和 html 页面一样,都是存放在 web 目录下。访问也跟访问 html 页面一样。

举个栗子:

在 web 目录下有如下的文件:

  • test.html 页面 访问地址是 =======>>>>>> http://ip:port/工程路径/test.html
  • test.jsp 页面 访问地址是 =======>>>>>> http://ip:port/工程路径/test.jsp

JSP的本质是什么

jsp 页面本质上是一个 Servlet 程序。

当我们第一次访问 jsp 页面的时候。Tomcat 服务器会帮我们把 jsp 页面翻译成为一个 java 源文件。并且对它进行编译成为.class 字节码程序。我们打开 java 源文件不难发现其里面的内容是:

在这里插入图片描述

我们跟踪原代码发现,HttpJspBase 类。它直接地继承了 HttpServlet 类。也就是说。jsp 翻译出来的 java 类,它间接了继承了 HttpServlet 类。也就是说,翻译出来的是一个 Servlet 程序:

在这里插入图片描述

所以说,通过翻译的 java 源代码我们就可以得到结果:jsp 就是 Servlet 程序。

可以去观察翻译出来的 Servlet 程序的源代码,不难发现。其底层实现,也是通过输出流。把 html 页面数据回传给客户端。

public void _jspService(final javax.servlet.http.HttpServletRequest request, final
javax.servlet.http.HttpServletResponse response)
throws java.io.IOException, javax.servlet.ServletException {

final java.lang.String _jspx_method = request.getMethod();
if (!"GET".equals(_jspx_method) && !"POST".equals(_jspx_method) && !"HEAD".equals(_jspx_method) && !javax.servlet.DispatcherType.ERROR.equals(request.getDispatcherType())) {

HEAD");

}

response.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, "JSPs only permit GET POST or return;


final javax.servlet.jsp.PageContext pageContext; javax.servlet.http.HttpSession session = null; final javax.servlet.ServletContext application; final javax.servlet.ServletConfig config; javax.servlet.jsp.JspWriter out = null;
final java.lang.Object page = this; javax.servlet.jsp.JspWriter _jspx_out = null; javax.servlet.jsp.PageContext _jspx_page_context = null;


try {
response.setContentType("text/html;charset=UTF-8");
pageContext = _jspxFactory.getPageContext(this, request, response,
null, true, 8192, true);
_jspx_page_context = pageContext;
application = pageContext.getServletContext(); config = pageContext.getServletConfig(); session = pageContext.getSession();
out = pageContext.getOut();
_jspx_out = out;

out.write("\r\n");
out.write("\r\n"); out.write("<html>\r\n"); out.write("<head>\r\n");
out.write("	<title>Title</title>\r\n"); out.write("</head>\r\n"); out.write("<body>\r\n");
out.write("	a.jsp 页面\r\n");
out.write("</body>\r\n"); out.write("</html>\r\n");
} catch (java.lang.Throwable t) {
if (!(t instanceof javax.servlet.jsp.SkipPageException)){ out = _jspx_out;
if (out != null && out.getBufferSize() != 0)
try {
if (response.isCommitted()) { out.flush();
} else {
out.clearBuffer();
}
} catch (java.io.IOException e) {}
if (_jspx_page_context != null) _jspx_page_context.handlePageException(t);
else throw new ServletException(t);
}
} finally {
_jspxFactory.releasePageContext(_jspx_page_context);
}
}

JSP的三种语法

JSP头部的 page 指令

jsp 的 page 指令可以修改 jsp 页面中一些重要的属性,或者行为。

<%@ page contentType="text/html;charset=UTF-8" language="java" %>

  1. language 属性 表示 jsp 翻译后是什么语言文件。暂时只支持 java。
  2. contentType 属性 表示 jsp 返回的数据类型是什么。也是源码中 response.setContentType()参数值
  3. pageEncoding 属性 表示当前 jsp 页面文件本身的字符集。

以下属性是给out 输出流使用:

  • autoFlush 属性 设置当 out 输出流缓冲区满了之后,是否自动刷新冲级区。默认值是 true。
  • buffer 属性 设置out 缓冲区的大小。默认是 8kb

在这里插入图片描述

  • errorPage 属性 设置当 jsp 页面运行时出错,自动跳转去的错误页面路径。

  • isErrorPage 属性 设置当前jsp 页面是否是错误信息页面。默认是 false。如果是 true 可以获取异常信息。

  • session 属性 设置访问当前 jsp 页面,是否会创建HttpSession 对象。默认是true。

  • extends 属性 设置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 页面上输出数据。

表达式脚本的特点:

  1. 所有的表达式脚本都会被翻译到_jspService() 方法中
  2. 表达式脚本都会被翻译成为 out.print()输出到页面上
  3. 由于表达式脚本翻译的内容都在_jspService() 方法中,所以_jspService()方法中的对象都可以直接使用。
  4. 表达式脚本中的表达式不能以分号结束
<%=12 %> <br>
<%=12.12 %> <br>
<%="我是字符串" %> <br>
<%=map%> <br>
<%=request.getParameter("username")%>
代码脚本

代码脚本的格式是:

<%
 	java 语句
 	%>

代码脚本的作用是:可以在 jsp 页面中,编写我们自己需要的功能(写的是 java 语句)。

代码脚本的特点是:

  1. 代码脚本翻译之后都在_jspService 方法中
  2. 代码脚本由于翻译到_jspService()方法中,所以在_jspService()方法中的现有对象都可以直接使用。
  3. 还可以由多个代码脚本块组合完成一个完整的 java 语句。
  4. 代码脚本还可以和表达式脚本一起组合使用,在 jsp 页面上输出数据
<%
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);
 	%>
JSP中的三种注释
html注释
<!-- 这是html 注释 -->

html 注释会被翻译到 java 源代码中。在_jspService 方法里,以 out.writer 输出到客户端。

Java注释
<%
 	// 单行java 注释
 	/* 多行java 注释 */
%>

java 注释会被翻译到 java 源代码中。

JSP注释
<%-- 这是jsp 注释 --%>

jsp 注释可以注掉,jsp 页面中所有代码。

JSP九大内置对象

JSP 中的内置对象,是指 Tomcat 在翻译 jsp 页面成为 Servlet 源代码后,内部提供的九大对象,叫内置对象。

在这里插入图片描述

JSP四大域对象

对象名 有效范围
pageContext PageContextImpl 类 当前 jsp 页面范围内有效
request HttpServletRequest 类 一次请求内有效
session HttpSession 类 一个会话范围内有效(打开浏览器访问服务器,直到关闭浏览器)
application ServletContext 类 整个 web 工程范围内都有效(只要 web 工程不停止,数据都在)

域对象是可以像 Map 一样存取数据的对象。四个域对象功能一样。不同的是它们对数据的存取范围。

虽然四个域对象都可以存取数据。在使用上它们是有优先顺序的。

四个域在使用的时候,优先顺序分别是,他们从小到大的范围的顺序。

pageContext ====>>> request ====>>> session ====>>> application

scope.jsp 页面:

<%@ page contentType="text/html;charset=UTF-8"
         errorPage="/error.jsp"
         language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<h1>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")%>
</body>
</html>

scope2.jsp 页面

<%@ page contentType="text/html;charset=UTF-8"
         errorPage="/error.jsp"
         language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<h1>jsp2页面</h1>
    pageContext域的值是:<%=pageContext.getAttribute("key")%><br>
    request域的值是:<%=request.getAttribute("key")%><br>
    session域的值是:<%=session.getAttribute("key")%><br>
    application域的值是:<%=application.getAttribute("key")%>
</body>
</html>

JSP中的out输出和response.getWriter输出的区别

response 中表示响应,我们经常用于设置返回给客户端的内容(输出)

out 也是给用户做输出使用的。

在这里插入图片描述

由于 jsp 翻译之后,底层源代码都是使用 out 来进行输出,所以一般情况下。我们在 jsp 页面中统一使用 out 来进行输出。避免打乱页面输出内容的顺序。

out.write() 输出字符串没有问题

out.print() 输出任意数据都没有问题(都转换成为字符串后调用的 write 输出)

结论:在 jsp 页面中,可以统一使用 out.print()来进行输出

JSP的常用标签

JSP静态包含

<%--
<%@ include file=""%> 就是静态包含
file 属性指定你要包含的jsp 页面的路径
地址中第一个斜杠 / 表示为http://ip:port/工程路径/ 映射到代码的web 目录
--%>

<%@ include file="/include/footer.jsp"%>

静态包含的特点:

  1. 静态包含不会翻译被包含的jsp 页面。
  2. 静态包含其实是把被包含的jsp 页面的代码拷贝到包含的位置执行输出。

JSP动态包含

<%--
<jsp:include page=""></jsp:include>	这是动态包含
page 属性是指定你要包含的jsp 页面的路径
动态包含也可以像静态包含一样。把被包含的内容执行输出到包含位置
--%>
<jsp:include page="/include/footer.jsp">
<jsp:param name="username" value="bbj"/>
<jsp:param name="password" value="root"/>
</jsp:include>

动态包含的特点:

1.动态包含会把包含的jsp 页面也翻译成为java 代码

2.动态包含底层代码使用如下代码去调用被包含的jsp 页面执行输出。

JspRuntimeLibrary.include(request, response, “/include/footer.jsp”, out, false);

3.动态包含,还可以传递参数

在这里插入图片描述

JSP标签**-**转发

<%--
<jsp:forward page=""></jsp:forward> 是请求转发标签,它的功能就是请求转发
page 属性设置请求转发的路径
--%>
<jsp:forward page="/scope2.jsp"></jsp:forward>

九九乘法口诀表

<%@ page contentType="text/html;charset=UTF-8"
         errorPage="/error.jsp"
         language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<h1 align="center">乘法口诀</h1>
<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>

JSP输出一个表格,里面有 10 个学生信息

在这里插入图片描述

Student 类:

package com.jsp.pojo;

/**
 * ClassName: Student
 * Description:
 * date: 2022/1/24 17:57
 *
 * @author Ekertree
 * @since JDK 1.8
 */
public class Student {
    private Integer id;
    private String name;
    private Integer age;
    private String phone;

    public Student() {
    }

    public Student(Integer id, String name, Integer age, String phone) {
        this.id = id;
        this.name = name;
        this.age = age;
        this.phone = phone;
    }

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    public String getPhone() {
        return phone;
    }

    public void setPhone(String phone) {
        this.phone = phone;
    }

    @Override
    public String toString() {
        return "Student{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", age=" + age +
                ", phone='" + phone + '\'' +
                '}';
    }
}

SearchStudentServlet 程序:

public class SearchStudentServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) 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(1+t,"name_"+t, 18+t,"100860"+t));
        }
        //保存查询到的结果(学生信息)到request域中
        request.setAttribute("studentList",studentList);
        //请求转发到showStudent.jsp页面
        request.getRequestDispatcher("showStudent.jsp").forward(request, response);
    }
}

showStudent.jsp 页面:

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
  <head>
    <title>$Title$</title>
    <style type="text/css">
      table{
        border: 1px blue solid; width: 600px;
        border-collapse: collapse;
      }
      td,th{
        border: 1px blue solid;
      }
    </style>
  </head>
  <body>
  <%
    List<Student> students = (List<Student>) request.getAttribute("studentList");
  %>
  <table align="center">
    <tr>
      <td align="center">id</td>
      <td align="center">姓名</td>
      <td align="center">年龄</td>
      <td align="center">电话</td>
    </tr>
<%for (Student student : students) {%>
    <tr>
      <td align="center"><%=student.getId()%></td>
      <td align="center"><%=student.getName()%></td>
      <td align="center"><%=student.getAge()%></td>
      <td align="center"><%=student.getPhone()%></td>
    </tr>
<%}%>
  <table/>
  </body>
</html>

什么是 Listener 监听器?

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

ServletContextListener监听器

ServletContextListener 它可以监听 ServletContext 对象的创建和销毁。ServletContext 对象在 web 工程启动的时候创建,在 web 工程停止的时候销毁。监听到创建和销毁之后都会分别调用 ServletContextListener 监听器的方法反馈。

两个方法分别是:

 @Override
    public void contextInitialized(ServletContextEvent servletContextEvent) {
        System.out.println("对象被创建");
    }

    @Override
    public void contextDestroyed(ServletContextEvent servletContextEvent) {
        System.out.println("对象被销毁");
    }

如何使用 ServletContextListener 监听器监听 ServletContext 对象。

使用步骤如下:

  1. 编写一个类去实现 ServletContextListener
  2. 实现其两个回调方法
  3. 到 web.xml 中去配置监听器

监听器实现类:

public class MyServletContextListenerImpl implements ServletContextListener {

    @Override
    public void contextInitialized(ServletContextEvent servletContextEvent) {
        System.out.println("对象被创建");
    }

    @Override
    public void contextDestroyed(ServletContextEvent servletContextEvent) {
        System.out.println("对象被销毁");
    }
}

web.xml 中的配置:

<listener>
        <listener-class>com.jsp.listener.MyServletContextListenerImpl</listener-class>
</listener>
Logo

CSDN联合极客时间,共同打造面向开发者的精品内容学习社区,助力成长!

更多推荐