day11-SpringBoot中註入Servlet&Filter&Listener

来源:https://www.cnblogs.com/liyuelian/archive/2023/03/23/17249551.html
-Advertisement-
Play Games

SpringBoot中註入Servlet&Filter&Listener 1.基本介紹 文檔:SpringBoot中註入Servlet&Filter&Listener 考慮到實際開發業務非常複雜和相容問題,SpringBoot支持將Servlet、Filter、Listener註入spring容器中 ...


SpringBoot中註入Servlet&Filter&Listener

1.基本介紹

文檔:SpringBoot中註入Servlet&Filter&Listener

  1. 考慮到實際開發業務非常複雜和相容問題,SpringBoot支持將Servlet、Filter、Listener註入spring容器中,成為Spring Bean
  2. 也就是說,SpringBoot開放了和原生WEB組件(Servlet、Filter、Listener)的相容
  3. SpringBoot註入Servlet、Filter、Listener,有兩種方式:
    • 通過註解方式註入
    • 使用RegistrationBean方式註入

2.通過註解方式註入

2.1@WebServlet

屬性名 對應標簽 描述
name <servlet-name> 指定 Servlet 的 name 屬性。 如果沒有顯式指定,則取值為該 Servlet 的完全限定名,即包名+類名
value <url-pattern> 該屬性等價於 urlPatterns 屬性,兩者不能同時指定。 如果同時指定,通常是忽略 value 的取值
urlPatterns <url-pattern> 指定一組 Servlet 的 URL 匹配模式
loadOnStartup <load-on-startup> 指定 Servlet 的載入順序
initParams <init-param> 指定一組 Servlet 初始化參數
asyncSupported <async-supported> 聲明 Servlet 是否支持非同步操作模式
description <description> 指定該 Servlet 的描述信息
displayName <display-name> 指定該 Servlet 的顯示名

例子--使用@WebServlet註入Servlet

(1)MyServlet.java

  1. 通過繼承HttpServlet來開發原生的Servlet

  2. 使用@WebServlet,表示將其標識的對象註入到Spring容器中

  3. urlPatterns = {"servlet01","servlet02"} 對此servlet配置了映射路徑

  4. 對於開發的原生的Servlet,需要使用@ServletComponentScan在SpringBoot主程式中,指定要掃描的原生Servlet,這樣該Servlet才能註入容器

package com.li.thymeleaf.servlet;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * @author 李
 * @version 1.0
 */
@WebServlet(urlPatterns = {"/servlet01", "/servlet02"})
public class MyServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        resp.getWriter().write("Hello,MyServlet!");
    }
}

(2)Application.java主程式

package com.li.thymeleaf;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.servlet.ServletComponentScan;

/**
 * @author 李
 * @version 1.0
 */
//指定掃描Servlet
@ServletComponentScan(basePackages = "com.li.thymeleaf")
@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class,args);
    }
}

(3)瀏覽器訪問地址:http://localhost:8080/servlet01獲者 http://localhost:8080/servlet02,返回如下:

image-20230323182004415

註意:註入的Servlet不會被SpringBoot的攔截器攔截(因為原生Servlet和前端控制器DispatcherServlet是統一級別的,而攔截器在DispatcherServlet中)

image-20230322173641992

2.2@WebFilter

屬性名 說 明
description 該過濾器的描述信息,等價於 <description>標簽。
displayName 該過濾器的顯示名,通常配合工具使用,等價於 <display-name> 標簽
initParams 指定一組過濾器初始化參數,等價於 <init-param> 標簽。
filterName 指定過濾器的 name 屬性,等價於 <filter-name>
servletNames 指定過濾器將應用於哪些 Servlet。取值是 @WebServlet 中的 name 屬性的取值,或者是 web.xml 中 <servlet-name> 的取值
value/urlPatterns 過濾器的 URL 匹配模式,等價於<url-pattern>標簽
dispatcherTypes 指定過濾器的轉發模式。具體取值包括: ASYNC、ERROR、FORWARD、INCLUDE、REQUEST。
asyncSupported 聲明過濾器是否支持非同步操作模式, 等價於<async-supported>標簽

