SpringBoot2.x结合ThreadLocal自定义简单的RequestHolder实现线程封闭

本文介绍如何在SpringBoot2.x中使用过滤器、拦截器和ThreadLocal实现线程封闭,确保线程间数据隔离。通过自定义RequestHolder类、过滤器和拦截器,结合配置类,演示了具体实现步骤及效果验证。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

开篇:ThreadLocal就不多说了,想必这个大家都很熟悉了,实现单线程共享和安全的有效途径,下面是我在GitHub上的一篇浅析ThreadLocal!

ThreadLocal的简单剖析

目标:通过SpringBoot2.x的过滤器,拦截器和ThreadLocal实现简单的线程封闭。并且促进理解RequestContextHolder(持有上下文的Request容器,也是一个很好用的工具类,它的实现关键也是依赖于两个ThreadLocal)。

图解本文进行测试案例

在这里插入图片描述

实现步骤

根据实现流程,写实现步骤

1.自定义一个RequestHolder

使用ThreadLocal自定义一个简单类型的RequestHolder,并增加增删查的方法

package com.wrial.main.example.threadLocal;

public class RequestHolder {

    private final static ThreadLocal<Long> requestHolder = new ThreadLocal<>();

    public static void add(Long id){
        requestHolder.set(id);
    }
    public static void remove(){
        requestHolder.remove();
    }
    public static Long get(){
        return requestHolder.get();
    }

}

2.自定义一个过滤器(Filter)

在SpringBoot中定义一个过滤器,其实SpringBoot的思想就是手动加自动配置,理解其思想就很容易写出自定义代码。需要自定义类继承Filter并且在配置类中进行配置。

package com.wrial.main;

import com.wrial.main.example.threadLocal.RequestHolder;
import lombok.extern.slf4j.Slf4j;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;

@Slf4j
public class HttpFilter implements Filter {


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

         //打印路径和线程id日志
        log.info("do filter,{},{}",servlet.getServletPath(),Thread.currentThread().getId());

        //将线程Id放在自定义的RequestHolder
        RequestHolder.add(Thread.currentThread().getId());
       
        filterChain.doFilter(servletRequest,servletResponse);
    }

}

配置类中进行配置(FilterRegistrationBean)

  @Bean
    public FilterRegistrationBean httpFilter(){
        FilterRegistrationBean registrationBean = new FilterRegistrationBean();
        registrationBean.setFilter(new HttpFilter());
        registrationBean.addUrlPatterns("/threadLocal/*");
        return registrationBean;
    }

这样我们自定义的简单过滤器就ok了。

3.自定义一个拦截器(Interceptor)

编写一个拦截器继承HandlerInterceptorAdaptor

package com.wrial.main;

import com.wrial.main.example.threadLocal.RequestHolder;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

@Slf4j
public class HttpInterceptor extends HandlerInterceptorAdapter {

    //请求前
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        log.info("preHandle");
        return true;
    }

    //请求后,postHandle是正常处理
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        //移除RequestHolder中的线程Id
        RequestHolder.remove();

        log.info("afterCompletion");
        return;
    }
}

配置类,使用WebMvcConfigurer来添加我们自定义的拦截器

 @Bean
    public WebMvcConfigurer webMvcConfigurer() {
        WebMvcConfigurer webMvcConfigurer = new WebMvcConfigurer() {
            @Override
            public void addInterceptors(InterceptorRegistry registry) {
                //添加自定义拦截器,并模式匹配所以路径
                registry.addInterceptor(new HttpInterceptor()).addPathPatterns("/**");
            }
        };
        return webMvcConfigurer;
    }

在以前的SpringBoot中是用WebMvcConfigurerAdaptor,SpringBoot2.x不推荐使用。当然,那样配置也是可以生效的。

4.编写ThreadLocalController
package com.wrial.main.example.threadLocal;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

@Controller
//此请求映射路径在拦截器范围内
@RequestMapping("/threadLocal")
public class ThreadLocalController {

    @RequestMapping("test")
    @ResponseBody
    public Long test(){

        return RequestHolder.get();
    }
}

配置方法2:

5.PostMan接口测试

访问:http://localhost:8080/threadLocal/test
在这里插入图片描述
后台输出结果
在这里插入图片描述
可以看到我们定义的拦截器和过滤器已经在接口访问过程中起作用了。

在Spring Boot中创建AOP,可以使用Spring框架提供的@Aspect注解和@Around注解来实现。下面是一个使用ThreadLocal自定义连接、关闭数据库操作的示例: 1. 创建一个ThreadLocal对象来存储数据库连接: ``` public class DBContextHolder { private static final ThreadLocal<Connection> contextHolder = new ThreadLocal<>(); public static void setConnection(Connection connection) { contextHolder.set(connection); } public static Connection getConnection() { return contextHolder.get(); } public static void clear() { contextHolder.remove(); } } ``` 2. 创建一个切面类来拦截所有带有@DBConnection注解的方法: ``` @Aspect @Component public class DBConnectionAspect { @Around("@annotation(com.example.demo.annotation.DBConnection)") public Object around(ProceedingJoinPoint point) throws Throwable { Connection connection = null; try { connection = getConnection(); DBContextHolder.setConnection(connection); return point.proceed(); } finally { if (connection != null) { connection.close(); DBContextHolder.clear(); } } } private Connection getConnection() { // 获取数据库连接 return null; } } ``` 3. 在需要使用数据库连接的方法上添加@DBConnection注解: ``` @Service public class UserService { @DBConnection public void save(User user) { // ... } } ``` 在调用UserService的save方法时,AOP切面会拦截该方法并自动获取数据库连接,并将连接存储到ThreadLocal中,以便在同一线程中的其他方法中使用。当方法执行完毕后,AOP切面会自动关闭数据库连接,并清空ThreadLocal中的连接。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值