Servlet学习详解--基本涵盖所有Servlet知识点

news2025/1/13 2:18:51

目录

    • 一、Servlet
    • 二、 Servlet入门
      • 2.1. 执行原理
      • 2.2. 实现Servlet接口重写其五个方法及其生命周期
    • 三、Request请求对象
      • 3.1. 获取请求消息数据
      • 3.2. 获取请求头数据
      • 3.3. 获取请求体数据
      • 3.4. 设置编码
      • 3.5. 其他通用功能
      • 3.6. 请求转发(Forward)
      • 3.7. 转发共享数据
    • 四、Response对象
      • 4.1. response常用方法
      • 4.2. 重定向
    • 五、ServletContext对象
      • 5.1 ServletContext对象获取方式
      • 5.2 获取MIME类型
      • 5.2 共享数据
    • 六、Cookie对象
      • 6.1 Cookie介绍
      • 6.2 Cookie实现原理
      • 6.3 实现方式
      • 6.4 Cookie的细节
    • 七、Session对象
      • 7.1 Session介绍
      • 7.2 实现原理
      • 7.3 Session实现
      • 7.4 Session的细节
      • 7.5 Session的特点
    • 八、过滤器
      • 8.1 基本介绍
      • 8.2 过滤器实现原理
      • 8.3 实现步骤
      • 8.4 过滤器的生命周期
      • 8.5 过滤器配置
      • 8.6 过滤链(多个过滤器)

一、Servlet

  1. Servlet(Server Applet),全称Java Servlet。是用Java编写的服务器端程序,其主要功能在于交互式地浏览和修改数据,生成动态Web内容。狭义的Servlet是指Java语言实现的一个接口,广义的Servlet是指任何实现了这个Servlet接口的类,一般情况下,人们将Servlet理解为后者。
  2. Servlet运行于支持Java的应用服务器中。从实现上讲,Servlet可以响应任何类型的请求,但绝大多数情况下,Servlet只用来扩展基于HTTP协议的Web服务器
    在这里插入图片描述
  3. Servlet工作模式:
    ① 客户端发送请求至服务器
    ②服务器启动并调用Servlet,Servlet根据客户端请求生成响应内容并将其传给服务器
    ③ 服务器将响应返回客户端

在这里插入图片描述
在这里插入图片描述

二、 Servlet入门

编写Servlet类

//@WebServlet("/demo03")
//使用继承Servlet接口的方式
public class ServletDemo01 implements Servlet {
    @Override
    public void init(ServletConfig servletConfig) throws ServletException {
        System.out.println("这里是初始化方法");
    }
    @Override
    public ServletConfig getServletConfig() {

        return null;
    }
    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
        System.out.println("service执行了......");
    }
    @Override
    public String getServletInfo() {
        return "()";
    }
    @Override
    public void destroy() {
        System.out.println("这里是销毁方法");
    }
}

//@WebServlet("/demo02")
//使用继承GenericServlet类的方式
public class ServletDemo02 extends GenericServlet {
    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
        System.out.println("service方法执行了");
    }
}

//@WebServlet("/demo03")
//使用继承HttpServlet的方式
public class ServletDemo03 extends HttpServlet {

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("执行了doget");
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {

    }
}

web.xml配置文件

<servlet>
   	<!--Servlet的自定义名称-->
    <servlet-name>demo01</servlet-name>
    <!--Servlet资源的全限定路径-->
    <servlet-class>com.aoyezongguanjun.servlet.ServletDemo01</servlet-class>
</servlet>
<servlet-mapping>
  	<!--资源对应的Servlet-->
    <servlet-name>demo01</servlet-name>
    <!--资源路径-->
    <url-pattern>/demo01</url-pattern>
</servlet-mapping>

<servlet>
    <servlet-name>demo02</servlet-name>
    <servlet-class>com.tencent.servlet.ServletDemo02</servlet-class>
</servlet>
<servlet-mapping>
    <servlet-name>demo02</servlet-name>
    <url-pattern>/demo02</url-pattern>
</servlet-mapping>

<servlet>
    <servlet-name>demo03</servlet-name>
    <servlet-class>com.tencent.servlet.ServletDemo03</servlet-class>
</servlet>
<servlet-mapping>
    <servlet-name>demo03</servlet-name>
    <url-pattern>/demo03</url-pattern>
</servlet-mapping>

