欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談Springboot實現(xiàn)攔截器的兩種方式

 更新時間:2021年08月19日 15:20:50   作者:怪咖軟妹@  
本文詳細的介紹了Springboot攔截器的兩種方式實現(xiàn),一種就是用攔截器,一種就是過濾器,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下

實現(xiàn)過濾請求有兩種方式:

一種就是用攔截器,一種就是過濾器

攔截器相對來說比較專業(yè),而過濾器雖然不專業(yè)但是也能完成基本的攔截請求要求。

一、攔截器方式

1、配置HandlerInterceptor

下面這個也是我們公司項目攔截器的寫法,總體來說感覺還不錯,我就記錄了下來。
利用了一個靜態(tài)Pattern變量存儲不走攔截器的路徑,然后在preHandle方法當中進行過濾,讓他返回true。

@Component
public class LoginInterceptor implements HandlerInterceptor{
	
	private static final Pattern SHOULD_NOT_FILTER_URL_PATTERN;
	
	static {
		List<String> urlList = new ArrayList<>();
		// 將不走攔截器的請求存放到Pattern
		urlList.add("(socket/.*)");
		urlList.add("(user/findUserList)");
		StringBuilder sb = new StringBuilder();
		for (String url : urlList) {
			sb.append(url);
			sb.append("|");
		}
		sb.setLength(sb.length() - 1);
		SHOULD_NOT_FILTER_URL_PATTERN = Pattern.compile(sb.toString());
	}
	
	/**
	 * 在請求處理之前進行調(diào)用(Controller方法調(diào)用之前)
	 */
	@Override
	public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
			throws Exception {
		
		HttpSession session = request.getSession();
		// 獲取訪問的url
 		String servletPath = request.getServletPath();
		// 排除特定請求
		if (SHOULD_NOT_FILTER_URL_PATTERN.matcher(servletPath).find()) {
			return true;
		}
		if (session.getAttribute("user") != null) {
			// 可能有的項目在校驗完session,還會校驗token
			String token = request.getHeader("access_token");
			// 此處業(yè)務(wù)省略。。。
			return true;
		}
		return false;
	}

	/**
	 * 請求處理之后進行調(diào)用,但是在視圖被渲染之前(Controller方法調(diào)用之后)
	 */
	@Override
	public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
			ModelAndView modelAndView) throws Exception {
		// TODO Auto-generated method stub
		
	}

	/**
	 * 在整個請求結(jié)束之后被調(diào)用,也就是在DispatcherServlet 渲染了對應(yīng)的視圖之后執(zhí)行(主要是用于進行資源清理工作)
	 */
	@Override
	public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
			throws Exception {
		// TODO Auto-generated method stub
		
	}
}

2、注冊攔截器

配置完上面的攔截器還需要注冊攔截器。

WebMvcConfigurerAdapter類是 Spring內(nèi)部的一種配置方式
采用JavaBean的形式來代替?zhèn)鹘y(tǒng)的xml配置文件形式進行針對框架個性化定制

@Configuration
public class MyWebAppConfigurer extends WebMvcConfigurerAdapter {
	
	@Autowired
    private LoginInterceptor loginInterceptor;
	

	@Override
	public void addInterceptors(InterceptorRegistry registry) {
		// 多個攔截器組成一個攔截器鏈
		// addPathPatterns 用于添加攔截規(guī)則
		// excludePathPatterns 用戶排除攔截
		registry.addInterceptor(loginInterceptor).addPathPatterns("/**");
		super.addInterceptors(registry);
	}
}

3、使用攔截器的坑

繼承WebMvcConfigurerAdapter 重寫addInterceptors方法的時候,一定要使用注入方式,將loginInterceptor注入到變量當中,要么就使用@Bean注解的方式,將loginInterceptor(攔截器)注入到容器。

之所以這么搞是因為攔截器在 Bean 初始化之前進行,所以在攔截器中無法像這樣注入 Bean。
就算加了@Component,他是存放于容器當中了,但是他存放容器當中的對象屬性,是空屬性。

在WebMvcConfigurerAdapter 使用@Autowired注入了一遍攔截器,屬性就有值了。

說白了不管采用哪種方案,目的只有一個,讓對象的屬性有值,因為攔截器比其他對象初始化早,導(dǎo)致屬性為空,想讓他有值,就想辦法讓他重新走一遍spring注入容器,也可以采用這種方式:

@Configuration
public class MyWebAppConfigurer extends WebMvcConfigurerAdapter {
	
	// 這么寫的目的是為了在SessionInterceptor中能注入spring中的service
	@Bean
	LoginInterceptor loginInterceptor() {
		return new LoginInterceptor();
	}

	@Override
	public void addInterceptors(InterceptorRegistry registry) {
		// 多個攔截器組成一個攔截器鏈
		// addPathPatterns 用于添加攔截規(guī)則
		// excludePathPatterns 用戶排除攔截
		registry.addInterceptor(loginInterceptor()).addPathPatterns("/**");
		super.addInterceptors(registry);
	}
}

二、過濾器方式

本人這一篇博客寫了關(guān)于filter的一些知識:http://www.dbjr.com.cn/article/204084.htm

1、實現(xiàn)Filter接口

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.regex.Pattern;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import org.springframework.stereotype.Component;

@Component
@WebFilter(filterName = "requestParamFilter",urlPatterns = "/*")
public class RequestParamFilter implements Filter {

