spring-boot學習筆記之Web攔截器

spring-boot在web層面使用了spring mvc的攔截器功能,并沒有做其他處理沐兵,故我們只要熟悉mvc的攔截器,自然而然可以將攔截器加入到spring-boot上

攔截器接口
package org.springframework.web.servlet;  
public interface HandlerInterceptor {  
    boolean preHandle(  
            HttpServletRequest request, HttpServletResponse response,   
            Object handler)   
            throws Exception;  
  
    void postHandle(  
            HttpServletRequest request, HttpServletResponse response,   
            Object handler, ModelAndView modelAndView)   
            throws Exception;  
  
    void afterCompletion(  
            HttpServletRequest request, HttpServletResponse response,   
            Object handler, Exception ex)  
            throws Exception;  
}   

我們可能注意到攔截器一個有3個回調(diào)方法响迂,而一般的過濾器Filter才兩個蹋笼,這是怎么回事呢?馬上分析蜕该。

preHandle****:預處理回調(diào)方法犁柜,實現(xiàn)處理器的預處理(如登錄檢查),第三個參數(shù)為響應的處理器(如我們上一章的Controller實現(xiàn))蛇损;
返回值:true表示繼續(xù)流程(如調(diào)用下一個攔截器或處理器)赁温;
false表示流程中斷(如登錄檢查失敗)淤齐,不會繼續(xù)調(diào)用其他的攔截器或處理器股囊,此時我們需要通過response來產(chǎn)生響應;
postHandle****:
后處理回調(diào)方法更啄,實現(xiàn)處理器的后處理(但在渲染視圖之前)稚疹,此時我們可以通過modelAndView(模型和視圖對象)對模型數(shù)據(jù)進行處理或?qū)σ晥D進行處理,modelAndView也可能為null祭务。
afterCompletion****:整個請求處理完畢回調(diào)方法内狗,即在視圖渲染完畢時回調(diào),如性能監(jiān)控中我們可以在此記錄結(jié)束時間并輸出消耗時間义锥,還可以進行一些資源清理柳沙,類似于try-catch-finally中的finally,但僅調(diào)用處理器執(zhí)行鏈中preHandle返回true的攔截器的afterCompletion拌倍。

攔截運行圖
正常流程
中斷流程

中斷流程中赂鲤,比如是HandlerInterceptor2中斷的流程(preHandle返回false),此處僅調(diào)用它之前攔截器的preHandle返回true的afterCompletion方法柱恤。

DispatcherServlet內(nèi)部工作:


//doDispatch方法
//1数初、處理器攔截器的預處理(正序執(zhí)行)
HandlerInterceptor[] interceptors = mappedHandler.getInterceptors();
if (interceptors != null) {
    for (int i = 0; i < interceptors.length; i++) {
    HandlerInterceptor interceptor = interceptors[i];
        if (!interceptor.preHandle(processedRequest, response, mappedHandler.getHandler())) {
            //1.1、失敗時觸發(fā)afterCompletion的調(diào)用
            triggerAfterCompletion(mappedHandler, interceptorIndex, processedRequest, response, null);
            return;
        }
        interceptorIndex = i;//1.2梗顺、記錄當前預處理成功的索引
}
}
//2泡孩、處理器適配器調(diào)用我們的處理器
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
//當我們返回null或沒有返回邏輯視圖名時的默認視圖名翻譯(詳解4.15.5 RequestToViewNameTranslator)
if (mv != null && !mv.hasView()) {
    mv.setViewName(getDefaultViewName(request));
}
//3、處理器攔截器的后處理(逆序)
if (interceptors != null) {
for (int i = interceptors.length - 1; i >= 0; i--) {
      HandlerInterceptor interceptor = interceptors[i];
      interceptor.postHandle(processedRequest, response, mappedHandler.getHandler(), mv);
}
}
//4寺谤、視圖的渲染
if (mv != null && !mv.wasCleared()) {
render(mv, processedRequest, response);
    if (errorView) {
        WebUtils.clearErrorRequestAttributes(request);
}
//5仑鸥、觸發(fā)整個請求處理完畢回調(diào)方法afterCompletion
triggerAfterCompletion(mappedHandler, interceptorIndex, processedRequest, response, null);



// triggerAfterCompletion方法
private void triggerAfterCompletion(HandlerExecutionChain mappedHandler, int interceptorIndex,
            HttpServletRequest request, HttpServletResponse response, Exception ex) throws Exception {
        // 5吮播、觸發(fā)整個請求處理完畢回調(diào)方法afterCompletion (逆序從1.2中的預處理成功的索引處的攔截器執(zhí)行)
        if (mappedHandler != null) {
            HandlerInterceptor[] interceptors = mappedHandler.getInterceptors();
            if (interceptors != null) {
                for (int i = interceptorIndex; i >= 0; i--) {
                    HandlerInterceptor interceptor = interceptors[i];
                    try {
                        interceptor.afterCompletion(request, response, mappedHandler.getHandler(), ex);
                    }
                    catch (Throwable ex2) {
                        logger.error("HandlerInterceptor.afterCompletion threw exception", ex2);
                    }
                }
            }
        }
    }


spring boot添加攔截器
  • 方法一
@SpringBootApplication
public class Application {

    public static void main(String[] args) {
        SpringApplication application = new SpringApplication(Application.class);
        // application.addListeners(new ApplicationListenerEnvironmentPrepared());
        // application.addListeners(new ApplicationListenerFailed());
        // application.addListeners(new ApplicationListenerPrepared());
        // application.addListeners(new ApplicationListenerStarted());
        application.run(args);
    }

    @Configuration
    static class WebMvcConfigurer extends WebMvcConfigurerAdapter {

        public void addInterceptors(InterceptorRegistry registry) {
            registry.addInterceptor(new HandlerInterceptorAdapter() {

                @Override
                public boolean preHandle(HttpServletRequest request, HttpServletResponse response,
                                         Object handler) throws Exception {
                    request.getContextPath();
                    System.out.println("interceptor====");
                    return true;
                }
            }).addPathPatterns("/*");
        }
    }
}
  • 方法二
@Configuration
public class WebMvcConfigurer extends WebMvcConfigurerAdapter {

    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new HandlerInterceptorAdapter() {

            @Override
            public boolean preHandle(HttpServletRequest request, HttpServletResponse response,
                                     Object handler) throws Exception {
                System.out.println("interceptor====222");
                return true;
            }
        }).addPathPatterns("/*");
    }
}

