Spring5源碼解析-Spring中的處理攔截器

在Java的Web應(yīng)用程序中通常使用過濾器(即filter)來捕獲HTTP請求愉昆。但它們僅為webapps保留泄隔。Spring引入了一種新的方法來實現(xiàn)敦姻,更通用,稱為處理程序攔截器铐刘。

本文將分3部分陪每。第一部分來講Spring處理程序攔截器的理論概念。第二部分镰吵,說一說默認(rèn)的Spring攔截器檩禾。最后一部分老規(guī)矩,應(yīng)用實戰(zhàn)捡遍,我們將寫我們自己的處理程序攔截器锌订。


什么是Spring中的處理程序攔截器?

要了解Spring攔截器的作用画株,我們需要先解釋一下HTTP請求的執(zhí)行鏈。DispatcherServlet捕獲每個請求。調(diào)度員做的第一件事就是將接收到的URL和相應(yīng)的controller進(jìn)行映射(controller必須恰到好處地處理當(dāng)前的請求)谓传。但是蜈项,在到達(dá)對應(yīng)的controller之前,請求可以被攔截器處理续挟。這些攔截器就像過濾器紧卒。只有當(dāng)URL找到對應(yīng)于它們的映射時才調(diào)用它們。在通過攔截器(攔截器預(yù)處理诗祸,其實也可以說前置處理)進(jìn)行前置處理后跑芳,請求最終到達(dá)controller。之后直颅,發(fā)送請求生成視圖博个。但是在這之前,攔截器還是有可能來再次處理它(攔截器后置處理)功偿。只有在最后一次操作之后盆佣,視圖解析器才能捕獲數(shù)據(jù)并輸出視圖。

處理程序映射攔截器基于org.springframework.web.servlet.HandlerInterceptor接口械荷。和之前簡要描述的那樣共耍,它們可以在將其發(fā)送到控制器(方法前使用preHandle)之前或之后(方法后使用postHandle)攔截請求。preHandle方法返回一個布爾值吨瞎,如果返回false痹兜,則可以在執(zhí)行鏈中執(zhí)行中斷請求處理。此接口中還有一個方法afterCompletion颤诀,只有在preHandler方法發(fā)送為true時才會在渲染視圖后調(diào)用它(完成請求處理后的回調(diào)字旭,即渲染視圖后)。

攔截器也可以在新線程中啟動着绊。在這種情況下谐算,攔截器必須實現(xiàn)org.springframework.web.servlet.AsyncHandlerInterceptor接口。它繼承HandlerInterceptor并提供一個方法afterConcurrentHandlingStarted归露。每次處理程序得到正確執(zhí)行時洲脂,都會調(diào)用此方法而不是調(diào)用postHandler()和afterCompletion()。它也可以對發(fā)送請求進(jìn)行異步處理剧包。通過Spring源碼此方法注釋可以知道恐锦,這個方法的典型的應(yīng)用是可以用來清理本地線程變量。