2.1. 执行原理

  1. 当服务器收到客户端发送的请求后,会解析对应的URL路径,获取访问的Servlet的资源路径名称
  2. tomcat会查询web.xml配置文件,<url-pattern>内配置对应Servlet资源路径名称,并且根据<servlet-name>的配置拿到Servlet类的全限定类名
  3. tomcat根据Servlet的全限定类名通过反射获取Servlet对象,然后调用其方法。

2.2. 实现Servlet接口重写其五个方法及其生命周期

  1. init( )初始化方法,当访问该servlet时会调用一次。注:该方法只会被调用一次,通常用于加载资源。我们在<servlet>标签下配置<load-on-startup>标签可以改变init()执行的时机,当配置的是0或正整数时,第一次被访问时才执行,当配置负数时,服务器启动时,就会执行该方法。由init()方法只被执行一次说明Servlet是单例的,可能存在线程安全问题。
  2. service()处理或响应客户端请求时执行,每访问都会调用该方法。
  3. destroy()销毁方法,服务器正常关闭时,会调用该方法,通常用于释放资源,该方法只会被调用一次
  4. getServletInfo():该方法用于获取 servlet 容器的名称和当前版本的信息。
  5. getServletConfig( ) :获取servlet配置对象。
  6. @WebServlet注解
    在Servlet3.0后支持注解配置Servlet,不再需要web.xml文件了,可以使用该注解将进行配置,配置方式@WebServlet("资源路径")
  7. HttpServlet的doGet和doPost的执行原理。
    HttpServlet继承了GenericServlet类,实质还是重写了Service方法,只是会根据不同的请求方式调用不同的方法。
    在这里插入图片描述

三、Request请求对象

Request对象代表客户端的请求,当客户端通过HTTP协议访问服务器时,HTTP请求头中的所有信息都封装在这个对象中,开发人员通过这个对象的方法,可以获得客户这些信息。简单来说,要得到浏览器信息,就找HttpServletRequest对象。

在这里插入图片描述

3.1. 获取请求消息数据

  1. 获取请求方式:String getMethod()
  2. 获取请求虚拟路径:String getContextPath()
  3. 获取servlet路径:String getServletPath()
  4. 获取get请求方式参数:String getQueryString()
  5. 获取请求URI:String getRequestURI()
  6. 获取请求URL:StringBuffer getRequestURL()
  7. 获取协议及版本:String getProtocol()
  8. 获取客户机的IP地址:String getRemoteAddr()

3.2. 获取请求头数据

  1. 获取请求头:String getHeader(String name)
  2. 获取所有请求头的名称: Enumeration<String> getHeaderNames()
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    	//请求地址:http://localhost:8080/contextPath/demo03?username=zhangsan
        String method = req.getMethod();
        System.out.println(method);//GET
    
        String contextPath = req.getContextPath();
        System.out.println(contextPath);// /contextPath
    
        String servletPath = req.getServletPath();
        System.out.println(servletPath);//    /demo03
    
        String queryString = req.getQueryString();
        System.out.println(queryString);//    username=zhangsan
    
        String requestURI = req.getRequestURI();
        System.out.println(requestURI);//    /contextPath/demo03
    
        StringBuffer requestURL = req.getRequestURL();
        System.out.println(requestURL.toString());//    http://localhost:8080/contextPath/demo03
    
        String protocol = req.getProtocol();
        System.out.println(protocol);//    HTTP/1.1
    
        String remoteAddr = req.getRemoteAddr();
        System.out.println(remoteAddr);//    0:0:0:0:0:0:0:1,由于这里是本地调用,如果是其他客户端调用会返回对应的IP地址
    
        String header = req.getHeader("user-Agent");
        System.out.println(header);//    PostmanRuntime/7.26.8
        
        Enumeration<String> headerNames = req.getHeaderNames();
        List<String> headerNameList = new ArrayList<>();
        while (headerNames.hasMoreElements()){
            headerNameList.add(headerNames.nextElement());
        }
        System.out.println(String.join(";",headerNameList));//user-agent;accept;postman-token;host;accept-encoding;connection
    }
    

3.3. 获取请求体数据

  1. BufferedReader getReader():获取字符输入流,只能操作字符数据
  2. ServletInputStream getInputStream():获取字节输入流,可以操作所有类型数据

3.4. 设置编码

  1. setCharacterEncoding("utf-8")

3.5. 其他通用功能

