日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

springboot實現(xiàn)攔截器的3種方式及異步執(zhí)行的思考

瀏覽:28日期:2023-02-26 17:33:28
目錄springboot 攔截器springboot 入門案例maven 引入啟動類定義 Controller攔截器定義基于 Aspect基于 HandlerInterceptor基于 ResponseBodyAdvice測試異步執(zhí)行定義異步線程池異步執(zhí)行的 Controller思考測試反思springboot 攔截器

實際項目中,我們經(jīng)常需要輸出請求參數(shù),響應(yīng)結(jié)果,方法耗時,統(tǒng)一的權(quán)限校驗等。

本文首先為大家介紹 HTTP 請求中三種常見的攔截實現(xiàn),并且比較一下其中的差異。(1)基于 Aspect 的攔截器(2)基于 HandlerInterceptor 的攔截器(3)基于 ResponseBodyAdvice 的攔截器

推薦閱讀:

統(tǒng)一日志框架: https://github.com/houbb/auto-log

springboot實現(xiàn)攔截器的3種方式及異步執(zhí)行的思考

springboot 入門案例

為了便于大家學(xué)習(xí),我們首先從最基本的 springboot 例子講起。

maven 引入

引入必須的 jar 包。

<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>1.5.9.RELEASE</version></parent><dependencies> <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency><groupId>org.aspectj</groupId><artifactId>aspectjrt</artifactId><version>1.8.10</version> </dependency> <dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId><version>1.8.10</version> </dependency></dependencies><!-- Package as an executable jar --><build> <plugins><plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId></plugin> </plugins></build>啟動類

實現(xiàn)最簡單的啟動類。

@SpringBootApplicationpublic class Application { public static void main(String[] args) {SpringApplication.run(Application.class, args); }}定義 Controller

為了演示方便,我們首先實現(xiàn)一個簡單的 controller。

@RestControllerpublic class IndexController { @RequestMapping('/index') public AsyncResp index() {AsyncResp asyncResp = new AsyncResp();asyncResp.setResult('ok');asyncResp.setRespCode('00');asyncResp.setRespDesc('成功');System.out.println('IndexController#index:' + asyncResp);return asyncResp; }}

其中 AsyncResp 的定義如下:

