Java随记 —— Servlet 教程笔记

发布于:2023-01-20 ⋅ 阅读:(308) ⋅ 点赞:(0)

1. 概念

Servlet:server applet

  • Servlet 即运行在服务器端的小程序
  • Servlet 就是一个接口,定义了 Java 类被浏览器访问到(Tomcat识别)的规则。
  • 将来我们自定义一个类,实现 Servlet 接口,复写方法。

2.步骤

1. 创建 JavaEE 项目

在这里插入图片描述

2. 定义一个类,实现 Servlet 接口

package com.example.webdemo;  
  
import javax.servlet.*;  
import java.io.IOException;  
  
public class HelloServlet implements Servlet {  

}

3. 实现接口中的抽象方法

在这里插入图片描述

package com.example.webdemo;  
  
import javax.servlet.*;  
import java.io.IOException;  
  
public class HelloServlet implements Servlet {  
  
    @Override  
    public void init(ServletConfig servletConfig) throws ServletException {  
  
    }  
  
    @Override  
    public ServletConfig getServletConfig() {  
        return null;  
    }  
  
    @Override  
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {  
        System.out.println("Hello Servlet");  
    }  
  
    @Override  
    public String getServletInfo() {  
        return null;  
    }  
  
    @Override  
    public void destroy() {  
  
    }  
}

4. 配置 Servlet

在这里插入图片描述

  • <web-app> 根标签里配置 <servlet><servlet-mapping> 中的内容
  • 注意不要写到根标签外
<?xml version="1.0" encoding="UTF-8"?>  
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"  
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"  
         version="4.0">  
  
    <!--配置内容-->  
    <servlet>  
        <servlet-name>demo1</servlet-name>  
        <servlet-class>com.example.webdemo.HelloServlet</servlet-class>  
    </servlet>        
    
    <servlet-mapping>  
        <servlet-name>demo1</servlet-name>  
        <url-pattern>/demo1</url-pattern>  
    </servlet-mapping>  
    
</web-app>

3. 执行原理

配置 Tomcat 虚拟目录

  • 后期项目数量增加,配置 Tomcat 虚拟目录方便管理。
  • Run --> Edit Configurations... --> Tomcat 9.0.46 --> Deployment --> Application context

在这里插入图片描述

Servlet 执行原理

  1. 当服务器接收到客户端浏览器的请求后,会解析请求 URL路径,获取访问的 Servlet 的资源路径
  2. 查找 web.xml 文件,是否有对应的 <url-pattern> 标签体内容
  3. 如果有,则再找到对应的 <servlet-class> 全类名
  4. Tomcat 会将字节码文件加载进内存,并创建其对象
  5. 调用其方法

在这里插入图片描述

4. 生命周期

1. 被创建:执行 init 方法,只执行一次

