Filter吝梅、Interceptor、Aspect

filter惹骂、interceptor苏携、aspect

  1. Filter過濾器
  2. Interceptor攔截器
  3. Aspect切片

Filter過濾器

  • 過濾器依賴于servlet容器。在實(shí)現(xiàn)上对粪,基于函數(shù)回調(diào)右冻,它可以對(duì)幾乎所有請(qǐng)求進(jìn)行過濾,一個(gè)過濾器實(shí)例只能在容器初始化時(shí)調(diào)用一次著拭。

  • 過濾器可以攔截到方法的請(qǐng)求和響應(yīng)(ServletRequest request, ServletResponse response),并對(duì)請(qǐng)求響應(yīng)做出過濾操作纱扭。

  • 使用過濾器的目的是用來做一些過濾操作,對(duì)請(qǐng)求的數(shù)據(jù)進(jìn)行獲取儡遮,或者替換請(qǐng)求數(shù)據(jù)乳蛾,或者權(quán)限攔截檢查,比如:在過濾器中修改字符編碼鄙币;在過濾器中修改HttpServletRequest的一些參數(shù)肃叶,過濾關(guān)鍵字:過濾低俗文字、危險(xiǎn)字符等爱榔。

切面的方法說明

  • @Aspect 作用是把當(dāng)前類標(biāo)識(shí)為一個(gè)切面供容器讀取

  • @Before 標(biāo)識(shí)一個(gè)前置增強(qiáng)方法被环,相當(dāng)于BeforeAdvice的功能

  • @AfterReturning 后置增強(qiáng),相當(dāng)于AfterReturningAdvice详幽,方法退出時(shí)執(zhí)行

  • @AfterThrowing 異常拋出增強(qiáng),相當(dāng)于ThrowsAdvice

  • @After final增強(qiáng)浸锨,不管是拋出異炒狡福或者正常退出都會(huì)執(zhí)行

  • @Around 環(huán)繞增強(qiáng),相當(dāng)于MethodInterceptor

  • 除了@Around外柱搜,每個(gè)方法里都可以加或者不加參數(shù)JoinPoint迟郎,如果有用JoinPoint的地方就加,不加也可以聪蘸,JoinPoint里包含了類名宪肖、被切面的方法名表制,參數(shù)等屬性,可供讀取使用控乾。@Around參數(shù)必須為ProceedingJoinPoint么介,pjp.proceed相應(yīng)于執(zhí)行被切面的方法。@AfterReturning方法里蜕衡,可以加returning = “XXX”壤短,XXX即為在controller里方法的返回值,本例中的返回值是“first controller”慨仿。@AfterThrowing方法里久脯,可以加throwing = "XXX"

切面PointCut的切入點(diǎn)

execution切點(diǎn)函數(shù)

  • execution函數(shù)用于匹配方法執(zhí)行的連接點(diǎn),語(yǔ)法為:
execution(方法修飾符(可選)  返回類型  方法名  參數(shù)  異常模式(可選))

例如:
execution(* com.imooc.controller.UserController.*(..))  
第一個(gè)  * 代表的是所有的返回值類型镰吆,
com.imooc.controller.UserController.*代表的是com.imooc.controller包下UserController類的所有方法帘撰,
(..)代表的是所有的參數(shù)

Filter代碼實(shí)例

@Component
public class TimeFilter implements Filter {
    
    private static final Logger LOGGER = LoggerFactory.getLogger(TimeFilter.class);

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

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) servletRequest;

        String uri = request.getRequestURI();

        long start = System.currentTimeMillis();
        
        filterChain.doFilter(servletRequest,servletResponse);

        long end = System.currentTimeMillis();

        LOGGER.info("請(qǐng)求地址 : [{}], 耗時(shí) : [{}]",uri, (end-start)/3600);
    }

    @Override
    public void destroy() {
    }
}
  • 實(shí)現(xiàn)javax.servlet.Filter接口,并重寫接口的三個(gè)方法万皿,其中doFilter方法是必須實(shí)現(xiàn)的摧找,其它兩個(gè)方法是接口的默認(rèn)方法;
  • doFilter方法 提供請(qǐng)求的ServletRequest和響應(yīng)的ServletResponse對(duì)象
  • Filter中無法獲取其它對(duì)象相寇,例如Controller 即Filter中不支持對(duì)象注入(@AutoWired). 在Spring中 web容器的加載順序是:Listener -> Filter -> Servlet, 先初始化Listener慰于,然后進(jìn)行Filter初始化,再接口進(jìn)行Servlet的初始化過程唤衫。然后我們的Controller對(duì)象即在Servlet容器初始化后進(jìn)行初始化的婆赠,所以在Filter初始化時(shí),注解獲取Bean的還沒有進(jìn)行初始化佳励,即沒法注入休里。
