JavaWeb学习第七章——JSP

1 篇文章 0 订阅

JSP概述

JSP的全称是 java server pages ( Java 的服务器页面)。jsp的主要作用是 代替Servlet程序回传html页面的数据 。因为Servlet程序回传html 页面数据是一件非常繁锁的事情,开发成本和维护成本都高。
Servlet 回传 html 页面数据的代码:(回传的网页内容写在servlet中)
public class PringHtml extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException,IOException {
    //设置回传的字符格式,,避免乱码
    resp.setContentType("text/html; charset=UTF-8");
    //通过响应的回传流回传html页面数据
    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");
    }
}

由上可看出Servlet回传网页数据很繁琐. 

创建jsp的页面

输入需要创建的jsp文件名敲回车即可

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

比如:
在web目录下有如下的文件:
web目录
a.html页面访问地址是 =======>>>>>> http://ip:port/工程路径/a.html
b.jsp页面访问地址是 =======>>>>>> http://ip:port/工程路径/b.jsp

jsp的本质 

jsp 页面本质上是一个 Servlet 程序。
当我们第一次访问jsp页面的时候。 Tomcat 服务器会帮我们把 jsp 页面翻译成为一个 java 源文件,并且对它进行编译成为.class字节码程序。我们打开java 源文件不难发现其里面的内容是:C:\Users\admin\AppData\Local\JetBrains\IntelliJIdea2020.1\tomcat\Tomcat_9_0_0_M18_JavaWeb_7\work\Catalina\localhost\08_jsp\org\apache\jsp

我们跟踪原代码发现, 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())) {
      response.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, "JSPs only permit GET POST or HEAD");
      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 页面文件本身的字符集。
4. import 属性跟 java 源代码中一样。用于导包,导类。
========================两个属性是给out 输出流使用 ============================
autoFlush 属性设置当 out 输出流缓冲区满了之后,是否自动刷新冲级区。默认值是 true
6. buffer 属性 设置 out 缓冲区的大小。默认是 8kb
缓冲区溢出错误:

其他属性:

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

<!--errorPage 表示错误后自动跳转去的路径 <br/>
这个路径一般都是以斜杠打头,它表示请求地址为 http://ip:port/工程路径/
映射到代码的 Web 目录-->

8. isErrorPage 属性设置当前jsp页面是否是错误信息页面。默认是false。如果是true可以获取异常信息。
9. session 属性设置访问当前jsp页面,是否会创建HttpSession对象。默认true。
10. extends 属性设置jsp翻译出来的java类默认继承谁。

jsp中的常用脚本

1. 声明脚本(极少使用)

声明脚本的格式是: <%! 声明 java 代码 %>
作用:可以给jsp翻译出来的 java 类定义属性和方法甚至是静态代码块,内部类等。
练习:
1、声明类属性
2、声明 static 静态代码块
3、声明类方法
4、声明内部类
代码示例:
<%@ page import="java.util.Map" %>
<%@ page import="java.util.HashMap" %><%--自动导的包--%>

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
这是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";
    }
    %>
</body>
</html>
jsp中声明脚本代码在
C:\Users\admin\AppData\Local\JetBrains\IntelliJIdea2020.1\tomcat\Tomcat_9_0_0_M18_JavaWeb_7\work\Catalina\localhost\08_jsp\org\apache\jsp.a_jsp.java中翻译对照:

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

翻译对照:

 iii. 代码脚本

代码脚本的格式是:<%java 语句%>
代码脚本的作用是:可以在jsp页面中,编写我们自己需要的功能(写的是 java 语句)。
代码脚本的特点是:
    1、代码脚本翻译之后都在_jspService 方法中
    2、代码脚本由于翻译到_jspService()方法中,所以在_jspService()方法中的现有对象都可以直接使用。
    3、
还可以由多个代码脚本块组合完成一个完整的java语句
    4、代码脚本还可以和表达式脚本一起组合使用,在 jsp 页面上输出数据

练习:
1. 代码脚本----if语句
2. 代码脚本----for循环语句
3. 翻译后java文件中_jspService方法内的代码都可以写
示例代码:

<%--1.代码脚本--if--语句--%>
    <%
        int i = 13 ;
        if (i == 12) {
    %>
        <h1>国哥好帅</h1> //可以直接在页面输出
    <%
        } else {
     %>
        <h1>国哥又骗人了!</h1>
    <%
        }
    %>
<br>

<%--2.代码脚本----for--循环语句--%>//效果:输出10行
    <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);
    %>

翻译之后的对比:

c)jsp中的三种注释

i. html 注释
<!-- 这是html注释 -->

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

ii. java 注释

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

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