获取请求头参数通用方式,不论get还是post请求方式都可以使用下列方法来获取请求参数

  1. 根据参数名称获取对应的参数值:String getParameter(String name)
  2. 根据参数名称获取参数值的数组:String[] getParameterValues(String name)
  3. 获取所有请求的参数名称:Enumeration<String> getParameterNames()
  4. 获取所有参数的map集合: Map<String,String[]> getParameterMap()

3.6. 请求转发(Forward)

  1. 工作原理:转发是在服务器内部进行的,用户发送的请求由一个Servlet接收后,可以将请求内部转发给另一个Servlet或JSP处理,最终将处理结果返回给用户。对于用户而言,他们只发出了一次请求,也只收到了一次响应
  2. URL保持不变:因为转发是在服务器内部进行的,所以用户浏览器的地址栏URL不会发生变化
  3. 性能较高:转发不涉及到客户端的重新请求,因此通常比重定向快。
  4. 共享请求数据:在转发过程中,原始请求和响应对象保持不变,可以共享请求中的数据
  5. 实现步骤:
    1. 通过request对象获取请求转发器对象:RequestDispatchergetRequestDispatcher(String path)
    2. 使用RequestDispatcher对象来进行转发:forward(ServletRequest request,ServletResponse response)
    3. 请求转发的特点:浏览器地址栏路径不发生变化;只能转发到当前服务器内部资源中;转发是一次请求。
      req.getRequestDispatcher("/demo01").forward(req,resp);
      

3.7. 转发共享数据

  1. 域对象:一个有作用范围的对象,可以在范围内共享数据
    在这里插入图片描述
    当浏览器访问浏览器时,如果服务器需要进行请求转发,服务器会转发到指定资源,注意这次是服务器转发的,用户端浏览器没有任何感知,如果需要在转发时共享数据,可以先将数据存储在Servlet转发域中,将来转发请求指定资源时,可以去转发域中获取共享的数据。转发资源响应后,再响应到浏览器中,该转发共享域也随之销毁。

  2. Request域:一般用于请求转发的多个资源中共享数据,Request域的作用范围是一次请求。

  3. 方法:

    • void setAttribute(String name,Object obj):存储数据
    • Object getAttitude(String name):通过键获取值
    • void removeAttribute(String name):通过键移除键值对

四、Response对象

Response是Web应用程序用来封装向客户端响应信息的,是Servlet接口的service()方法的一个参数,类型为javax.servlet.http.HttpServletResponse。客户端每次发送请求时,服务器都会创建一个Response对象,并传递给Servlet接口的service()方法,来完成向客户端的响应工作。
在这里插入图片描述

4.1. response常用方法

  1. 设置响应状态码:
    • setStatus(int httpStatusCode)
  2. 设置响应头
    • setHeader(String name,String value)
      //以附件形式打开响应体,可以进行文件下载,并且指定文件名称
      resp.setHeader("content-disposition","attachment;filename=xxxxxx");
      
    • setContentType(String MIME)
      //设置content-type响应头,告诉浏览器,以json的格式进行解析,且编码方式是utf-8
      resp.setContentType("text/json;charset=utf-8");
      
  3. 设置响应体
    1. 获取字符输出流
      • PrintWriter getWriter(),获取的流的默认编码方式是ISO-8859-1
      /**
      * 输出字符到浏览器
      */
       @Override
      protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
         //表示将response的流编码方式为utf-8,且告诉浏览器以json格式解析数据,并且浏览器也是以utf-8编码方式解析数据
      resp.setContentType("text/json;charset=utf-8");
      User user = new User();
      user.setUsername("张三");
      String json = JSONArray.toJSONString(user);
      resp.getWriter().println(json);
      }
      
    2. 获取字节输出流
      • ServletOutputStream getOutputStream()
      /**
      * 实现文件下载
      */
       protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws IOException {
              ServletContext servletContext = req.getServletContext();
              String imagePath = servletContext.getRealPath("/img/JR史密斯.jpg");
              String mimeType = servletContext.getMimeType(imagePath);
              resp.setHeader("content-type",mimeType);
              //读取文件,获取文件输出流
              InputStream inputStream = new FileInputStream(imagePath);
              //获取response字节流
              ServletOutputStream respOutputStream = resp.getOutputStream();
              //建立缓冲区
              byte[] buffer = new byte[1024 * 8];
              int len = 0;
              //读写文件
              while ((len = inputStream.read(buffer)) != -1){
                  respOutputStream.write(buffer,0,len);
              }
              inputStream.close();
              String fileName = URLEncoder.encode("JR史密斯" + System.currentTimeMillis(), "UTF-8").replaceAll("\\+", "%20");
              //设置响应体
              resp.setHeader("Content-disposition", "attachment;filename*=utf-8''" + fileName + ".jpg");
          }
      