	private static final Pattern SHOULD_NOT_FILTER_URL_PATTERN;
	
	static {
		List<String> urlList = new ArrayList<>();
		// 將不走攔截器的請求存放到Pattern
		urlList.add("(socket/.*)");
		urlList.add("(user/findUserList)");
		StringBuilder sb = new StringBuilder();
		for (String url : urlList) {
			sb.append(url);
			sb.append("|");
		}
		sb.setLength(sb.length() - 1);
		SHOULD_NOT_FILTER_URL_PATTERN = Pattern.compile(sb.toString());
	}

    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

    }
	
	// 發(fā)送請求會執(zhí)行這個方法
    // 一個doFilter相當于攔截器的執(zhí)行前和執(zhí)行后
    // filterChain.doFilter后面的內(nèi)容就是執(zhí)行后的內(nèi)容,假如不執(zhí)行filterChain.doFilter方法相當于方法被攔截
    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse,
                         FilterChain filterChain) throws IOException, ServletException {
        System.out.println("sunhan---請求參數(shù)過濾器!---test1");
        HttpServletRequest request = (HttpServletRequest)servletRequest;
        HttpSession session = request.getSession();
		// 獲取訪問的url
 		String servletPath = request.getServletPath();
		// 排除特定請求
		if (SHOULD_NOT_FILTER_URL_PATTERN.matcher(servletPath).find()) {
			filterChain.doFilter(servletRequest,servletResponse);
		}
        System.out.println("開始攔截了................");
        //業(yè)務(wù)代碼
    }

    @Override
    public void destroy() {

    }
}

2、使用過濾器需要注意的

攔截器可以獲取IOC容器中的各個bean,而過濾器就不行,這點很重要,在攔截器里注入一個service,可以調(diào)用業(yè)務(wù)邏輯

執(zhí)行順序如下:

在這里插入圖片描述

到此這篇關(guān)于淺談Springboot實現(xiàn)攔截器的兩種方式的文章就介紹到這了,更多相關(guān)Springboot 攔截器內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Kotlin 基礎(chǔ)教程之類、對象、接口

    Kotlin 基礎(chǔ)教程之類、對象、接口

    這篇文章主要介紹了Kotlin 基礎(chǔ)教程之類、對象、接口的相關(guān)資料,需要的朋友可以參考下
    2017-06-06
  • RabbitMQ的ACK確認機制保障消費端消息的可靠性詳解

    RabbitMQ的ACK確認機制保障消費端消息的可靠性詳解

    這篇文章主要介紹了RabbitMQ的ACK確認機制保障消費端消息的可靠性詳解,簡單來說,就是你必須關(guān)閉 RabbitMQ 的自動ack ,可以通過一個 api 來調(diào)用就行,然后每次你自己代碼里確保處理完的時候,再在程序里 ack 一把,需要的朋友可以參考下
    2023-12-12
  • Javaweb項目session超時解決方案

    Javaweb項目session超時解決方案

    這篇文章主要介紹了Javaweb項目session超時解決方案,關(guān)于解決方案分類比較明確,內(nèi)容詳細,需要的朋友可以參考下。
    2017-09-09
  • java開發(fā)之讀寫txt文件操作的實現(xiàn)

    java開發(fā)之讀寫txt文件操作的實現(xiàn)

    本篇文章介紹了,java開發(fā)之讀寫txt文件操作的實現(xiàn)。需要的朋友參考下
    2013-05-05
  • Java整合mybatis實現(xiàn)過濾數(shù)據(jù)

    Java整合mybatis實現(xiàn)過濾數(shù)據(jù)

    這篇文章主要介紹了Java整合mybatis實現(xiàn)過濾數(shù)據(jù),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友們下面隨著小編來一起學(xué)習吧
    2023-01-01
  • SpringBoot+MinIO實現(xiàn)文件切片極速詳解

    SpringBoot+MinIO實現(xiàn)文件切片極速詳解

    在現(xiàn)代Web應(yīng)用中,文件上傳是一個常見的需求,尤其是對于大文件的上傳,如視頻、音頻或大型文檔,所以本文就來為大家介紹一下如何使用Spring Boot和MinIO實現(xiàn)文件切片極速上傳技術(shù)吧
    2023-12-12
  • java validation 后臺參數(shù)驗證的使用詳解

    java validation 后臺參數(shù)驗證的使用詳解

    本篇文章主要介紹了java validation 后臺參數(shù)驗證的使用詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-10-10
  • Java中Lambda表達式的使用詳解

    Java中Lambda表達式的使用詳解

    Lambda 表達式,也可稱為閉包,它是推動 Java 8 發(fā)布的最重要新特性。Lambda 允許把函數(shù)作為一個方法的參數(shù)(函數(shù)作為參數(shù)傳遞進方法中)。使用 Lambda 表達式可以使代碼變的更加簡潔緊湊
    2021-09-09
  • SpringMVC 參數(shù)綁定意義及實現(xiàn)過程解析

    SpringMVC 參數(shù)綁定意義及實現(xiàn)過程解析

    這篇文章主要介紹了SpringMVC 參數(shù)綁定意義及實現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友可以參考下
    2019-11-11
  • JDK源碼Enum類原理及代碼實例解析

    JDK源碼Enum類原理及代碼實例解析

    這篇文章主要介紹了JDK源碼Enum類原理及代碼實例解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友可以參考下
    2020-11-11

最新評論