iii. jsp 注释
<%-- 这是 jsp 注释 --%>
jsp注释可以注掉 jsp 页面中所有代码。

JSP对象

九大内置对象 

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

四大域对象 

四个域对象分别是:
  pageContext (PageContextImpl 类)         当前jsp 页面范围内有效
  request (HttpServletRequest 类)              一次请求内有效
  session (HttpSession 类)                         一个会话范围内有效(打开浏览器访问服务器,直到关闭浏览器)
  application (ServletContext 类)                整个 web 工程范围内都有效(只要 web 工程不停止,数据都在)
scope.jsp 页面
<body>
    <h1>scope.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>

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

虽然四个域对象都可以存取数据。在使用上它们是有优先顺序的。四个域在使用的时候,优先顺序分别是,他们从小到大的范围的顺序。

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

out 输出和 response.getWriter输出的区别
response 中表示响应,我们经常用于设置返回给客户端的内容(输出) out 也是给用户做输出使用的。
<body>
<%
 out.write("out输出1<br/>");
 out.write("out输出2<br/>");

 response.getWriter().write("response1输出1<br/>");
 response.getWriter().write("response2输出1<br/>");
%>
</body>

结果:

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

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

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

jsp的常用标签

 jsp静态包含

<%@ include file=""%> 就是静态包含
file 属性指定你要包含的 jsp 页面的路径
地址中第一个斜杠 / 表示为 http://ip:port/工程路径/ 映射到代码的 web 目录
main.jsp中
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
    头部信息<br/>
    主体内容<br/>
    <%@include file="/include/footer.jsp"%>
</body>
</html>

footer.jsp中
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
页脚信息<br/>
</body>
</html>
静态包含的特点:在.java翻译文件中
1、静态包含不会翻译被包含的jsp页面(不会生成被包含的.class文件)。
2、静态包含其实是把被包含的jsp页面的代码拷贝到包含的位置执行输出。

jsp动态包含

<jsp:include page="/include/footer.jsp"></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>

jsp标签-转发 

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

< jsp :forward page ="/scope2.jsp" >jsp:forward>
练习
jsp 页面中输出九九乘法口诀表
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
    <style type="text/css">
        table{
            width:650px;
        }
    </style>
</head>
<table>
    <h1 align="center">九九乘法表</h1>
    <%for (int i = 1; i <= 9; i++) {%>
    <tr>
        <%
        for (int j = 1; j <= i; j++) {%>
        <td><%=j+"X"+i+"="+j*i%></td>
        <%}%>
    </tr>
    <%}%>
</table>
<body>
</body>
</html>
jsp 输出一个表格,里面有 10 个学生信息

使用jsp输出返回的信息

Student 类:
public class Student {
    private Integer id;
    private String name;
    private Integer age;
    private String phone;
    ... ...
}
SearchStudentServlet.java
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.LinkedList" %>
<%@ page import="pojo.Student" %>
<%@ page import="java.util.ArrayList" %>
<%@ page import="java.util.List" %><%--
  Created by IntelliJ IDEA.
  User: admin
  Date: 2021/9/1
  Time: 22:40
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
    <style>
        table{
            border: 1px red solid;
            width:600px;
            border-collapse: collapse;
        }
        td,th{
            border:1px red solid;
        }
    </style>
</head>
<body>
    <%
        List<Student> list = new ArrayList<Student>();
        for (int i = 0; i < 10; i++) {
            int j = i+1;
            list.add(new Student(j,"name"+j,18+j,"phone"+j));
        }
    %>

    <table>
        <tr>
            <td>编号</td>
            <td>姓名</td>
            <td>年龄</td>
            <td>电话</td>
            <td>操作</td>
        </tr>
        <%for(Student stu:list){%>
        <tr>
            <td><%=stu.getId()%></td>
            <td><%=stu.getName()%></td>
            <td><%=stu.getAge()%></td>
            <td><%=stu.getPhone()%></td>
            <td>删除、修改</td>
        </tr>
        <%}%>
    </table>

</body>
</html>

Listener监听器

1 Listener 监听器它是 JavaWeb 的三大组件之一。 JavaWeb 的三大组件分别是: Servlet 程序、 Filter 过滤器、 Listener 监听器。
2、Listener 它是 JavaEE 的规范,就是接口
3、监听器的作用是,监听某种事物的变化。然后通过回调函数,反馈给客户(程序)去做一些相应的处理。
ServletContextListener 监听器
ServletContextListener 它可以监听 ServletContext 对象的创建和销毁。
ServletContext 对象在web 工程启动的时候创建,在 web 工程停止的时候销毁。
监听到创建和销毁之后都会分别调用ServletContextListener监听器的方法反馈。

两个方法分别是:

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 中去配置监听器

监听器实现类:

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>


 

 

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值