4.2. 重定向

在Java Web开发中,重定向(Redirect)是一种常见的技术,用于将用户从一个URL地址自动重定向到另一个URL地址。这在很多情况下都非常有用,例如在用户登录后将其重定向到其个人资料页面,或者在进行某些操作后将其重定向到一个感谢页面。它通常用于以下情况:

  • 将用户从一个页面引导到另一个页面。
  • 更改或更新URL以反映新的资源位置。
  • 处理用户登录后的跳转。

重定向可以是临时的或永久的。临时重定向(HTTP状态码为302)通常用于暂时将用户导向另一个地址,而永久重定向(HTTP状态码为301)则表示资源已永久移动到新的URL地址。

在这里插入图片描述
浏览器请求服务器指定资源,如果该资源需要进行重定向,会将这次请求的响应码设置为302,且响应头location设置为重定向路径地址。当浏览器收到响应后,会解析location设置的重定向路径地址,然后重新发起一次新的请求去请求指定资源。

重定向实现:

@WebServlet("/AServletDemo ")
public class AServletDemo extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//		resp.setStatus(302);
//		resp.setHeader("location",req.getContextPath()+"/BServletDemo");
		//或者使用sendRedirect方法 
		//跳转路径规则:判断定义的路径是给谁用的?判断请求将来从哪儿发出 
		//给客户端浏览器使用:需要加虚拟目录(项目的访问路径)
		//给服务器使用:不需要加虚拟目录
		resp.sendRedirect(req.getContextPath()+"/BServletDemo");
    }
}

重定向的特点:

  1. 重定向是两次请求,两次响应。
  2. 重定向前后,浏览器的地址栏会发生变化
  3. 重定向前后的request对象不是同一个,所以不能使用request对象来共享域。
  4. 重定向前后的两个资源可以是来自不同的web应用,甚至可以是不同的服务器。

五、ServletContext对象

ServletContext是Servlet规范中的一个对象,它代表了当前Web应用程序的上下文(Context)。这个上下文包括了整个Web应用程序的信息,可以被Web应用中的所有Servlet共享。可以将ServletContext看作是一个全局存储区,用于存储和访问Web应用中的全局数据和资源。可以读取全局配置参数,搜索当前工程目录下面的资源文件等。

5.1 ServletContext对象获取方式

  1. 通过Request对象获取
ServletContext servletContext = req.getServletContext();
  1. 通过HttpServlet获取
@WebServlet("/demo04")
public class ServletDemo04 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) {
        ServletContext servletContext = this.getServletContext();
    }

5.2 获取MIME类型

MIME类型:在互联网通信过程中定义的一种文件数据类型,浏览器通常使用MIME类型(而不是文件扩展名)来确定如何处理URL,因此Web服务器在响应头中添加正确的MIME类型非常重要。如果配置不正确,浏览器可能会曲解文件内容,网站将无法正常工作,并且下载的文件也会被错误处理。比如给浏览器返回Json格式数据,MIME类型就是application/json,给浏览器返回excel的xlsx格式表格,MIME类型就是application/vnd.openxmlformats-officedocument.spreadsheetml.sheet

  • 实现方法:String getMimeType(String fileName)
     String mimeType = this.getServletContext().getMimeType("列表.xlsx");
    

5.2 共享数据

  1. ServletContext域:在整个Web应用程序中共享数据,所有Servlet都可以访问。
  2. 一旦获得了ServletContext对象,您可以使用其域对象来存取数据。这可以通过以下方法之一来完成
    • setAttribute(String name,Object value):其中name是存储数据的名称,value是数据的值。

      servletContext .getServletContext().setAttribute("name","zhangsan");
      
    • getAttribute(String name):获取数据。

      String name = (String)servletContext.getServletContext().getAttribute("name");
      
    • removeAttribute(String name):删除数据

      servletContext.getServletContext().removeAttribute("name");
      
    • 使用init-param元素在web.xml文件中配置全局参数,以供整个Web应用程序使用。

  3. ServletContext共享域生命周期

    ServletContext域的生命周期与Web应用程序的生命周期相同。它在Web应用程序启动时创建,而在Web应用程序关闭时销毁。这意味着在Web应用程序启动期间存储的数据将在整个应用程序的生命周期内保持不变。由于ServletContext域的数据在整个Web应用程序中可见,因此要小心确保不要意外覆盖或混淆数据。

  4. 获取文件(服务器文件)的真实路径
    • String getRealPath(String path)
      //web目录下资源访问
      String b = servletContext.getRealPath("/b.txt");
      //WEB-INF目录下的资源访问
      String c = servletContext.getRealPath("/WEB-INF/c.txt");
      //src目录下的资源访问
      String a = servletContext.getRealPath("/WEB-INF/classes/a.txt");
      

