当前位置:网站首页>Custom filters and interceptors implement ThreadLocal thread closure
Custom filters and interceptors implement ThreadLocal thread closure
2022-08-05 10:50:00 【HUAWEI CLOUD】
线程封闭
线程封闭一般通过以下三个方法:
- Ad-hoc线程封闭:程序控制实现,最糟糕,忽略
- 堆栈封闭:局部变量,无并发问题
- ThreadLocal线程封闭:特别好的封闭方法
方法2是最常用的,变量定义在接口内,本文主要讲解方法三,SpringBoot项目通过自定义过滤器和拦截器实现ThreadLocal线程封闭.实现Filter接口自定义过滤器和继承HandlerInterceptorAdapter自定义拦截器.
ThreadLocal线程封闭实现步骤
封装ThredLocal的方法
/** * <p>自定义RequestHolder</p></p> * * @Author zjq * @Date 2021/12 */public class RequestHolder { private final static ThreadLocal<Long> requestHolder = new ThreadLocal<>(); public static void set(Long id) { requestHolder.set(id); } public static Long get() { return requestHolder.get(); } public static void remove() { requestHolder.remove(); }}
自定义过滤器
自定义定义拦截器继承Filter接口,实现ThredLocal.add()方法
/** * <p>自定义过滤器</p> * * @Author zjq * @Date 2021/12/7 */@Slf4jpublic class HttpFilter implements Filter { /** * 为Filter初始化 提供支持 * * @param filterConfig * @throws ServletException */ @Override public void init(FilterConfig filterConfig) throws ServletException { } /** * 拦截到要执行的请求时,doFilter就会执行.这里我们可以写对请求和响应的预处理. * FilterChain把请求和响应传递给下一个 Filter处理 * * @param servletRequest * @param servletResponse * @param filterChain * @throws IOException * @throws ServletException */ @Override public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException { //把普通servlet强转成httpServlet HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest; Long threadId = Thread.currentThread().getId(); log.info("do filter,threadId:{} servletPath:{}", threadId, httpServletRequest.getServletPath()); //把当前线程id放入requestHolder RequestHolder.set(threadId); //放行 filterChain.doFilter(httpServletRequest, servletResponse); } /** * Filter 实例销毁前的准备工作 */ @Override public void destroy() { }}
自定义拦截器
自定义拦截器在线程使用完毕后移除ThredLocal中内容,避免内存溢出
/** * <p>自定义拦截器</p> * * @Author zjq * @Date 2021/12/7 */@Slf4jpublic class HttpInterceptor extends HandlerInterceptorAdapter { /** * 拦截处理程序的执行.在 HandlerMapping 确定合适的处理程序对象之后,在 HandlerAdapter 调用处理程序之前调用. * @param request * @param response * @param handler * @return * @throws Exception */ @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { log.info("preHandle执行..."); return true; } /** * 请求处理完成后(渲染视图后)的回调.将在处理程序执行的任何结果上调用,从而允许进行适当的资源清理. * @param request * @param response * @param handler * @param ex * @throws Exception */ @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { RequestHolder.remove(); log.info("afterCompletion执行..."); return; }}
Application类启动类中配置自定义过滤器和拦截器
/** * * @author zjq */@SpringBootApplicationpublic class Application extends WebMvcConfigurationSupport { public static void main(String[] args) { SpringApplication.run(ConcurrencyApplication.class, args); } /** * 自定义过滤器 * @return */ @Bean public FilterRegistrationBean filterRegistrationBean(){ FilterRegistrationBean filterRegistrationBean = new FilterRegistrationBean(); filterRegistrationBean.setFilter(new HttpFilter()); //设置自定义过滤器拦截的url filterRegistrationBean.addUrlPatterns("/threadLocal/*"); return filterRegistrationBean; } /** * 定义自定义拦截器原先需要继承WebMvcConfigurerAdapter * SpringBoot2.0后WebMvcConfigurerAdapter被定义成过时了,推荐使用继承WebMvcConfigurationSupport * @param registry */ @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(new HttpInterceptor()).addPathPatterns("/**"); }}
定义调用接口
/** * ThreadLocal测试controller * @author zjq */@Controller@RequestMapping("/threadLocal")public class ThreadLocalController { @RequestMapping("/test") @ResponseBody public Long test() { return RequestHolder.get(); }}
请求访问验证
访问调用接口,控制台输出如下:
边栏推荐
- Chapter 5: Activiti process shunting judgment, judging to go to different task nodes
- 数据可视化(一)
- 第四章:activiti RuntimeService设置获和取流程变量,及与taskService的区别,开始和完成任务时设置流程变量[通俗易懂]
- E-sports, convenience, efficiency, security, key words for OriginOS functions
- 【C语言指针】用指针提升数组的运算效率
- 导火索:OAuth 2.0四种授权登录方式必读
- Microcontroller: temperature control DS18B20
- Common operations of oracle under linux and daily accumulation of knowledge points (functions, timed tasks)
- 第七章,activiti个人任务分配,动态指定和监听器指定任务委派人「建议收藏」
- SQL外连接之交集、并集、差集查询
猜你喜欢
电气工程的标准是什么
Dynamics 365Online PDF导出及打印
今天告诉你界面控件DevExpress WinForms为何弃用经典视觉样式
【加密解密】明文加密解密-已实现【已应用】
sqlserver编写通用脚本实现获取一年前日期的方法
使用Windbg过程中两个使用细节分享
单片机:温度控制DS18B20
Login function and logout function (St. Regis Takeaway)
PCB layout must know: teach you to correctly lay out the circuit board of the op amp
用KUSTO查询语句(KQL)在Azure Data Explorer Database上查询LOG实战
随机推荐
OpenHarmony如何查询设备类型
[Translation] Chaos Net + SkyWalking: Better observability for chaos engineering
产品太多了,如何实现一次登录多产品互通?
用KUSTO查询语句(KQL)在Azure Data Explorer Database上查询LOG实战
RT - Thread record (a, RT, RT Thread version - Thread Studio development environment and cooperate CubeMX quick-and-dirty)
2022 Huashu Cup Mathematical Modeling Question A Optimization Design Ideas for Ring Oscillators Code Sharing
解决【命令行/终端】颜色输出问题
Opencv图像缩放和平移
例题 可达性统计+bitset的使用
2022杭电多校 第6场 1008.Shinobu Loves Segment Tree 规律题
2022 Hangzhou Electric Power Multi-School Session 6 1008.Shinobu Loves Segment Tree Regular Questions
How can project cost control help project success?
字节一面:TCP 和 UDP 可以使用同一个端口吗?
The host computer develops C# language: simulates the STC serial port assistant to receive the data sent by the microcontroller
2022 Huashu Cup Mathematical Modeling Ideas Analysis and Exchange
The fuse: OAuth 2.0 four authorized login methods must read
[Android]如何使用RecycleView in Kotlin project
【C语言指针】用指针提升数组的运算效率
【MindSpore Easy-Diantong Robot-01】You may have seen many knowledge quiz robots, but this one is a bit different
What are the standards for electrical engineering