/**
 * Extends {@code HandlerInterceptor} with a callback method invoked after the
 * start of asynchronous request handling.
 *
 * <p>When a handler starts an asynchronous request, the {@link DispatcherServlet}
 * exits without invoking {@code postHandle} and {@code afterCompletion} as it
 * normally does for a synchronous request, since the result of request handling
 * (e.g. ModelAndView) is likely not yet ready and will be produced concurrently
 * from another thread. In such scenarios, {@link #afterConcurrentHandlingStarted}
 * is invoked instead, allowing implementations to perform tasks such as cleaning
 * up thread-bound attributes before releasing the thread to the Servlet container.
 *
 * <p>When asynchronous handling completes, the request is dispatched to the
 * container for further processing. At this stage the {@code DispatcherServlet}
 * invokes {@code preHandle}, {@code postHandle}, and {@code afterCompletion}.
 * To distinguish between the initial request and the subsequent dispatch
 * after asynchronous handling completes, interceptors can check whether the
 * {@code javax.servlet.DispatcherType} of {@link javax.servlet.ServletRequest}
 * is {@code "REQUEST"} or {@code "ASYNC"}.
 *
 * <p>Note that {@code HandlerInterceptor} implementations may need to do work
 * when an async request times out or completes with a network error. For such
 * cases the Servlet container does not dispatch and therefore the
 * {@code postHandle} and {@code afterCompletion} methods will not be invoked.
 * Instead, interceptors can register to track an asynchronous request through
 * the {@code registerCallbackInterceptor} and {@code registerDeferredResultInterceptor}
 * methods on {@link org.springframework.web.context.request.async.WebAsyncManager
 * WebAsyncManager}. This can be done proactively on every request from
 * {@code preHandle} regardless of whether async request processing will start.
 *
 * @author Rossen Stoyanchev
 * @since 3.2
 * @see org.springframework.web.context.request.async.WebAsyncManager
 * @see org.springframework.web.context.request.async.CallableProcessingInterceptor
 * @see org.springframework.web.context.request.async.DeferredResultProcessingInterceptor
 */
public interface AsyncHandlerInterceptor extends HandlerInterceptor {
    /**
     * Called instead of {@code postHandle} and {@code afterCompletion}, when
     * the a handler is being executed concurrently.
     * <p>Implementations may use the provided request and response but should
     * avoid modifying them in ways that would conflict with the concurrent
     * execution of the handler. A typical use of this method would be to
     * clean up thread-local variables.
     * @param request the current request
     * @param response the current response
     * @param handler the handler (or {@link HandlerMethod}) that started async
     * execution, for type and/or instance examination
     * @throws Exception in case of errors
     */
    void afterConcurrentHandlingStarted(HttpServletRequest request, HttpServletResponse response, Object handler)
            throws Exception;
}


攔截器和過濾器之間的區(qū)別

攔截器看起來很像servlet過濾器疆液,為什么Spring不采用默認(rèn)的Java解決方案一铅?這其中主要區(qū)別就是兩者的作用域的問題。過濾器只能在servlet容器下使用堕油。而我們的Spring容器不一定運(yùn)行在web環(huán)境中潘飘,在這種情況下過濾器就不好使了肮之,而攔截器依然可以在Spring容器中調(diào)用。

Spring通過攔截器為請求提供了一個更細(xì)粒度的控制卜录。就像我們之前看到的那樣戈擒,它們可以在controller對請求處理之前或之后被調(diào)用,也可以在將渲染視圖呈現(xiàn)給用戶之后被調(diào)用艰毒。如果是過濾器的話筐高,只能在將響應(yīng)返回給最終用戶之前使用它們。

下一個不同之處在于中斷鏈執(zhí)行的難易程度丑瞧。攔截器可以通過在preHandler()方法內(nèi)返回false來簡單實現(xiàn)柑土。而在過濾器的情況下,它就變得復(fù)雜了绊汹,因為它必須處理請求和響應(yīng)對象來引發(fā)中斷稽屏,需要一些額外的動作,比如如將用戶重定向到錯誤頁面灸促。


什么是默認(rèn)的Spring攔截器诫欠?

Spring主要將攔截器用于切換操作。比如我們最常用的功能之一是區(qū)域設(shè)置更改(也就是本地化更改)浴栽。請查看org.springframework.web.servlet.i18n.LocaleChangeInterceptor類中源碼荒叼,可以通過我們所定義的語言環(huán)境解析器來對HTTP請求進(jìn)行分析來實現(xiàn)。所有區(qū)域設(shè)置解析器都會分析請求元素(headers典鸡,Cookie)被廓,以確定向用戶提供哪種本地化語言設(shè)置。

另一個本地攔截器是org.springframework.web.servlet.theme.ThemeChangeInterceptor萝玷,它允許更改視圖的主題(見此類的注釋)嫁乘。它還使用主題解析器更精確地來知道要使用的主題(參照下面preHandle方法)。它的解析器也基于請求分析(cookie球碉,會話或參數(shù))蜓斧。