public class AsyncResp { private String respCode; private String respDesc; private String result; // getter & setter & toString()}攔截器定義基于 Aspect

import org.aspectj.lang.ProceedingJoinPoint;import org.aspectj.lang.annotation.Around;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Pointcut;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.context.annotation.EnableAspectJAutoProxy;import org.springframework.stereotype.Component;import java.util.Arrays;/** * * @author binbin.hou * @since 1.0.0 */@Aspect@Component@EnableAspectJAutoProxypublic class AspectLogInterceptor { /** * 日志實例 * @since 1.0.0 */ private static final Logger LOG = LoggerFactory.getLogger(AspectLogInterceptor.class); /** * 攔截 controller 下所有的 public方法 */ @Pointcut('execution(public * com.github.houbb.springboot.learn.aspect.controller..*(..))') public void pointCut() {// } /** * 攔截處理 * * @param point point 信息 * @return result * @throws Throwable if any */ @Around('pointCut()') public Object around(ProceedingJoinPoint point) throws Throwable {try { //1. 設(shè)置 MDC // 獲取當前攔截的方法簽名 String signatureShortStr = point.getSignature().toShortString(); //2. 打印入?yún)⑿畔? Object[] args = point.getArgs(); LOG.info('{} 參數(shù): {}', signatureShortStr, Arrays.toString(args)); //3. 打印結(jié)果 Object result = point.proceed(); LOG.info('{} 結(jié)果: {}', signatureShortStr, result); return result;} finally { // 移除 mdc} }}

這種實現(xiàn)的優(yōu)點是比較通用,可以結(jié)合注解實現(xiàn)更加靈活強大的功能。

是個人非常喜歡的一種方式。主要用途:(1)日志的出參/入?yún)ⅲ?)統(tǒng)一設(shè)置 TraceId(3)方法的調(diào)用耗時統(tǒng)計

基于 HandlerInterceptor

import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.stereotype.Component;import org.springframework.web.servlet.HandlerInterceptor;import org.springframework.web.servlet.ModelAndView;import javax.servlet.DispatcherType;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;/** * @author binbin.hou * @since 1.0.0 */@Componentpublic class LogHandlerInterceptor implements HandlerInterceptor { private Logger logger = LoggerFactory.getLogger(LogHandlerInterceptor.class); @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {// 統(tǒng)一的權(quán)限校驗、路由等logger.info('LogHandlerInterceptor#preHandle 請求地址:{}', request.getRequestURI());if (request.getDispatcherType().equals(DispatcherType.ASYNC)) { return true;}return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {logger.info('LogHandlerInterceptor#postHandle 調(diào)用'); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { }}

然后需要指定對應(yīng)的 url 和攔截器之間的關(guān)系才會生效:

import com.github.houbb.springboot.learn.aspect.aspect.LogHandlerInterceptor;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.context.annotation.Configuration;import org.springframework.web.servlet.config.annotation.InterceptorRegistry;import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;/** * spring mvc 配置 * @since 1.0.0 */@Configurationpublic class SpringMvcConfig extends WebMvcConfigurerAdapter { @Autowired private LogHandlerInterceptor logHandlerInterceptor; @Override public void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(logHandlerInterceptor).addPathPatterns('/**').excludePathPatterns('/version');super.addInterceptors(registry); }}

這種方式的優(yōu)點就是可以根據(jù) url 靈活指定不同的攔截器。缺點是主要用于 Controller 層。

基于 ResponseBodyAdvice

此接口有beforeBodyWrite方法,參數(shù)body是響應(yīng)對象response中的響應(yīng)體,那么我們就可以用此方法來對響應(yīng)體做一些統(tǒng)一的操作。

比如加密,簽名等。

import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.core.MethodParameter;import org.springframework.http.MediaType;import org.springframework.http.converter.HttpMessageConverter;import org.springframework.http.server.ServerHttpRequest;import org.springframework.http.server.ServerHttpResponse;import org.springframework.http.server.ServletServerHttpRequest;import org.springframework.web.bind.annotation.ControllerAdvice;import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;import javax.servlet.http.HttpServletRequest;/** * @author binbin.hou * @since 1.0.0 */@ControllerAdvicepublic class MyResponseBodyAdvice implements ResponseBodyAdvice<Object> { /** * 日志實例 * @since 1.0.0 */ private static final Logger LOG = LoggerFactory.getLogger(MyResponseBodyAdvice.class); @Override public boolean supports(MethodParameter methodParameter, Class aClass) {//這個地方如果返回false, 不會執(zhí)行 beforeBodyWrite 方法return true; } @Override public Object beforeBodyWrite(Object resp, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {String uri = serverHttpRequest.getURI().getPath();LOG.info('MyResponseBodyAdvice#beforeBodyWrite 請求地址:{}', uri);ServletServerHttpRequest servletServerHttpRequest = (ServletServerHttpRequest) serverHttpRequest;HttpServletRequest servletRequest = servletServerHttpRequest.getServletRequest();// 可以做統(tǒng)一的攔截器處理// 可以對結(jié)果做動態(tài)修改等LOG.info('MyResponseBodyAdvice#beforeBodyWrite 響應(yīng)結(jié)果:{}', resp);return resp; }}測試

我們啟動應(yīng)用,頁面訪問:http://localhost:18080/index頁面響應(yīng):{'respCode':'00','respDesc':'成功','result':'ok'}

后端日志:

c.g.h.s.l.a.a.LogHandlerInterceptor : LogHandlerInterceptor#preHandle 請求地址:/indexc.g.h.s.l.a.aspect.AspectLogInterceptor : IndexController.index() 參數(shù): []IndexController#index:AsyncResp{respCode=’00’, respDesc=’成功’, result=’ok’}c.g.h.s.l.a.aspect.AspectLogInterceptor : IndexController.index() 結(jié)果: AsyncResp{respCode=’00’, respDesc=’成功’, result=’ok’}c.g.h.s.l.a.aspect.MyResponseBodyAdvice : MyResponseBodyAdvice#beforeBodyWrite 請求地址:/indexc.g.h.s.l.a.aspect.MyResponseBodyAdvice : MyResponseBodyAdvice#beforeBodyWrite 響應(yīng)結(jié)果:AsyncResp{respCode=’00’, respDesc=’成功’, result=’ok’}c.g.h.s.l.a.a.LogHandlerInterceptor : LogHandlerInterceptor#postHandle 調(diào)用

這里執(zhí)行的先后順序也比較明確,此處不再贅述。

異步執(zhí)行

當然,如果只是上面這些內(nèi)容,并不是本篇文章的重點。接下來,我們一起來看下,如果引入了異步執(zhí)行會怎么樣。

定義異步線程池

springboot 中定義異步線程池,非常簡單。

import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.core.task.AsyncTaskExecutor;import org.springframework.scheduling.annotation.EnableAsync;import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;/** * 請求異步處理配置 * * @author binbin.hou */@Configuration@EnableAsyncpublic class SpringAsyncConfig { @Bean(name = 'asyncPoolTaskExecutor') public AsyncTaskExecutor taskExecutor() {ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();executor.setMaxPoolSize(10);executor.setQueueCapacity(10);executor.setCorePoolSize(10);executor.setWaitForTasksToCompleteOnShutdown(true);return executor; }}異步執(zhí)行的 Controller

@RestControllerpublic class MyAsyncController extends BaseAsyncController<String> { @Override protected String process(HttpServletRequest request) {return 'ok'; } @RequestMapping('/async') public AsyncResp hello(HttpServletRequest request) {AsyncResp resp = super.execute(request);System.out.println('Controller#async 結(jié)果:' + resp);return resp; }}

其中 BaseAsyncController 的實現(xiàn)如下:

@RestControllerpublic abstract class BaseAsyncController<T> { protected abstract T process(HttpServletRequest request); @Autowired private AsyncTaskExecutor taskExecutor; protected AsyncResp execute(HttpServletRequest request) {// 異步響應(yīng)結(jié)果AsyncResp resp = new AsyncResp();try { taskExecutor.execute(new Runnable() {@Overridepublic void run() { try {T result = process(request);resp.setRespCode('00');resp.setRespDesc('成功');resp.setResult(result.toString()); } catch (Exception exception) {resp.setRespCode('98');resp.setRespDesc('任務(wù)異常'); }} });} catch (TaskRejectedException e) { resp.setRespCode('99'); resp.setRespDesc('任務(wù)拒絕');}return resp; }}

execute 的實現(xiàn)也比較簡單:(1)主線程創(chuàng)建一個 AsyncResp,用于返回。(2)線程池異步執(zhí)行具體的子類方法,并且設(shè)置對應(yīng)的值。

思考

接下來,問大家一個問題。如果我們請求 http://localhost:18080/async,那么:(1)頁面得到的返回值是什么?(2)Aspect 日志輸出的返回值是?(3)ResponseBodyAdvice 日志輸出的返回值是什么?你可以在這里稍微停一下,記錄下你的答案。

測試

我們頁面請求 http://localhost:18080/async。

頁面響應(yīng)如下:

{'respCode':'00','respDesc':'成功','result':'ok'}

后端的日志:

c.g.h.s.l.a.a.LogHandlerInterceptor : LogHandlerInterceptor#preHandle 請求地址:/asyncc.g.h.s.l.a.aspect.AspectLogInterceptor : MyAsyncController.hello(..) 參數(shù): [org.apache.catalina.connector.RequestFacade@7e931750]Controller#async 結(jié)果:AsyncResp{respCode=’null’, respDesc=’null’, result=’null’}c.g.h.s.l.a.aspect.AspectLogInterceptor : MyAsyncController.hello(..) 結(jié)果: AsyncResp{respCode=’null’, respDesc=’null’, result=’null’}c.g.h.s.l.a.aspect.MyResponseBodyAdvice : MyResponseBodyAdvice#beforeBodyWrite 請求地址:/asyncc.g.h.s.l.a.aspect.MyResponseBodyAdvice : MyResponseBodyAdvice#beforeBodyWrite 響應(yīng)結(jié)果:AsyncResp{respCode=’00’, respDesc=’成功’, result=’ok’}c.g.h.s.l.a.a.LogHandlerInterceptor : LogHandlerInterceptor#postHandle 調(diào)用

對比一下,可以發(fā)現(xiàn)我們上面問題的答案:(1)頁面得到的返回值是什么?

{'respCode':'00','respDesc':'成功','result':'ok'}

可以獲取到異步執(zhí)行完成的結(jié)果。(2)Aspect 日志輸出的返回值是?

AsyncResp{respCode=’null’, respDesc=’null’, result=’null’}

無法獲取異步結(jié)果。(3)ResponseBodyAdvice 日志輸出的返回值是什么?

AsyncResp{respCode=’00’, respDesc=’成功’, result=’ok’}

可以獲取到異步執(zhí)行完成的結(jié)果。

反思

可以發(fā)現(xiàn),spring 對于頁面的響應(yīng)也許和我們想的有些不一樣,并不是直接獲取同步結(jié)果。寫到這里,發(fā)現(xiàn)自己對于 mvc 的理解一直只是停留在表面,沒有真正理解整個流程。Aspect 的形式在很多框架中都會使用,不過這里會發(fā)現(xiàn)無法獲取異步的執(zhí)行結(jié)果,存在一定問題。

到此這篇關(guān)于springboot實現(xiàn)攔截器的3種方式及異步執(zhí)行的思考的文章就介紹到這了,更多相關(guān)springboot 攔截器內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標簽: Spring
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
91九色精品| 国产午夜精品一区二区三区欧美 | 国产亚洲在线| 日韩影院精彩在线| 日韩欧美2区| 精品久久久亚洲| 欧洲激情综合| 日韩精品欧美大片| 精品三级av| 免费国产自久久久久三四区久久| 丝瓜av网站精品一区二区| 日本不卡不码高清免费观看| 国产乱子精品一区二区在线观看| 国产一区二区三区四区二区| 欧美成人久久| 人人爱人人干婷婷丁香亚洲| sm捆绑调教国产免费网站在线观看| 亚洲伦乱视频| 亚洲区国产区| 水蜜桃精品av一区二区| 国产精品日本| 久久不见久久见中文字幕免费| 麻豆视频在线观看免费网站黄| 2023国产精品久久久精品双| 日本v片在线高清不卡在线观看| 麻豆精品蜜桃视频网站| 久久免费国产| 97成人在线| 香蕉人人精品| 久久狠狠亚洲综合| 国产伊人精品| 国产精品videossex| 国产高清久久| 精品72久久久久中文字幕| 久久成人亚洲| 国产高潮在线| 日韩精品一区二区三区免费视频| 成人欧美一区二区三区的电影| 视频一区视频二区在线观看| 九九久久国产| 日韩精品高清不卡| 成人久久久久| 国产免费播放一区二区| 欧美~级网站不卡| 国产精品一区三区在线观看| 狠狠干综合网| 午夜av不卡| 欧美日韩伊人| 男女性色大片免费观看一区二区| 国产一区二区三区四区五区| 亚洲一区导航| 亚洲91视频| 精品国产成人| 日本aⅴ免费视频一区二区三区| 亚洲国产日韩欧美在线| 黄色精品视频| 久久精品 人人爱| 视频一区视频二区在线观看| 欧美中文字幕一区二区| 久久久精品区| 欧美精品国产一区| 日韩精品一区二区三区中文字幕| 国产精品三上| 国产国产精品| 久久久9色精品国产一区二区三区| 国产精品第一国产精品| 日韩区一区二| 亚洲一二av| aa亚洲婷婷| 欧美在线资源| 狠狠久久婷婷| 免费视频一区三区| 精品香蕉视频| 麻豆精品视频在线观看| 国产精品探花在线观看| 欧美亚洲一区二区三区| 日本亚洲最大的色成网站www| 日韩午夜av| 欧美日韩国产在线观看网站| 久久中文字幕av一区二区不卡| 成人在线黄色| 久久久久亚洲精品中文字幕| 国产欧美日韩精品一区二区免费| 日本一区免费网站| 日韩二区在线观看| 91精品国产自产在线丝袜啪| 日欧美一区二区| 免费在线观看日韩欧美| 午夜在线精品| 免费人成精品欧美精品| 亚洲欧美日韩国产一区| 免费看精品久久片| 亚洲精品第一| 日韩1区2区日韩1区2区| 久久国产精品免费精品3p| 国产精品分类| 丁香婷婷久久| 欧美aa在线观看| 国产一在线精品一区在线观看| 欧美日韩水蜜桃| 在线亚洲观看| 日韩精品中文字幕一区二区| 日本成人在线不卡视频| 国产极品模特精品一二| 国产欧美一区二区三区精品酒店| 日本а中文在线天堂| 色88888久久久久久影院| 99热精品在线| 欧美日韩网址| 福利一区视频| 91精品成人| 亚洲精品无播放器在线播放| 国产欧美日韩影院| 青青青免费在线视频| 狠狠爱成人网| 91精品视频一区二区| 久久精品国产网站| 日韩国产一区| 亚洲一卡久久| 国产免费av一区二区三区| 国产中文在线播放| 每日更新成人在线视频| 欧美亚洲三区| 国产aⅴ精品一区二区四区| 99久久精品费精品国产| 亚洲一区二区三区久久久| 国产精品久久久久9999高清| 五月激情久久| 中文字幕日韩高清在线| 精品国产成人| 喷白浆一区二区| 精品国产三区在线| 国产99精品一区| 国产丝袜一区| 亚洲婷婷在线| 国产精品久久久久久模特 | 亚洲一二三区视频| 国产一区二区精品久| 亚洲欧美激情诱惑| 久久精品网址| 一区二区电影| 久久久久国产| 国产精品.xx视频.xxtv| 久久国产主播| 国产伦理久久久久久妇女| 91九色精品| 精品久久一区| 日韩精品成人| 91精品1区| 久久精品国产免费| 视频一区欧美精品| 国产高潮在线| 国产欧美在线| 免费不卡在线观看| 久久精品动漫| 久久一区精品| 日韩av影院| 1024精品久久久久久久久| 国产日韩欧美一区在线| 91久久黄色| 日韩综合精品| 老司机免费视频一区二区三区| 麻豆9191精品国产| 久久久精品五月天| 精品免费av一区二区三区| 亚洲精品在线a| 亚洲成人精选| 91精品国产福利在线观看麻豆| 国产精品激情电影| 日韩1区2区3区| 丝袜美腿成人在线| 性欧美69xoxoxoxo| 日韩欧美国产精品综合嫩v| 日韩高清不卡在线| 国产农村妇女精品一区二区| 久久国产日本精品| www.九色在线| 美女视频黄免费的久久| 婷婷综合电影| 亚洲精选久久| 丝袜亚洲精品中文字幕一区| 婷婷综合社区| 丝袜av一区| 另类专区亚洲| 国内精品麻豆美女在线播放视频| 国产另类在线| 久久精品97| 久久国内精品自在自线400部| 中文字幕免费一区二区| 丝袜美腿亚洲一区二区图片| 国产午夜精品一区二区三区欧美| 久久在线免费| 99精品视频在线观看免费播放| 日韩电影免费网址| 日韩欧美一区二区三区免费观看| 色偷偷色偷偷色偷偷在线视频| 麻豆理论在线观看| 日韩亚洲一区在线| 久久久精品网| 亚洲精品一区二区在线看|