例子--使用@WebFilter註入Filter

  1. @WebFilter標識一個過濾器,並註入spring容器

  2. urlPatterns = {"/css/*", "/images/*"}表示請求/css/目錄或者/images/目錄下的資源時,請求會經過這個過濾器

  3. 需要在主程式中,指定要掃描的Filter,這樣該Filter才能註入容器

package com.li.thymeleaf.filter;

import lombok.extern.slf4j.Slf4j;

import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;

/**
 * @author 李
 * @version 1.0
 * 開發Filter並註入spring容器
 */
@Slf4j
@WebFilter(urlPatterns = {"/css/*", "/images/*"})
public class MyFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        log.info("MyFilter的init()方法被執行...");
    }

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        log.info("MyFilter的doFilter()方法被執行...");
        HttpServletRequest httpServletRequest = (HttpServletRequest) request;
        log.info("過濾器處理的uri={}", httpServletRequest.getRequestURI());
        chain.doFilter(request, response);//放行
    }

    @Override
    public void destroy() {
        log.info("MyFilter的destroy()方法被執行...");
    }
}

(2)在主程式中配置掃描該過濾器(略)

(3)在瀏覽器訪問地址:http://localhost:8080/images/login.jpg,後臺輸出:

2023-03-23 18:59:36.685  INFO 39228 --- [nio-8080-exec-6] com.li.thymeleaf.filter.MyFilter         : MyFilter的doFilter()方法被執行...
2023-03-23 18:59:36.685  INFO 39228 --- [nio-8080-exec-6] com.li.thymeleaf.filter.MyFilter         : 過濾器處理的uri=/images/login.jpg

有時候後臺沒有輸出,可能是瀏覽器緩存問題

2.3@WebListener

(1)MyListener.java

package com.li.thymeleaf.listener;

import lombok.extern.slf4j.Slf4j;

import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.annotation.WebListener;

/**
 * @author 李
 * @version 1.0
 */
@Slf4j
@WebListener
public class MyListener implements ServletContextListener {
    @Override
    public void contextInitialized(ServletContextEvent sce) {
        //可以加入項目初始化相關的業務
        log.info("MyListener-contextInitialized()-項目初始化OK~");
    }

    @Override
    public void contextDestroyed(ServletContextEvent sce) {
        //可以加入業務
        log.info("MyListener-contextDestroyed()-項目初銷毀...");
    }
}

(2)在主程式 Application.java配置掃描該監聽器

package com.li.thymeleaf;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.servlet.ServletComponentScan;
import org.springframework.context.ConfigurableApplicationContext;

/**
 * @author 李
 * @version 1.0
 */
//指定掃描監聽器
@ServletComponentScan(basePackages = "com.li.thymeleaf")
@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        ConfigurableApplicationContext ioc =
                SpringApplication.run(Application.class, args);
        //監聽器的contextDestroyed()方法在容器銷毀時觸發
        ioc.stop();
    }
}

(3)啟動項目,控制台輸出:

image-20230323191501341

3.使用RegistrationBean方式註入

RegistrationConfig.java:

package com.li.thymeleaf.config;

import com.li.thymeleaf.filter.MyFilter;
import com.li.thymeleaf.listener.MyListener;
import com.li.thymeleaf.servlet.MyServlet;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.boot.web.servlet.ServletListenerRegistrationBean;
import org.springframework.boot.web.servlet.ServletRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import java.util.Arrays;

/**
 * @author 李
 * @version 1.0
 * RegistrationConfig是一個配置類,
 * 預設為單實例模式 proxyBeanMethods=true
 */
@Configuration
public class RegistrationConfig {
    //使用RegistrationBean方式註入Servlet
    @Bean
    public ServletRegistrationBean servlet_() {
        MyServlet myServlet = new MyServlet();
        //將myServlet關聯到ServletRegistrationBean對象
        //可以指定多個映射url
        return new ServletRegistrationBean(myServlet, "/servlet01", "/servlet02");
    }