/**
 * Interceptor that allows for changing the current theme on every request,
 * via a configurable request parameter (default parameter name: "theme").
 *
 * @author Juergen Hoeller
 * @since 20.06.2003
 * @see org.springframework.web.servlet.ThemeResolver
 */
public class ThemeChangeInterceptor extends HandlerInterceptorAdapter {
    /**
     * Default name of the theme specification parameter: "theme".
     */
    public static final String DEFAULT_PARAM_NAME = "theme";
    private String paramName = DEFAULT_PARAM_NAME;
    /**
     * Set the name of the parameter that contains a theme specification
     * in a theme change request. Default is "theme".
     */
    public void setParamName(String paramName) {
        this.paramName = paramName;
    }
    /**
     * Return the name of the parameter that contains a theme specification
     * in a theme change request.
     */
    public String getParamName() {
        return this.paramName;
    }
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
            throws ServletException {
        String newTheme = request.getParameter(this.paramName);
        if (newTheme != null) {
            ThemeResolver themeResolver = RequestContextUtils.getThemeResolver(request);
            if (themeResolver == null) {
                throw new IllegalStateException("No ThemeResolver found: not in a DispatcherServlet request?");
            }
            themeResolver.setThemeName(request, response, newTheme);
        }
        // Proceed in any case.
        return true;
    }
}


在Spring中自定義處理程序攔截器

我們寫一個例子來簡單實現(xiàn)HandlerInterceptor。一個樂透彩票的場景睁冬,這個自定義的攔截器將分析每個請求挎春,并決定是否是彩票的“l(fā)ottery winner”。為了簡化代碼邏輯豆拨,只有用于生成一個隨機(jī)數(shù)并通過取模判斷是否返回0的請求直奋。

public class LotteryInterceptor implements HandlerInterceptor {

    public static final String ATTR_NAME = "lottery_winner";
    private static final Logger LOGGER = LoggerFactory.getLogger(LotteryInterceptor.class);

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception exception) throws Exception {
        LOGGER.debug("[LotteryInterceptor] afterCompletion");

    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView view) throws Exception {
        LOGGER.debug("[LotteryInterceptor] postHandle");

    }

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        LOGGER.debug("[LotteryInterceptor] preHandle");
        if (request.getSession().getAttribute(ATTR_NAME) == null) {
            Random random = new Random();
            int i = random.nextInt(10);
            request.getSession().setAttribute(ATTR_NAME, i%2 == 0);
        }
        return true;
    }

}

關(guān)于相應(yīng)controller中要展示的信息:

@Controller
public class TestController {
        private static final Logger LOGGER = LoggerFactory.getLogger(TestController.class);
    @RequestMapping(value = "/test", method = RequestMethod.GET)
    public String test(HttpServletRequest request) {
        LOGGER.debug("Controller asks, are you a lottery winner ? "+request.getSession().getAttribute(LotteryInterceptor.ATTR_NAME));
        return "test";
    }
}

如果我們嘗試訪問/test,我們將看不到攔截器的日志施禾,因為它沒有在配置中定義脚线。如果我們是使用注解來配置的webapp。我們需要將下面這個配置添加到應(yīng)用程序的上下文文件中(Springboot配置個相應(yīng)的bean就可):

<mvc:interceptors>
    <bean class="com.waitingforcode.interceptors.LotteryInterceptor" />
</mvc:interceptors>

現(xiàn)在我們可以訪問/ test頁面并檢查日志:

[LotteryInterceptor] preHandle
Controller asks, are you a lottery winner ? false
[LotteryInterceptor] postHandle
[LotteryInterceptor] afterCompletion

總結(jié)一下弥搞,攔截器是一種可以應(yīng)用到整個Spring生態(tài)系統(tǒng)中的servlet過濾器邮绿。它們可以在請求之前或之后啟動渠旁,也可以在視圖呈現(xiàn)之后啟動。它們也可以通過AsyncHandlerInterceptor接口的實現(xiàn)達(dá)到異步處理的效果斯碌。