public interface Filter {
    default void init(FilterConfig filterConfig) throws ServletException {
    }

    void doFilter(ServletRequest var1, ServletResponse var2, FilterChain var3) throws IOException, ServletException;

    default void destroy() {
    }
}

Filter配置

Filter過濾器實(shí)例已經(jīng)書寫完畢,但是使其生效需要進(jìn)行配置赃承,一般有兩種方式進(jìn)行配置:

  • 第一個(gè)方案在Filter上面加上@Component
@Component 
public class TimeFilter implements Filter {
  ...
}
  • 第二個(gè)方案配置化注冊(cè)過濾器
    該方案的特點(diǎn)就是可以細(xì)化到過濾哪些規(guī)則的URL
@Configuration
public class FilterConfig {

    @Bean
    FilterRegistrationBean<TimeFilter> filterRegistrationBean() {
        FilterRegistrationBean<TimeFilter> registrationBean = new FilterRegistrationBean();
        TimeFilter timeFilter = new TimeFilter();
        registrationBean.setFilter(timeFilter);
        registrationBean.setUrlPatterns(Arrays.asList("/*"));
        return registrationBean;
    }
}

容器啟動(dòng)輸出:

com.ams.admin.filters.TimeFilter         : init...

調(diào)用請(qǐng)求輸出

2019-09-04 14:11:27.424  INFO 66854 --- [io-10001-exec-2] com.ams.admin.filters.TimeFilter         : 請(qǐng)求地址 : [/group/single], 耗時(shí) : [10334] ms
2019-09-04 14:11:34.236  INFO 66854 --- [io-10001-exec-3] com.ams.admin.filters.TimeFilter         : 請(qǐng)求地址 : [/group/single], 耗時(shí) : [14] ms
2019-09-04 14:11:35.442  INFO 66854 --- [io-10001-exec-4] com.ams.admin.filters.TimeFilter         : 請(qǐng)求地址 : [/group/single], 耗時(shí) : [14] ms

容器銷毀輸出

2019-09-04 14:19:16.148  INFO 66854 --- [      Thread-31] o.s.s.concurrent.ThreadPoolTaskExecutor  : Shutting down ExecutorService 'applicationTaskExecutor'
2019-09-04 14:19:16.149  INFO 66854 --- [      Thread-31] com.alibaba.druid.pool.DruidDataSource   : {dataSource-3} closed
2019-09-04 14:19:16.158  INFO 66854 --- [      Thread-31] com.ams.admin.filters.TimeFilter         : destroy...

Filter 總結(jié)

  • 過濾器(Filter)是基于函數(shù)回調(diào)妙黍;
  • Filter隨web應(yīng)用的啟動(dòng)而啟動(dòng),只初始化一次瞧剖,隨web應(yīng)用的停止而銷毀拭嫁;
  • 啟動(dòng)服務(wù)器時(shí)加載過濾器的實(shí)例,并調(diào)用init()方法來初始化實(shí)例抓于;
  • 每一次請(qǐng)求時(shí)都只調(diào)用方法doFilter()進(jìn)行處理做粤;
  • 停止服務(wù)器時(shí)調(diào)用destroy()方法,銷毀實(shí)例捉撮。

Interceptor攔截器