Servlet 什么时候被创建 ?

  • 默认情况下,第一次被访问时,Servlet 被创建
    • 可以配置执行 Servlet 的创建时机。[[1. Servlet 教程#配置执行 Servlet 的创建时机]]

配置执行 Servlet 的创建时机

  • 配置 <load-on-startup>
      1. 第一次访问时,创建 --> <load-on-startup> 为负值
      1. 服务器启动时,创建 --> <load-on-startup> 为 0 或 正整数

<servlet>  
    <servlet-name>Demo1</servlet-name>  
    <servlet-class>com.example.webdemo.Demo1</servlet-class>  
  
    <!-- 指定 Servlet 的创建时机  
  
        1. 第一次被访问时,创建  
            * <load-on-startup> 的值为负值  
        2. 在服务器启动时,创建  
            * <load-on-startup> 的值为0或正整数 
    -->  
    <load-on-startup>5</load-on-startup>  
  
  
</servlet>

<load-on-startup> 为正值的结果。

在这里插入图片描述

Servlet 的线程安全问题

  • Servlet 的 init 方法,只执行一次,说明一个 Servlet 在内存中只存在一个对象,Servlet 是单例的。
    • 多个用户同时访问时,可能存在线程安全问题
    • 解决:尽量不要在 Servlet 中定义成员变量。即使定义了成员变量,也不要对其修改值
      • number 不共享(解决问题)在这里插入图片描述

2. 提供服务:执行 service 方法,执行多次

  • 每次访问 Servlet 时,Servlet 方法都会被调用一次

3. 被销毁:执行 destroy 方法,只执行一次

  • Servlet 被销毁时执行。 服务器关闭时,Servlet 被销毁
  • 只有服务器正常关闭时,才会执行 destroy 方法
  • destroy 方法在 Servlet 被销毁之前执行,一般用于释放资源(临终交代遗言)

代码

package com.example.webdemo;  
  
import javax.servlet.*;  
import java.io.IOException;  
  
public class HelloServlet implements Servlet {  
  
    /**  
     * 初始化方法  
     * 在 Servlet 被创建时,执行。只会执行一次  
     * 
     * @param servletConfig  
     * @throws ServletException  
     */    @Override  
    public void init(ServletConfig servletConfig) throws ServletException {  
        System.out.println("init......");  
    }  


    /**  
     * 提供服务方法  
     * 每一次Servlet被访问,执行。执行多次  
     * 
     * @return  
     */  
    @Override  
    public ServletConfig getServletConfig() {  
        return null;  
    }  
  
  
    /**  
     * 提供服务方法  
     * 每一次 Servlet 被访问是,执行,执行多次  
     * 
     * @param servletRequest  
     * @param servletResponse  
     * @throws ServletException  
     * @throws IOException  
     */    @Override  
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {  
        System.out.println("service......");  
    }  
  
      
    /**  
     * 获取 Servlet 的一些信息,版本,作者等等。。。  
     * 
     * @return  
     */  
    @Override  
    public String getServletInfo() {  
        return null;  
    }  
  
    /**  
     * 销毁方法  
     * 在服务器正常关闭是,执行,执行一次  
     * 
     */  
    @Override  
    public void destroy() {  
        System.out.println("destroy......");  
    }  
}

配置执行 Servlet 的创建时机

  • 配置 <load-on-startup>
      1. 第一次访问时,创建 --> <load-on-startup> 为负值
      1. 服务器启动时,创建 --> <load-on-startup> 为 0 或 正整数

<servlet>  
    <servlet-name>Demo1</servlet-name>  
    <servlet-class>com.example.webdemo.Demo1</servlet-class>  
  
    <!-- 指定 Servlet 的创建时机  
  
        1. 第一次被访问时,创建  
            * <load-on-startup> 的值为负值  
        2. 在服务器启动时,创建  
            * <load-on-startup> 的值为0或正整数 
    -->  
    <load-on-startup>5</load-on-startup>  
  
  
</servlet>

<load-on-startup> 为正值的结果。

在这里插入图片描述

5. Servlet 3.0

Servlet 3.0 的好处

支持注解配置。可以不需要 web.xml 了

步骤:

  1. 创建 JavaEE 项目,选择 Servlet 的3.0以上的版本,可以不创建 web.xml
  2. 定义一个类,实现 Servlet 接口
  3. 复写方法
  4. 在类上使用 @WebServlet 注解,进行配置

Servlet 3.0 注解配置方法

WebServlet("资源路径")

方法一

在类前加入 @WebServlet 注解并配置 urlPatterns 即可。

  • 示例
WebServlet(urlPatterns = "/demo1")

在这里插入图片描述

方法二

value 的特性: 表示最重要的属性

由于注解 @WebServlet 括号中只有一个属性,value 可以不写

示例:

@WebServlet(value = "/demo1")   //value 可以不写

@WebServlet("/demo1")

在这里插入图片描述

注意

  1. 从 Java EE 6 才开始支持 Web 3.0 --> 支持 Servlet 3.0

在这里插入图片描述

6. IDEA 与 Tomcat 的相关配置

  1. IDEA 会为每一个 Tomcat 部署的项目单独建立一份配置文件

    • 查看控制台的 log:Using CATALINA_BASE:
      • “C:\Users\fqy.IntelliJIdea2018.1\system\tomcat_itcast”
  2. 工作空间项目 和 Tomcat部署的web项目

    • Tomcat真正访问的是“tomcat部署的web项目”,
    • “Tomcat 部署的web项目"对应着"工作空间项目” 的web目录下的所有资源
    • WEB-INF目录下的资源不能被浏览器直接访问。
  3. 断点调试:使用"小虫子"启动 dubug 启动

6. Servlet 的体系结构

1. Servlet 的结构体系

Servlet – 接口

GenericServlet – 抽象类

HttpServlet – 抽象类

2. GenericServlet

代码 【GenericServlet.java】

//  
// Source code recreated from a .class file by IntelliJ IDEA  
// (powered by FernFlower decompiler)  
//  
  
package javax.servlet;  
  
import java.io.IOException;  
import java.io.Serializable;  
import java.util.Enumeration;  
import java.util.ResourceBundle;  
  
public abstract class GenericServlet implements Servlet, ServletConfig, Serializable {  
    private static final String LSTRING_FILE = "javax.servlet.LocalStrings";  
    private static ResourceBundle lStrings = ResourceBundle.getBundle("javax.servlet.LocalStrings");  
    private transient ServletConfig config;  
  
    public GenericServlet() {  
    }  
  
    public void destroy() {  
    }  
  
    public String getInitParameter(String name) {  
        ServletConfig sc = this.getServletConfig();  
        if (sc == null) {  
            throw new IllegalStateException(lStrings.getString("err.servlet_config_not_initialized"));  
        } else {  
            return sc.getInitParameter(name);  
        }  
    }  
  
    public Enumeration<String> getInitParameterNames() {  
        ServletConfig sc = this.getServletConfig();  
        if (sc == null) {  
            throw new IllegalStateException(lStrings.getString("err.servlet_config_not_initialized"));  
        } else {  
            return sc.getInitParameterNames();  
        }  
    }  
  
    public ServletConfig getServletConfig() {  
        return this.config;  
    }  
  
    public ServletContext getServletContext() {  
        ServletConfig sc = this.getServletConfig();  
        if (sc == null) {  
            throw new IllegalStateException(lStrings.getString("err.servlet_config_not_initialized"));  
        } else {  
            return sc.getServletContext();  
        }  
    }  
  
    public String getServletInfo() {  
        return "";  
    }  
  
    public void init(ServletConfig config) throws ServletException {  
        this.config = config;  
        this.init();  
    }  
  
    public void init() throws ServletException {  
    }  
  
    public void log(String msg) {  
        this.getServletContext().log(this.getServletName() + ": " + msg);  
    }  
  
    public void log(String message, Throwable t) {  
        this.getServletContext().log(this.getServletName() + ": " + message, t);  
    }  
  
    public abstract void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;  
  
    public String getServletName() {  
        ServletConfig sc = this.getServletConfig();  
        if (sc == null) {  
            throw new IllegalStateException(lStrings.getString("err.servlet_config_not_initialized"));  
        } else {  
            return sc.getServletName();  
        }  
    }  
}

由于我们常用的只有 service 方法,所以 GenericServlet 就将其它方法默认空实现,只把 service 方法抽象实现。

  • GenericServlet:将Servlet接口中其他的方法做了默认空实现,只将service()方法作为抽象
    • 将来定义Servlet类时,可以继承GenericServlet,实现service()方法即可
    • 需要其它方法直接复写即可。

3. HttpServlet

  • HttpServlet:对http协议的一种封装,简化操作
    1. 定义类继承 HttpServlet
    2. 复写 doGet/doPost 方法

在这里插入图片描述

7. Servlet 相关配置

  • urlpartten:Servlet访问路径
    1. 一个Servlet可以定义多个访问路径
      • @WebServlet({"/d4","/dd4","/ddd4"})
    2. 路径定义规则:
      1. /xxx :路径匹配 --> @WebServlet({"/user/demo4"})
      2. /xxx/xxx :多层路径,目录结构 --> @WebServlet("/user/demo4")
      3. *.do :扩展名匹配 --> @WebServlet("/*")
本文含有隐藏内容,请 开通VIP 后查看