六、Cookie对象

6.1 Cookie介绍

Cookie 是一种在网站和应用程序中用于存储用户信息的小型文本文件。在一次会话的范围内的多次请求间,共享数据。当用户访问一个网站或应用程序时,该网站或应用程序会将一个包含用户信息的 Cookie 发送到用户的浏览器。浏览器会将该 Cookie 存储在用户的计算机上,并在以后的访问中将该 Cookie 发送回网站或应用程序。
虽然 Cookie 对于提供个性化体验和方便用户来说非常有用,但它们也引发了一些隐私和安全问题。例如,第三方 Cookie 可以用于跟踪用户在多个网站上的活动,可能会侵犯用户的隐私。出于隐私和安全的考虑,现代浏览器通常允许用户控制哪些 Cookie 被接受和存储,并提供了清除 Cookie 的选项。
会话:浏览器第一次给服务器资源发送请求,会话建立,直到有一方断开为止,一次会话中包含多次请求和响应。

6.2 Cookie实现原理

在这里插入图片描述
浏览器第一次请求服务器,服务器如果需要设置Cookie,会在响应头添加set-cookie的响应头,值为键值对形式。如set-cookie:key=name。当浏览器收到服务器响应,会去查询是否有set-cookie的响应头,如果有会将响应头的值存储在浏览器中。当浏览器再次访问该服务器时,会自定将cookie以请求头的格式发送给服务器,具体格式为cookie:key=value。服务器收到请求后,如果需要获取cookie,可以解析请求头进行获取。

6.3 实现方式

添加cookie

  @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //创建cookie对象
        Cookie cookie1 = new Cookie("username","zhangsan");
        Cookie cookie2 = new Cookie("username2","zhangsan");
        //添加到响应头
        resp.addCookie(cookie1);
        resp.addCookie(cookie2);
    }

获取cookie

 @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //获取cookie
        Cookie[] cookies = req.getCookies();
        for (Cookie cookie : cookies) {
            System.out.println(cookie.getName()+"="+cookie.getValue());
        }
    }

6.4 Cookie的细节

  1. Cookie一次可以创建多个,创建多个cookie后请求头携带Cookie的格式为Cookie:username=zhangsan; username2=zhangsan
  2. Cookie的存活时长默认是浏览器关闭后,Cookie数据被销毁,我们也可以使用cookie.setMaxAge(int seconds)执行cookie的存活时间,当浏览器关闭后,会自动持久化到浏览器中,其中seconds参数正数表示存活时长(s)。负数(默认值)表示浏览器关闭后自动销毁。零表示立即删除cookie消息。
  3. 在tomcat8之前,Cookie中不能直接存储中文数据,一般需要使用URL编码,在tomcat8之后,cookie支持中文数据,但是特殊字符还是不支持,建议使用URL编码存储,使用URL编码解析。
  4. 默认情况下一个服务器下多个web项目Cookie是不能共享的,如果需要共享可以使用cookie.setPath(String path)设置Cookie的获取范围,默认情况是设置当前的虚拟目录,如果需要共享可以将path设置为"/"。如果多个tomcat服务器之间需要共享Cookie,可以使用cookie.setDomain(String path)实现,如果设置一级域名相同,那么可以实现共享Cookie。如 setDomain(“.baidu.com”),那么tieba.baidu.com和news.baidu.com中cookie可以共享。
  5. 浏览器对于单个Cookie大小限制4kb,对于同一个域名下cookie总数限制20个
  6. 由于Cookie是存储在用户浏览器中,所以数据安全性并不高。一般用于存储用户少量不太敏感的数据,比如用户的登录信息和电商平台在不登陆的情况下加入购物车等功能。

七、Session对象

7.1 Session介绍

session在网络应用中称为“会话控制”,是服务器为了保存用户状态而创建的一个特殊的对象。简而言之,session就是一个对象,用于存储信息。 Session类似于一个Map,里面可以存放多个键值对,是以key-value进行存放的。key必须是一个字符串,value是一个对象

