<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
本文基於的Spring Boot的版本是2.6.7 。
Spring MVC
中的攔截器(Interceptor
)類似於ServLet中的過濾器(Filter
),它主要用於攔截使用者請求並作出相應的處理。例如通過攔截器可以進行許可權驗證、記錄請求資訊的紀錄檔、判斷使用者是否登入等。
一個攔截器,只有preHandle
方法返回true
,postHandle
、afterCompletion
才有可能被執行;如果preHandle
方法返回false
,則該攔截器的postHandle
、afterCompletion
必然不會被執行。攔截器不是Filter,卻實現了Filter的功能,其原理在於:
1.所有的攔截器(Interceptor)
和處理器(Handler)
都註冊在HandlerMapping
中。
2.Spring MVC
中所有的請求都是由DispatcherServlet
分發的。
3.當請求進入DispatcherServlet.doDispatch()
時候,首先會得到處理該請求的Handler
(即Controller
中對應的方法)以及所有攔截該請求的攔截器。攔截器就是在這裡被呼叫開始工作的。
如果在Interceptor1.preHandle中報錯或返回false ,那麼接下來的流程就會被中斷,但注意被執行過的攔截器的afterCompletion仍然會執行。
攔截器本質上是面向切面程式設計(AOP),符合橫切關注點的功能都可以放在攔截器中來實現,主要的應用場景包括:
自定義一個攔截器非常簡單,只需要實現HandlerInterceptor
這個介面即可,該介面有三個可以實現的方法,如下:
preHandle()
方法:改方法會在控制方法前執行,器返回值表示是否知道如何寫一個介面。中斷後續操作。當其返回值為true時,表示繼續向下執行;當其返回值為false
時,會中斷後續的所有操作(包括呼叫下一個攔截器和控制器類中的方法執行等 )其實想要在Spring Boot生效其實很簡單,只需要定義一個設定類,實現WebMvcConfigurer
這個介面,並且實現其中的addInterceptiors()
方法即可,程式碼演示如下:
@Configuration public class WebConfig implements WebMvcConfigurer { @Autowired private XXX xxx; @Override public void addInterceptors(InterceptorRegistry registry) { //不需要攔截的url final String[] commonExclude={}; registry.addInterceptor(xxx).excludePathPatterns(commonExclude) } }
通過攔截器防止使用者暴力請求連線,使用使用者IP來限制存取次數 。達到多少次數禁止該IP存取。
記錄使用者IP存取次數,第一次存取時在redis中建立一個有效時長1秒的key,當第二次存取時key值+1,當值大於等於5時在redis中建立一個5分鐘的key,當攔截器查詢到reids中有當前IP的key值時返回false限制使用者請求介面 。
第一步,建立一個攔截器,程式碼如下:
@Slf4j public class IpUrlLimitInterceptor implements HandlerInterceptor { @Resource RedisUtils redisUtils; private static final String LOCK_IP_URL_KEY="lock_ip_"; private static final String IP_URL_REQ_TIME="ip_url_times_"; //存取次數限制 private static final long LIMIT_TIMES=5; //限制時間 秒為單位 private static final int IP_LOCK_TIME=300; @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { log.info("request請求地址uri={},ip={}",request.getRequestURI(), IpUtils.getRequestIP(request)); if(ipIsLock(IpUtils.getRequestIP(request))){ log.info("ip存取被禁止={}",IpUtils.getRequestIP(request)); throw new Exception("當前操作過於頻繁,請5分鐘後重試"); } if (!addRequestTime(IpUtils.getRequestIP(request),request.getRequestURI())){ log.info("當前{}操作過於頻繁,請5分鐘後重試",IpUtils.getRequestIP(request)); throw new Exception("當前操作過於頻繁,請5分鐘後重試"); } return true; } private boolean addRequestTime(String ip, String uri) { String key = IP_URL_REQ_TIME+ip+uri; if(redisUtils.hasKey(key)){ long time=redisUtils.incr(key,(long)1); if(time >=LIMIT_TIMES){ redisUtils.set(LOCK_IP_URL_KEY+ip,IP_LOCK_TIME); return false; } }else { boolean set = redisUtils.set(key, (long) 1, 1); } return true; } private boolean ipIsLock(String ip) { if(redisUtils.hasKey(LOCK_IP_URL_KEY+ip)){ return true; } return false; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { HandlerInterceptor.super.postHandle(request, response, handler, modelAndView); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { HandlerInterceptor.super.afterCompletion(request, response, handler, ex); } }
第二步,定義一個獲取IP的工具類,程式碼如下:
@Slf4j public class IpUtils { public static String getRequestIP(HttpServletRequest request){ String ip = request.getHeader("x-forwarded-for"); if(ip != null && ip.length() !=0 && "unknown".equalsIgnoreCase(ip)){ // 多次反向代理後會有多個ip值,第一個ip才是真實ip if( ip.indexOf(",")!=-1 ){ ip = ip.split(",")[0]; } } if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)){ ip = request.getHeader("Proxy-Client-IP"); log.info("Proxy-Client-IP ip: " + ip); } if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) { ip = request.getHeader("HTTP_CLIENT_IP"); log.info("HTTP_CLIENT_IP ip: " + ip); } if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) { ip = request.getHeader("HTTP_X_FORWARDED_FOR"); log.info("HTTP_X_FORWARDED_FOR ip: " + ip); } if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) { ip = request.getHeader("X-Real-IP"); log.info("X-Real-IP ip: " + ip); } if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) { ip = request.getRemoteAddr(); log.info("getRemoteAddr ip: " + ip); } return ip; } }
第二步,在Spring Boot中設定這個攔截器,程式碼如下:
@Configuration public class WebConfig implements WebMvcConfigurer { @Bean IpUrlLimitInterceptor getIpUrlLimitInterceptor(){ return new IpUrlLimitInterceptor(); }; @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(getIpUrlLimitInterceptor()).addPathPatterns("/**"); } }
該攔截器是全域性生效的,可能有些場景某個介面不需要限制,這樣我們可以把這個攔截器改造成註解方式應用。某些介面需要則加上註解即可。
以上就是詳解SpringBoot中自定義和設定攔截器的方法的詳細內容,更多關於SpringBoot攔截器的資料請關注it145.com其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45