總結(jié):只要能被springboot掃描到即可

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市锈候,隨后出現(xiàn)的幾起案子薄料,更是在濱河造成了極大的恐慌,老刑警劉巖泵琳,帶你破解...
    沈念sama閱讀 222,183評論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件摄职,死亡現(xiàn)場離奇詭異,居然都是意外死亡获列,警方通過查閱死者的電腦和手機谷市,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,850評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來击孩,“玉大人迫悠,你說我怎么就攤上這事」遥” “怎么了创泄?”我有些...
    開封第一講書人閱讀 168,766評論 0 361
  • 文/不壞的土叔 我叫張陵,是天一觀的道長括蝠。 經(jīng)常有香客問我鞠抑,道長,這世上最難降的妖魔是什么忌警? 我笑而不...
    開封第一講書人閱讀 59,854評論 1 299
  • 正文 為了忘掉前任搁拙,我火速辦了婚禮,結(jié)果婚禮上箕速,老公的妹妹穿的比我還像新娘。我一直安慰自己徙赢,他們只是感情好庭呜,可當我...
    茶點故事閱讀 68,871評論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著搀庶,像睡著了一般秸架。 火紅的嫁衣襯著肌膚如雪东抹。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,457評論 1 311
  • 那天附迷,我揣著相機與錄音喊儡,去河邊找鬼。 笑死箩朴,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的埠居。 我是一名探鬼主播,決...
    沈念sama閱讀 40,999評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼唠倦,長吁一口氣:“原來是場噩夢啊……” “哼冈止!你這毒婦竟也來了闺属?” 一聲冷哼從身側(cè)響起诗眨,我...
    開封第一講書人閱讀 39,914評論 0 277
  • 序言:老撾萬榮一對情侶失蹤芋簿,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后磷支,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體抵皱,經(jīng)...
    沈念sama閱讀 46,465評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,543評論 3 342
  • 正文 我和宋清朗相戀三年伤为,在試婚紗的時候發(fā)現(xiàn)自己被綠了味滞。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片剑鞍。...
    茶點故事閱讀 40,675評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡爽醋,死狀恐怖遂赠,靈堂內(nèi)的尸體忽然破棺而出抑诸,到底是詐尸還是另有隱情奸绷,我是刑警寧澤,帶...
    沈念sama閱讀 36,354評論 5 351
  • 正文 年R本政府宣布层玲,位于F島的核電站号醉,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏憨降。R本人自食惡果不足惜父虑,卻給世界環(huán)境...
    茶點故事閱讀 42,029評論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望授药。 院中可真熱鬧士嚎,春花似錦、人聲如沸悔叽。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,514評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽娇澎。三九已至笨蚁,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背括细。 一陣腳步聲響...
    開封第一講書人閱讀 33,616評論 1 274
  • 我被黑心中介騙來泰國打工伪很, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人奋单。 一個月前我還...
    沈念sama閱讀 49,091評論 3 378
  • 正文 我出身青樓锉试,卻偏偏與公主長得像,于是被迫代替她去往敵國和親览濒。 傳聞我的和親對象是個殘疾皇子呆盖,可洞房花燭夜當晚...
    茶點故事閱讀 45,685評論 2 360

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

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務發(fā)現(xiàn)贷笛,斷路器应又,智...
    卡卡羅2017閱讀 134,707評論 18 139
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,859評論 6 342
  • application的配置屬性。 這些屬性是否生效取決于對應的組件是否聲明為Spring應用程序上下文里的Bea...
    新簽名閱讀 5,381評論 1 27
  • 這些屬性是否生效取決于對應的組件是否聲明為 Spring 應用程序上下文里的 Bean(基本是自動配置的)乏苦,為一個...
    發(fā)光的魚閱讀 1,432評論 0 14
  • 中國的春節(jié)株扛,要求所有的人都停下手頭的工作,一起來歡呼和慶祝邑贴。 當然席里,人與人之間的差異存在導致不可能全部都快樂。 世...
    愿你我都幸福閱讀 396評論 2 1