7.2 实现原理

在这里插入图片描述
当浏览器第一次发送请求时,服务器会自动创建一个Session对象放到内存中,并且指定唯一sessionId,响应时服务器会将这个唯一的sessionId以响应cookie的方式响应给浏览器并且储存在浏览器。当浏览器第二次访问服务器时,服务器会解析请求头是否携带这个cookie,如果携带会根据cookie的sessionId去寻找执行sessionId的Session对象。

7.3 Session实现

向Session中存储数据。

 HttpSession session = req.getSession();
 List<User> users = new ArrayList<User>();
 User user = new User();
 user.setUsername("use1");
 users.add(user);
 session.setAttribute("users",users);

向Session中获取数据

 HttpSession session = req.getSession();
 List<User> userList = (List<User>) session.getAttribute("users"); 

7.4 Session的细节

  1. 当客户端关闭后,两次获取的Session并不是同一个对象,如果需要获取的是同一个对象,可以创建Cookie,键为JSESSIONID,设置最大存活时间,让cookie持久化保存。

    Cookie c = new Cookie("JSESSIONID",session.getId());
    c.setMaxAge(60*60);
    response.addCookie(c);
    
  2. 当浏览器关闭时,为防止session丢失,tomcat在关闭之前,会将Session钝化,即将Session对象存储到硬盘上,在tomcat启动后,可以将其活化,即读取硬盘重新加载到内存中。

  3. Session在30分钟后默认失效,或者使用session的invalidate()方法对session里的内容进行清空。我们也可以修改session的默认失效时间

    <session-config>
       <session-timeout>30</session-timeout>
    </session-config>
    

7.5 Session的特点

  1. session用于存储一次会话的多次请求,是存储在服务器中的。
  2. session相对cookie来说相对安全
  3. session的数量个数和数据的大小没有限制。

八、过滤器

8.1 基本介绍

过滤器,顾名思义就是对事物进行过滤的,在Web中的过滤器,当然就是对请求进行过滤,我们使用过滤器,就可以对请求进行拦截,然后做相应的处理,实现许多特殊功能。如登录控制,权限管理,过滤敏感词汇等。

8.2 过滤器实现原理

在这里插入图片描述

当服务器接受到浏览器发送请求,会根据请求资源路径和过滤器@WebFilter配置的拦截进行匹配,当匹配不通过会绕过拦截器直接访问目标资源。当匹配通过时进入过滤器执行doFilter方法,这里可以进行一些前置逻辑判断,当符合某些放行规则后,可以使用filterChain.doFilter(request,response)方法进行放行,访问完目标资源后,再次进入过滤器执行放行后的代码,最后响应浏览器。细节:过滤器Filter是在请求进入容器后,但在进入servlet之前进行预处理,请求结束是在servlet处理完以后。

8.3 实现步骤

  • 基于注解配置过滤器
@WebFilter("/user/*")
public class ServletFilter1 implements Filter {
	//初始化方法
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        Filter.super.init(filterConfig);
    }

	//拦截方法
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) throws IOException, ServletException {
        System.out.println("ServletFilter1拦截器进行拦截了......");
        //放行
        filterChain.doFilter(request,response);
        System.out.println("执行之后......");
    }

	//销毁方法
    @Override
    public void destroy() {
        Filter.super.destroy();
    }
}
  • 基于web.xml配置过滤器
<filter>
     <filter-name>demo1</filter-name>
     <filter-class>com.aoyezongguanjun.filter.ServletFilter1</filter-class>