    //使用RegistrationBean方式註入Filter
    @Bean
    public FilterRegistrationBean filter_() {
        MyFilter myFilter = new MyFilter();//創建原生的Filter對象
        FilterRegistrationBean filterRegistrationBean = new FilterRegistrationBean(myFilter);
        //設置filter的urlPattern
        filterRegistrationBean.setUrlPatterns(Arrays.asList("/css/*", "/images/*"));
        return filterRegistrationBean;
    }

    //使用RegistrationBean方式註入Listener
    @Bean
    public ServletListenerRegistrationBean listener_() {
        MyListener myListener = new MyListener();//創建原生的Listener對象
        return new ServletListenerRegistrationBean(myListener);
    }
}

使用RegistrationBean的方式註入,不必在主程式Application.java中配置掃描

運行程式,可以看到三個組件都被註入到容器中:

image-20230323201853568

4.註意事項和細節

4.1請求自定義Servlet時,為什麼不會到達攔截器?

原因分析:

註入的Servlet會存在Spring容器,DispatcherServlet也存在Spring容器。當多個Servlet都能處理到同一層路徑時,存在精確優先原則/最長首碼匹配原則:**精準匹配 > 目錄匹配 > 擴展名匹配 > /* > / **

如下圖:當瀏覽器請求路徑為/servlet01 時,MyServlet的映射路徑對與瀏覽器請求來說是精準匹配,因此此時MyServlet的映射路徑優先順序高於前端控制器的 /,請求路徑會走tomcat流程,不會到達前端控制器,也就不會執行攔截器。

image-20230323204226262

當然,在SpringBoot中,去調用@Controller目標方法,仍是按照DispatcherServlet分發匹配的機制

4.2DispatcherServlet在SpringBoot如何進行配置和註入

DispatcherServletAutoConfiguration 完成對 DispatcherServlet 的自動配置。

DispatcherServletAutoConfiguration 類,有一個內部類:

@Configuration(proxyBeanMethods = false)
@Conditional(DefaultDispatcherServletCondition.class)
@ConditionalOnClass(ServletRegistration.class)
@EnableConfigurationProperties(WebMvcProperties.class)
protected static class DispatcherServletConfiguration {

    @Bean(name = DEFAULT_DISPATCHER_SERVLET_BEAN_NAME)
    //創建了DispatcherServlet對象,併進行一系列設置並返回。
    public DispatcherServlet dispatcherServlet(WebMvcProperties webMvcProperties) {
        DispatcherServlet dispatcherServlet = new DispatcherServlet();
        dispatcherServlet.setDispatchOptionsRequest(webMvcProperties.isDispatchOptionsRequest());
        dispatcherServlet.setDispatchTraceRequest(webMvcProperties.isDispatchTraceRequest());
        dispatcherServlet.setThrowExceptionIfNoHandlerFound(webMvcProperties.isThrowExceptionIfNoHandlerFound());
        dispatcherServlet.setPublishEvents(webMvcProperties.isPublishRequestHandledEvents());
        dispatcherServlet.setEnableLoggingRequestDetails(webMvcProperties.isLogRequestDetails());
        return dispatcherServlet;
    }

    @Bean
    @ConditionalOnBean(MultipartResolver.class)
    @ConditionalOnMissingBean(name = DispatcherServlet.MULTIPART_RESOLVER_BEAN_NAME)
    public MultipartResolver multipartResolver(MultipartResolver resolver) {
        // Detect if the user has created a MultipartResolver but named it incorrectly
        return resolver;
    }

}

然後通過如下方法,創建DispatcherServletRegistrationBean對象,並將創建的DispatcherServlet對象關聯到這個DispatcherServletRegistrationBean對象中,將DispatcherServletRegistrationBean對象通過@Bean註入到容器中。

@Configuration(proxyBeanMethods = false)
@Conditional(DispatcherServletRegistrationCondition.class)
@ConditionalOnClass(ServletRegistration.class)
@EnableConfigurationProperties(WebMvcProperties.class)
@Import(DispatcherServletConfiguration.class)
protected static class DispatcherServletRegistrationConfiguration {

   @Bean(name = DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME)
   @ConditionalOnBean(value = DispatcherServlet.class, name = DEFAULT_DISPATCHER_SERVLET_BEAN_NAME)
   public DispatcherServletRegistrationBean dispatcherServletRegistration(DispatcherServlet dispatcherServlet,
         WebMvcProperties webMvcProperties, ObjectProvider<MultipartConfigElement> multipartConfig) {
      DispatcherServletRegistrationBean registration = new DispatcherServletRegistrationBean(dispatcherServlet,
            webMvcProperties.getServlet().getPath());//設置路徑 /
      registration.setName(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME);
      registration.setLoadOnStartup(webMvcProperties.getServlet().getLoadOnStartup());
      multipartConfig.ifAvailable(registration::setMultipartConfig);
      return registration;
   }

}
image-20230323212404193
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、案例背景 電腦包含記憶體(RAM),CPU 等硬體設備,根據如圖所示的“產品等級結構-產品族示意圖”,使用抽象工廠模式實現電腦設備創建過程並繪製類圖 二、實現步驟 根據題意,使用抽象工廠模式並畫出類圖,類圖中應包含一個抽象工廠類 AbstractFactory,PcFactory 和 MacF ...
  • 1. 理解可變性 1.1. 理解測試結果如何隨時間變化 1.2. 可以通過多次運行測試後取平均值來解決 1.3. 因代碼改進而進行的測試叫作回歸測試(regression testing) 1.3.1. 原本的代碼叫作基線(baseline) 1.3.2. 新的代碼叫作樣本(specimen) 1. ...
  • 用Python基於Google Bard做一個互動式的聊天機器人 之前已經通過瀏覽器試過了 Google Bard ,更多細節請看: Try out Google Bard, Will Google Bard beat the ChatGPT?. 現在我們想實現自動化,所以我用Python做一個交互 ...
  • 每次提交代碼的時候,你是否有為如何寫Commit Message而遲遲按不下提交的時刻呢?然後,死磨硬泡寫了一些並提交後,又被review的小伙伴吐槽了呢?相信很多小伙伴有過這樣的經歷吧? 趁著最近ChatGPT那麼火,就來順手推薦一個可以用於解決這個問題的VS Code插件:vscode-gpto ...
  • C++ STL標準庫中提供了多個用於排序的Sort函數,常用的包括有sort() / stable_sort() / partial_sort(),具體的函數用法如下表所示: | 函數 | 用法 | | | | | std::sort(first,last) | 對容器或數組first~last範圍 ...
  • SpringBoot內置Tomcat的配置和切換 1.基本介紹 SpringBoot支持的webServer:Tomcat,Jetty,Undertow 因為在spring-boot-starter-web中,預設導入的是tomcat,因此啟動時使用的web容器就是tomcat。 同時 Spring ...
  • 最近在看了《微信背後的產品觀 - 張小龍手抄版》,其中有段話如下: 用戶需求是零散的,解決方案是歸納抽象的過程 那如何歸納抽象呢?是否有一定的實踐方法論呢?經過一輪探討和學習,有這些答案: 5 Whys 分析法 U 型思考法 等等 二、5 Whys 分析法 5 Whys 法,最初由豐田佐吉開發,併在 ...
  • 多線程 多線程概述 多線程就是電腦用時運行多個任務 但實質上,同一個時間點,只會運行一個任務,只是電腦在不同任務之間來回切換而已。 併發和並行 並行:在同一時間,多個任務分別在多個CPU上進行。 併發:在同一時間,多個任務在同一個CPU交替進行。 線程和進程 進程 獨立性:進程是一個獨立運行的應 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...