原文:Spring5源碼解析-Spring中的處理攔截器
極樂科技:知乎專欄

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末一死,一起剝皮案震驚了整個濱河市肛度,隨后出現(xiàn)的幾起案子傻唾,更是在濱河造成了極大的恐慌,老刑警劉巖承耿,帶你破解...
    沈念sama閱讀 216,324評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件冠骄,死亡現(xiàn)場離奇詭異,居然都是意外死亡加袋,警方通過查閱死者的電腦和手機(jī)凛辣,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,356評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來职烧,“玉大人扁誓,你說我怎么就攤上這事∈粗” “怎么了蝗敢?”我有些...
    開封第一講書人閱讀 162,328評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長足删。 經(jīng)常有香客問我寿谴,道長,這世上最難降的妖魔是什么失受? 我笑而不...
    開封第一講書人閱讀 58,147評論 1 292
  • 正文 為了忘掉前任讶泰,我火速辦了婚禮,結(jié)果婚禮上拂到,老公的妹妹穿的比我還像新娘痪署。我一直安慰自己,他們只是感情好兄旬,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,160評論 6 388
  • 文/花漫 我一把揭開白布狼犯。 她就那樣靜靜地躺著,像睡著了一般辖试。 火紅的嫁衣襯著肌膚如雪辜王。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,115評論 1 296
  • 那天罐孝,我揣著相機(jī)與錄音呐馆,去河邊找鬼。 笑死莲兢,一個胖子當(dāng)著我的面吹牛汹来,可吹牛的內(nèi)容都是我干的续膳。 我是一名探鬼主播,決...
    沈念sama閱讀 40,025評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼收班,長吁一口氣:“原來是場噩夢啊……” “哼坟岔!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起摔桦,我...
    開封第一講書人閱讀 38,867評論 0 274
  • 序言:老撾萬榮一對情侶失蹤社付,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后邻耕,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體鸥咖,經(jīng)...
    沈念sama閱讀 45,307評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,528評論 2 332
  • 正文 我和宋清朗相戀三年兄世,在試婚紗的時候發(fā)現(xiàn)自己被綠了啼辣。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,688評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡御滩,死狀恐怖鸥拧,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情削解,我是刑警寧澤富弦,帶...
    沈念sama閱讀 35,409評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站钠绍,受9級特大地震影響舆声,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜柳爽,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,001評論 3 325
  • 文/蒙蒙 一媳握、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧磷脯,春花似錦蛾找、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,657評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至俩功,卻和暖如春幻枉,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背诡蜓。 一陣腳步聲響...
    開封第一講書人閱讀 32,811評論 1 268
  • 我被黑心中介騙來泰國打工熬甫, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人蔓罚。 一個月前我還...
    沈念sama閱讀 47,685評論 2 368
  • 正文 我出身青樓椿肩,卻偏偏與公主長得像瞻颂,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子郑象,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,573評論 2 353

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理贡这,服務(wù)發(fā)現(xiàn),斷路器厂榛,智...
    卡卡羅2017閱讀 134,651評論 18 139
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,803評論 6 342
  • spring mvc 工作機(jī)制(原理): DispatcherServlet主要用作職責(zé)調(diào)度工作盖矫,本身主要用于控制...
    java大濕兄閱讀 1,894評論 5 24
  • 什么是Spring Spring是一個開源的Java EE開發(fā)框架。Spring框架的核心功能可以應(yīng)用在任何Jav...
    jemmm閱讀 16,461評論 1 133
  • 2016年的最后一天噪沙。 2016年炼彪,跟之前的每一年都幾乎相似,卻又完全不同正歼。希臘哲人早就說過了:人永遠(yuǎn)無法踏進(jìn)同一...
    珠海兔子閱讀 356評論 0 1