Interceptor基于Java的反射機(jī)制怕品,屬于面向切面編程(AOP)的一種運(yùn)用,就是在一個(gè)方法前巾遭,調(diào)用一個(gè)方法肉康,或者在方法后闯估,調(diào)用一個(gè)方法。其實(shí)我們所了解的代理機(jī)制其實(shí)就是一種攔截器的實(shí)現(xiàn)吼和。而我們此處討論的攔截器是針對(duì)web框架所說的涨薪,所以依賴于web框架。

  • 在web開發(fā)中纹安,攔截器是經(jīng)常用到的功能尤辱。它可以幫我們驗(yàn)證是否登陸、預(yù)先設(shè)置數(shù)據(jù)以及統(tǒng)計(jì)方法的執(zhí)行效率等厢岂。在spring中攔截器有兩種光督,第一種是HandlerInterceptor,第二種是MethodInterceptor塔粒。HandlerInterceptor是SpringMVC中的攔截器结借,它攔截的是Http請(qǐng)求的信息,優(yōu)先于MethodInterceptor卒茬。而MethodInterceptor是springAOP的船老。前者攔截的是請(qǐng)求的地址,而后者是攔截controller中的方法圃酵,因?yàn)橄旅嬉獙spect柳畔,就不詳細(xì)講述MethodInterceptor

代碼實(shí)例

public class AmsInterceptor implements HandlerInterceptor {

    private static final Logger LOGGER = LoggerFactory.getLogger(AmsInterceptor.class);

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

        HandlerMethod method = (HandlerMethod) handler;
        String cn = method.getBean().getClass().getName();
        String mn = method.getMethod().getName();
        LOGGER.info("preHandle -> [{}]#[{}]#[{}]", cn, mn, readRaw(request.getInputStream()));
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        LOGGER.info("postHandle -> [{}]#[{}]", request.getRequestURI(), request.getPathInfo());
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        LOGGER.info("afterCompletion -> [{}]#[{}]", request.getRequestURI(), request.getPathInfo());
    }


    public static String readRaw(InputStream inputStream) {

        String result = "";
        try {
            ByteArrayOutputStream outSteam = new ByteArrayOutputStream();
            byte[] buffer = new byte[1024];

            int len;
            while ((len = inputStream.read(buffer)) != -1) {
                outSteam.write(buffer, 0, len);
            }

            outSteam.close();
            inputStream.close();

            result = new String(outSteam.toByteArray(), "UTF-8");

        } catch (IOException e) {
            e.printStackTrace();
        }

        return result;
    }
}

Interceptor配置

  • 繼承WebMvcConfigurationSupport
@Configuration
public class InterceptorConfig extends WebMvcConfigurationSupport {

    @Autowired
    AmsInterceptor amsInterceptor;

    /**
     * Override this method to add Spring MVC interceptors for
     * pre- and post-processing of controller invocation.
     *
     * @param registry
     * @see InterceptorRegistry
     */
    @Override
    protected void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(amsInterceptor).addPathPatterns(Arrays.asList("/**"));
    }
}

輸出

2019-09-05 11:27:29.585  INFO 77598 --- [io-10001-exec-4] com.ams.admin.filters.TimeFilter         : 請(qǐng)求地址 : [/group/single]
2019-09-05 11:27:29.586  INFO 77598 --- [io-10001-exec-4] c.ams.admin.interceptor.AmsInterceptor   : preHandle : [com.ams.admin.controller.AmsGroupController]#[operateEntity]#[{   "name":"三部",    "groupAmsid":"0003",    "remark":"三部",  "systemId": 1,  "operator":"maozw"}]
2019-09-05 11:27:29.586  INFO 77598 --- [io-10001-exec-4] c.a.admin.controller.AmsGroupController  : controller
2019-09-05 11:27:29.603  INFO 77598 --- [io-10001-exec-4] c.ams.admin.interceptor.AmsInterceptor   : postHandle : [/group/single]
2019-09-05 11:27:29.603  INFO 77598 --- [io-10001-exec-4] c.ams.admin.interceptor.AmsInterceptor   : afterCompletion : [/group/single]
2019-09-05 11:27:29.603  INFO 77598 --- [io-10001-exec-4] com.ams.admin.filters.TimeFilter         : 請(qǐng)求地址 : [/group/single], 耗時(shí) : [18] ms

執(zhí)行流程解釋

  • 簡(jiǎn)要說明一下Interceptor執(zhí)行流程,
  • 我們通過日志可以看出郭赐,執(zhí)行流程 TimeFilter -> preHandle -> controller -> postHandle -> afterCompletion
    我們看下SpringMVC的核心類 org.springframework.web.servlet.DispatcherServlet

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
    ...
    try {
        ...
        try {
            ...
            if (!mappedHandler.applyPreHandle(processedRequest, response)) {
                return;
            }

            // Actually invoke the handler.
            mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

            if (asyncManager.isConcurrentHandlingStarted()) {
                return;
            }

            applyDefaultViewName(processedRequest, mv);
            mappedHandler.applyPostHandle(processedRequest, response, mv);
        }
        catch (Exception ex) {
            ...
        }
    finally {
        ...
        if (asyncManager.isConcurrentHandlingStarted()) {
           // Instead of postHandle and afterCompletion
          if (mappedHandler != null) {
              mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);
          }
        }
    }
}   

