摘要:攔截器注冊(cè)配置攔截器注冊(cè)配置通過(guò)可以發(fā)現(xiàn)中的的清理工作沒(méi)有得到執(zhí)行。只會(huì)在出現(xiàn)異常,返回或正常執(zhí)行結(jié)束才會(huì)從索引依次往前執(zhí)行。
發(fā)現(xiàn)問(wèn)題最近在進(jìn)行壓測(cè)發(fā)現(xiàn),有一些接口時(shí)好時(shí)壞,通過(guò)sentry日志平臺(tái)及sky walking平臺(tái)跟蹤發(fā)現(xiàn),用戶張三獲取到的用戶上下文確是李四。
用戶登錄下上文
/**
* 用戶登錄下上文
*
* @author : jamesfu
* @date : 22/5/2019
* @time : 9:18 AM
*/
@Data
public class UserContext {
private final static ThreadLocal threadLocal = new ThreadLocal<>();
private Long id;
private String loginName;
public static UserContext get() {
UserContext context = threadLocal.get();
if (context == null) {
// TODO(james.h.fu):根據(jù)請(qǐng)求上下文獲取token, 然后恢復(fù)用戶登錄下上文
context = new UserContext() {{
setId(1L);
setLoginName("james.h.fu1");
}};
threadLocal.set(context);
}
return context;
}
public static void clear() {
threadLocal.remove();
}
public static void set(UserContext context) {
if (context != null) {
threadLocal.set(context);
}
}
}
在攔截器中有調(diào)用UserContext.set恢復(fù)用戶登錄上下文,并在請(qǐng)求結(jié)束時(shí)調(diào)用UserContext.clear清理用戶登錄上下文。
攔截器注冊(cè)配置
/**
* 攔截器注冊(cè)配置
*
* @author : jamesfu
* @date : 22/5/2019
* @time : 9:15 AM
*/
@Configuration
public class FilterConfig implements WebMvcConfigurer {
@Autowired
private JsonRpcInterceptor jsonRpcInterceptor;
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(jsonRpcInterceptor)
.addPathPatterns("/json.rpc");
}
}
通過(guò)debug可以發(fā)現(xiàn)UserContext中的ThreadLocal的清理工作沒(méi)有得到執(zhí)行。導(dǎo)致請(qǐng)求進(jìn)來(lái)時(shí),有可能ThreadLocal已存在了,就不會(huì)再根據(jù)請(qǐng)求上下文恢復(fù)了。
tomcat 在收到http請(qǐng)求后,最終會(huì)交由spring mvc的DispatcherServlet處理。 這里可以從doDispatch按圖索驥,順藤摸瓜地往下看起走。
/**
* Process the actual dispatching to the handler.
* The handler will be obtained by applying the servlet"s HandlerMappings in order.
* The HandlerAdapter will be obtained by querying the servlet"s installed HandlerAdapters
* to find the first that supports the handler class.
*
All HTTP methods are handled by this method. It"s up to HandlerAdapters or handlers
* themselves to decide which methods are acceptable.
* @param request current HTTP request
* @param response current HTTP response
* @throws Exception in case of any kind of processing failure
*/
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception
請(qǐng)求會(huì)得到分發(fā),然后執(zhí)行各個(gè)已注冊(cè)Handler的preHandle-->postHandle-->afterCompletion。
源碼走讀:HandlerExecutionChain/**
* Apply preHandle methods of registered interceptors.
* @return {@code true} if the execution chain should proceed with the
* next interceptor or the handler itself. Else, DispatcherServlet assumes
* that this interceptor has already dealt with the response itself.
*/
boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception {
HandlerInterceptor[] interceptors = getInterceptors();
if (!ObjectUtils.isEmpty(interceptors)) {
for (int i = 0; i < interceptors.length; i++) {
HandlerInterceptor interceptor = interceptors[i];
if (!interceptor.preHandle(request, response, this.handler)) {
triggerAfterCompletion(request, response, null);
return false;
}
this.interceptorIndex = i;
}
}
return true;
}
當(dāng)執(zhí)行到preHandle返回false時(shí),它就會(huì)從上一個(gè)返回true的handler依次往前執(zhí)行afterCompletion,它自己的afterCompletion得不到執(zhí)行。
/**
* Trigger afterCompletion callbacks on the mapped HandlerInterceptors.
* Will just invoke afterCompletion for all interceptors whose preHandle invocation
* has successfully completed and returned true.
*/
void triggerAfterCompletion(HttpServletRequest request, HttpServletResponse response, @Nullable Exception ex)
throws Exception {
HandlerInterceptor[] interceptors = getInterceptors();
if (!ObjectUtils.isEmpty(interceptors)) {
for (int i = this.interceptorIndex; i >= 0; i--) {
HandlerInterceptor interceptor = interceptors[i];
try {
interceptor.afterCompletion(request, response, this.handler, ex);
}
catch (Throwable ex2) {
logger.error("HandlerInterceptor.afterCompletion threw exception", ex2);
}
}
}
}
triggerAfterCompletion只會(huì)在(1)出現(xiàn)異常,(2)preHandle返回false 或(3)正常執(zhí)行結(jié)束才會(huì)從索引interceptorIndex依次往前執(zhí)行。
所以基于以上源碼可以得知,在寫(xiě)攔截器時(shí)preHandle返回false時(shí),afterCompletion是不會(huì)執(zhí)行的。所以一些必要的清理工作得不到執(zhí)行,會(huì)出現(xiàn)類似我們遇到的帳號(hào)串的問(wèn)題。
Spring 攔截器——HandlerInterceptor
SpringMVC HandlerInterceptor詭異問(wèn)題排查
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/6896.html
摘要:只要有一個(gè)攔截器不放行,不能執(zhí)行完成號(hào)不放行和號(hào)不放行測(cè)試結(jié)果總結(jié)只有前邊的攔截器方法放行,下邊的攔截器的才執(zhí)行。至于他們的攔截器鏈的調(diào)用順序,和的是沒(méi)有差別的。 前言 本博文主要講解的知識(shí)點(diǎn)如下: 校驗(yàn)器 統(tǒng)一處理異常 RESTful 攔截器 Validation 在我們的Struts2中,我們是繼承ActionSupport來(lái)實(shí)現(xiàn)校驗(yàn)的...它有兩種方式來(lái)實(shí)現(xiàn)校驗(yàn)的功能 手寫(xiě)...
摘要:簡(jiǎn)介注解用于修飾的方法,根據(jù)的的內(nèi)容,通過(guò)適當(dāng)?shù)霓D(zhuǎn)換為客戶端需要格式的數(shù)據(jù)并且寫(xiě)入到的數(shù)據(jù)區(qū),從而不通過(guò)視圖解析器直接將數(shù)據(jù)響應(yīng)給客戶端。并且這些解析器都實(shí)現(xiàn)了接口,在接口中有四個(gè)最為主要的接口方法。 SpringMVC 細(xì)節(jié)方面的東西很多,所以在這里做一篇簡(jiǎn)單的 SpringMVC 的筆記記錄,方便以后查看。 Spring MVC是當(dāng)前最優(yōu)秀的MVC框架,自從Spring 2.5版本...
摘要:中的攔截器在開(kāi)發(fā)中,攔截器是經(jīng)常用到的功能。該攔截器只能過(guò)濾請(qǐng)求,允許多個(gè)攔截器同時(shí)存在,通過(guò)攔截器鏈管理。當(dāng)時(shí)不再執(zhí)行后續(xù)的攔截器鏈及被攔截的請(qǐng)求。實(shí)現(xiàn)攔截器大致也分為兩種,一種是實(shí)現(xiàn)接口,另一種利用的注解或配置。 Spring中的攔截器 在web開(kāi)發(fā)中,攔截器是經(jīng)常用到的功能。它可以幫我們驗(yàn)證是否登陸、權(quán)限認(rèn)證、數(shù)據(jù)校驗(yàn)、預(yù)先設(shè)置數(shù)據(jù)以及統(tǒng)計(jì)方法的執(zhí)行效率等等。今天就來(lái)詳細(xì)的談...
摘要:概述通過(guò)前三章的分析,我們簡(jiǎn)要分析了和,但對(duì)攔截器部分做詳細(xì)的分析,攔截器的加載和初始化是三個(gè)相同的部分。 概述 通過(guò)前三章的分析,我們簡(jiǎn)要分析了SimpleUrlHandlerMapping、BeanNameUrlHandlerMapping和RequestMappingHandlerMapping,但對(duì)攔截器部分做詳細(xì)的分析,攔截器的加載和初始化是三個(gè)HandlerMapping相...
摘要:中添加攔截器配置如下攔截所有請(qǐng)求,也就是,只攔截開(kāi)頭的請(qǐng)求。在中并沒(méi)有提供配置文件的方式來(lái)配置攔截器,因此需要使用的代碼式配置,配置如下這個(gè)屬性通常并不需要手動(dòng)配置,高版本的會(huì)自動(dòng)檢測(cè)第四點(diǎn)講下靜態(tài)資源映射。 以下內(nèi)容,如有問(wèn)題,煩請(qǐng)指出,謝謝 上一篇講解了springboot的helloworld部分,這一篇開(kāi)始講解如何使用springboot進(jìn)行實(shí)際的應(yīng)用開(kāi)發(fā),基本上尋著sprin...
閱讀 649·2021-11-25 09:43
閱讀 1668·2021-11-18 10:02
閱讀 1041·2021-10-15 09:39
閱讀 1890·2021-10-12 10:18
閱讀 2122·2021-09-22 15:43
閱讀 773·2021-09-22 15:10
閱讀 2088·2019-08-30 15:53
閱讀 988·2019-08-30 13:00