</filter>
<filter-mapping>
     <filter-name>demo1</filter-name>
	 <!-- 拦截路径 -->
     <url-pattern>/user/*</url-pattern>
</filter-mapping>
@WebFilter("/user/*")
public class ServletFilter1 implements Filter {
	//初始化方法
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        Filter.super.init(filterConfig);
    }

	//拦截方法
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) throws IOException, ServletException {
        System.out.println("ServletFilter1拦截器进行拦截了......");
        //放行
        filterChain.doFilter(request,response);
        System.out.println("执行之后......");
    }

	//销毁方法
    @Override
    public void destroy() {
        Filter.super.destroy();
    }
}

8.4 过滤器的生命周期

  1. init:在服务器启动后,会自动创建Filiter对象,然后调用init方法,该方法只执行一次,一般用于加载资源。
  2. doFIlter:每次请求拦截时候,都会执行该方法。
  3. destroy:在服务器正常关闭后,Filter对象会被销毁,销毁前调用destroy方法,该方法只会执行一次,通常用于释放资源。

8.5 过滤器配置

  1. 具体路径资源:/user/servletDemo1 只有访问/user/servletDemo1时,过滤器才会执行。
  2. 拦截目录:/user/* 访问/user下的所有资源时,过滤器才会执行。
  3. 后缀名拦截:*.jsp 访问所有后缀名为jsp资源时,过滤器才会执行。
  4. 拦截所有资源:/* 访问所有资源过滤器都会执行。
  5. 拦截方式配置:可以设置dispatcherTypes属性
    • REQUEST:默认值。浏览器直接请求资源
    • FORWARD:转发访问资源
      * INCLUDE:包含访问资源
    • ERROR:错误跳转资源
    • ASYNC:异步访问资源
  • 可以使用web.xml的<dispatcher>标签配置过滤方式配置。

8.6 过滤链(多个过滤器)

  1. 当配置了多个过滤器匹配访问资源时候,如过滤器1和过滤器2匹配过滤,执行顺序为过滤器1->过滤器2->执行资源->过滤器2->过滤器1
  2. 当配置多个过滤器后,默认会按照类名的字符串比较,值小的优先执行。如果使用的是web.xml的方式执行,优先配置的先执行。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2133433.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

算法刷题[比较两个字符串的最大公字符串(滑动窗口实现)]

题目&#xff1a;编程实现&#xff1a;找出两个字符串中最大公共子字符串,如"abccade","dgcadde"的最大子串为"cad" 代码如下所示&#xff1a; #define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <string.h> #inclu…

Wophp靶场漏洞挖掘

首先进入网站发现有个搜索框&#xff0c;那么我们试一下xss和SQL注入 SQL注入漏洞 发现这里页面没有给我们回显 那么我们尝试sql注入 查数据库 查表 最后查出账号密码 找到账号密码之后我们去找后台登录 进入后台后发现这里有个flag flag 接着往下翻找到一个文件上传的地方 …

STM32启用FPU浮点运算

这篇文章产生背景&#xff1a;其他人的文章太杂了&#xff0c;对我这种菜鸡无法接受&#xff1b; 参考文章&#xff1a; stm32h743单片机嵌入式学习笔记7-FPU_stmh743vit4-CSDN博客 stm32F407 打开 FPU(浮点运算处理器)_stm32f407开启fpu-CSDN博客 STM32F4CubeMXHal库下使能…

基于微信小程序的食堂点餐预约管理系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、SSM项目源码 系统展示 基于微信小程序JavaSpringBootVueMySQL的食…

linux-L6 linux管理服务的启动、重启、停止、重载、查看状态命令

来重启一下某一个服务 1.使用命令查看所有的服务状态 Systemctl找到其中的相关的服务 systemctl status xxx_你的应用程序的服务__xxx3.重启该服务 systemctl restart xxx_你的应用程序的服务__xxx下面的是备用&#xff0c;需要用的时候&#xff0c;查看就好了 启动服务 …

[机器学习]聚类算法

1 聚类算法简介 # 导包 from sklearn.datasets import make_blobs import matplotlib.pyplot as plt from sklearn.cluster import KMeans from sklearn.metrics import calinski_harabasz_score # 构建数据 x,ymake_blobs(n_samples1000,n_features2,centers[[-1,-1],[0,0],[1…

如何正确使用布尔表达式

在Java编程语言中&#xff0c;布尔表达式&#xff08;Boolean Expressions&#xff09;是程序逻辑控制的核心部分。它们是用来表示“真”&#xff08;true&#xff09;或“假”&#xff08;false&#xff09;的逻辑语句&#xff0c;通常用于控制程序的执行流程&#xff0c;比如…

【例题】1 二极管

文章目录 二极管的理想模型例题二极管的恒压降模型例题恒压管例题二极管的理想模型例题 根据二极管的理想模型,正向电压断路,反向电压开路分析。 这里的 u 0 u_0 u

OpenHarmony(鸿蒙南向开发)——小型系统芯片移植指南(二)

往期知识点记录&#xff1a; 鸿蒙&#xff08;HarmonyOS&#xff09;应用层开发&#xff08;北向&#xff09;知识点汇总 鸿蒙&#xff08;OpenHarmony&#xff09;南向开发保姆级知识点汇总~ OpenHarmony&#xff08;鸿蒙南向开发&#xff09;——轻量系统芯片移植指南(一) Op…

安全工具 | 使用Burp Suite的10个小tips

Burp Suite 应用程序中有用功能的集合 img Burp Suite 是一款出色的分析工具&#xff0c;用于测试 Web 应用程序和系统的安全漏洞。它有很多很棒的功能可以在渗透测试中使用。您使用它的次数越多&#xff0c;您就越发现它的便利功能。 本文内容是我在测试期间学到并经常的主要…

CSS框架 Tailwind CSS

文章目录 前言一、Tailwind CSS是什么&#xff1f;二、项目中如何使用1.安装Tailwind CSS2.初始化Tailwind CSS该处使用的url网络请求的数据。3.引入Tailwind CSS样式4.进行配置&#xff08;tailwind.config.js&#xff09;5.全局引入注册6.使用Tailwind CSS 总结 前言 Tailwi…

基于鸿蒙API10的RTSP播放器(七:亮度调节功能测试)

目标&#xff1a; 当我的手指在设备左方进行上下移动的时候&#xff0c;可以进行屏幕亮度的调节&#xff0c;在调节的同时&#xff0c;有实时的调节进度条显示 步骤&#xff1a; 界面逻辑&#xff1a;使用Stack() 组件&#xff0c;完成音量图标和进度条的组合显示&#xff0c…

pytorch-AutoEncoders实战

目录 1. AutoEncoders回顾2. 实现网络结构3. 实现main函数 1. AutoEncoders回顾 如下图&#xff1a;AutoEncoders实际上就是重建自己的过程 2. 实现网络结构 创建类继承自nn.Model&#xff0c;并实现init和forward函数&#xff0c;init中实现encoder、decoder 直接上代码&a…

DataWind将string类型转化为int类型的报错解决

一、现象&#xff1a; toInt64([kernel_wakeup_top_count_str]) 二、日志&#xff1a; 遇到&#xff1a;错误: 直连查询失败&#xff0c;内部异常:<class aeolus.aeolus.libs.exception.aeolus_base_exception.AeolusBaseException>: aeolus/logicQuery/logicQueryMysq…

Java数据结构应用(力扣题20. 有效的括号)

给定一个只包括 (&#xff0c;)&#xff0c;{&#xff0c;}&#xff0c;[&#xff0c;] 的字符串 s &#xff0c;判断字符串是否有效。 有效字符串需满足&#xff1a; 左括号必须用相同类型的右括号闭合。左括号必须以正确的顺序闭合。每个右括号都有一个对应的相同类型的左括…

Unity Timeline

数据存储 TimeLine和Animation一样也是资源&#xff0c;以.playable的格式存储&#xff0c;可以通过Playable Director进行加载播放。 Playable具有以下优势&#xff1a; 结构简单&#xff1b; 运行时创建、添加和删除&#xff1b; 更加灵活&#xff0c;可以直接控制动画的各种…

Golang | Leetcode Golang题解之第402题移掉K位数字

题目&#xff1a; 题解&#xff1a; func removeKdigits(num string, k int) string {stack : []byte{}for i : range num {digit : num[i]for k > 0 && len(stack) > 0 && digit < stack[len(stack)-1] {stack stack[:len(stack)-1]k--}stack app…

react 事件处理

概述 Web应用中&#xff0c;事件处理是重要的一环&#xff0c;事件处理将用户的操作行为转换为相应的逻辑执行或界面更新。在React中&#xff0c;处理事件响应的方式有多种&#xff0c;本文将详细介绍每一种处理方式的用法、使用场景和优缺点。 如果原生DOM有一个监听事件&…

论文复现--基于LeNet网络结构的数字识别

前言 一直就听说学习深度学习无非就是看论文&#xff0c;然后复现&#xff0c;不断循环&#xff0c;这段时间也看了好几篇论文(虽然都是简单的)&#xff0c;但是对于我一个人自学&#xff0c;复现成功&#xff0c;我感觉还是挺开心的 本人初学看论文的思路&#xff1a;聚焦网络…

2-93 基于matlab的无人机FMCW(频率调制连续波)毫米波高度计雷达仿真

基于matlab的无人机FMCW&#xff08;频率调制连续波&#xff09;毫米波高度计雷达仿真&#xff0c;不考虑环境杂波和收发信号隔离泄漏。通过考虑雷达天线、波束形成、信号传播、回波接收等环节影响。建立FMCW毫米波雷达系统的数学模型&#xff0c;评估无人机在不同高度下的高度…