以上是刪除一些無關(guān)代碼之后的:

  • mappedHandler.applyPreHandle(processedRequest, response) 這個(gè)方法執(zhí)行薪韩,就是執(zhí)行的攔截器的preHandler方法

  • mappedHandler.applyPostHandle(processedRequest, response, mv); postHandler方法的執(zhí)行捌锭,當(dāng)controller內(nèi)部有異常俘陷,posthandler方法是不會(huì)執(zhí)行的。

  • mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response); ****afterCompletion方法观谦,不管controller內(nèi)部是否有異常拉盾,都會(huì)執(zhí)行此方法;此方法還會(huì)有個(gè)Exception ex這個(gè)參數(shù)豁状;如果有異常捉偏,ex會(huì)有異常值;沒有異常 此值為null

  • 根據(jù)上面的流程我們知道ha.handle(processedRequest, response, mappedHandler.getHandler());方法才會(huì)對(duì)請(qǐng)求參數(shù)進(jìn)行組裝方法參數(shù)泻红,所以我們是在攔截器中可以拿到Http原始請(qǐng)求和響應(yīng)信息告私,也可以獲取到ioc的bean(controller)但是這個(gè)這個(gè)方法的參數(shù)是沒法通過haner對(duì)象去獲取。

  • 值得一提的事情承桥,因?yàn)槲以贗nterceptor中獲取了ServletRequest的InputStream 這個(gè)操作會(huì)導(dǎo)致后續(xù)在Controller中獲取不到流,解決辦法我會(huì)在另一篇幅中進(jìn)行說明:詳見【】

Aspect切片

AOP操作可以對(duì)操作進(jìn)行橫向的攔截,最大的優(yōu)勢(shì)在于他可以獲取執(zhí)行方法的參數(shù),對(duì)方法進(jìn)行統(tǒng)一的處理根悼。常見使用日志,事務(wù),請(qǐng)求參數(shù)安全驗(yàn)證等凶异。

  • 可以拿得到方法響應(yīng)中參數(shù)的值蜀撑,但是拿不到原始的Http請(qǐng)求和相對(duì)應(yīng)響應(yīng)的方法,基于Controller層剩彬。對(duì)于統(tǒng)一異常處理和日志記錄非常方便酷麦,有效地減少了代碼的重復(fù)率。

代碼實(shí)例

@Aspect
@Component
@Slf4j
public class ControllerAspect {

    private static final Logger LOGGER = LoggerFactory.getLogger(ControllerAspect.class);

    @Around("execution(* com.ams.admin.controller.*.*(..))")
    public Object handlerControlerMethod(ProceedingJoinPoint joinPoint) throws Throwable {

        Object[] args = joinPoint.getArgs();
        LOGGER.info("aop before... [{}]", args);
        Object proceed = joinPoint.proceed();
        LOGGER.info("aop after...");
        return proceed;
    }
}

配置使其生效

@EnableTransactionManagement(proxyTargetClass = true)

輸出

2019-09-05 15:18:06.748  INFO 80313 --- [io-10001-exec-2] com.ams.admin.filters.TimeFilter         : 請(qǐng)求地址 : [/group/single]
2019-09-05 15:18:06.750  INFO 80313 --- [io-10001-exec-2] c.ams.admin.interceptor.AmsInterceptor   : preHandle : [com.ams.admin.controller.AmsGroupController$$EnhancerBySpringCGLIB$$7f16fb78]#[operateEntity]#[{  "name":"三部",    "groupAmsid":"0003",    "remark":"三部",  "systemId": 1,  "operator":"maozw"}]
2019-09-05 15:18:06.752  INFO 80313 --- [io-10001-exec-2] com.ams.admin.aspect.ControllerAspect    : aop before... [AmsGroup(id=null, groupAmsid=0003, name=三部, systemId=1, remark=三部, createDate=null, modifyDate=null, operator=maozw)]
2019-09-05 15:18:06.755  INFO 80313 --- [io-10001-exec-2] c.a.admin.controller.AmsGroupController  : controller
2019-09-05 15:18:16.909  INFO 80313 --- [io-10001-exec-2] com.ams.admin.aspect.ControllerAspect    : aop after...
2019-09-05 15:18:16.910  INFO 80313 --- [io-10001-exec-2] c.ams.admin.interceptor.AmsInterceptor   : postHandle : [/group/single]
2019-09-05 15:18:16.910  INFO 80313 --- [io-10001-exec-2] c.ams.admin.interceptor.AmsInterceptor   : afterCompletion : [/group/single]
2019-09-05 15:18:16.911  INFO 80313 --- [io-10001-exec-2] com.ams.admin.filters.TimeFilter         : 請(qǐng)求地址 : [/group/single], 耗時(shí) : [10163] ms

小結(jié)

Filter
參數(shù):ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain
作用:可以獲取原始的http請(qǐng)求和響應(yīng)對(duì)象喉恋,無法獲取請(qǐng)求控制器

Interceptor
參數(shù):HttpServletRequest request, HttpServletResponse response, Object handler
作用:可以獲取原始的http請(qǐng)求和響應(yīng)對(duì)象沃饶,可以獲取到控制器和方法,但獲取不到方法參數(shù)

Aspect
參數(shù):ProceedingJoinPoint joinPoint
作用:無法獲取原始的http請(qǐng)求和響應(yīng)對(duì)象轻黑,可以獲取到控制器和方法糊肤,也可以獲取方法參數(shù)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市氓鄙,隨后出現(xiàn)的幾起案子馆揉,更是在濱河造成了極大的恐慌,老刑警劉巖抖拦,帶你破解...
    沈念sama閱讀 221,576評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件升酣,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡态罪,警方通過查閱死者的電腦和手機(jī)噩茄,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,515評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來复颈,“玉大人绩聘,你說我怎么就攤上這事∪颍” “怎么了君纫?”我有些...
    開封第一講書人閱讀 168,017評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)芹彬。 經(jīng)常有香客問我蓄髓,道長(zhǎng),這世上最難降的妖魔是什么舒帮? 我笑而不...
    開封第一講書人閱讀 59,626評(píng)論 1 296
  • 正文 為了忘掉前任会喝,我火速辦了婚禮,結(jié)果婚禮上玩郊,老公的妹妹穿的比我還像新娘肢执。我一直安慰自己,他們只是感情好译红,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,625評(píng)論 6 397
  • 文/花漫 我一把揭開白布预茄。 她就那樣靜靜地躺著,像睡著了一般侦厚。 火紅的嫁衣襯著肌膚如雪耻陕。 梳的紋絲不亂的頭發(fā)上拙徽,一...
    開封第一講書人閱讀 52,255評(píng)論 1 308
  • 那天,我揣著相機(jī)與錄音诗宣,去河邊找鬼膘怕。 笑死,一個(gè)胖子當(dāng)著我的面吹牛召庞,可吹牛的內(nèi)容都是我干的岛心。 我是一名探鬼主播,決...
    沈念sama閱讀 40,825評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼篮灼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼忘古!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起穿稳,我...
    開封第一講書人閱讀 39,729評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤存皂,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后逢艘,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體旦袋,經(jīng)...
    沈念sama閱讀 46,271評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,363評(píng)論 3 340
  • 正文 我和宋清朗相戀三年它改,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了疤孕。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,498評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡央拖,死狀恐怖祭阀,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情鲜戒,我是刑警寧澤专控,帶...
    沈念sama閱讀 36,183評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站遏餐,受9級(jí)特大地震影響伦腐,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜失都,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,867評(píng)論 3 333
  • 文/蒙蒙 一柏蘑、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧粹庞,春花似錦咳焚、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,338評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春督惰,著一層夾襖步出監(jiān)牢的瞬間不傅,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,458評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工赏胚, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人商虐。 一個(gè)月前我還...
    沈念sama閱讀 48,906評(píng)論 3 376
  • 正文 我出身青樓觉阅,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親秘车。 傳聞我的和親對(duì)象是個(gè)殘疾皇子典勇,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,507評(píng)論 2 359

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