使用ThreadLocal在线程间传递参数

本文介绍了如何利用ThreadLocal和AOP在Spring环境中优雅地处理租户ID的传递问题。通过在请求过滤器中设置租户ID到ThreadLocal,然后在需要的地方直接获取,避免了参数传递的冗余,提高了代码的可读性和可维护性。同时,使用阿里巴巴的TransmittableThreadLocal库确保了线程间的数据安全性。

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

使用ThreadLocal在线程间传递参数

使用场景

用户参数透传,如租户ID,tid,userID,Token等。这些参数与具体业务无关,但是又是必须的(如:租户ID,在数据落库时需要传入,但是在其他业务代码中基本用不到)如果作为方法的参数一层一层向下传递,必然造成代码的冗余和扩展性查等影响。

整片文章就以租户ID为例子说明吧。

1.使用ThreadLocal结合AOP

大致思路:在请求接口时,租户ID往往可以从Token或用户信息中取出,这时,在请求拦截器中,将用户TenantID取出,放入线程中。后面在需要TenantID时,直接从线程中取出,这样,既保证了简洁性也增加了易用性。

2.Pom依赖

本文使用阿里线程工具。

<!--TTL-->
<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>transmittable-thread-local</artifactId>
    <version>2.11.4</version>
</dependency>

3.RequestFilter

import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.springframework.core.Ordered;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
import org.springframework.web.filter.GenericFilterBean;

import javax.servlet.FilterChain;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * @author litongzero
 */
@Slf4j
@Component
@Order(Ordered.HIGHEST_PRECEDENCE)
public class TenantContextHolderFilter extends GenericFilterBean {

	@Override
	@SneakyThrows
	public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) {
		HttpServletRequest request = (HttpServletRequest) servletRequest;
		HttpServletResponse response = (HttpServletResponse) servletResponse;
    // 从Token/Header/param中获取租户ID
		String headerTenantId = request.getHeader("TENANT_ID");

		log.debug("TenantContextHolderFilter|header中的TENANT_ID为:{}", headerTenantId);

    // 判断逻辑,自行定义
		if (headerTenantId != null && headerTenantId != "" && !headerTenantId.equals("null")) {
			TenantContextHolder.setTenantId(Integer.parseInt(headerTenantId));
    } else {
      // 默认值
			TenantContextHolder.setTenantId(0);
		}
		filterChain.doFilter(request, response);
    // 请求结束,一定要清除线程中自己设置的参数。
		TenantContextHolder.clear();
	}

}

4.TenantContextHolder(线程参数设置工具)

/**
 * @author litongzero
 */
@UtilityClass
public class TenantContextHolder {

	private final ThreadLocal<Integer> THREAD_LOCAL_TENANT = new TransmittableThreadLocal<>();

	/**
	 * TTL 设置租户ID<br/>
	 * @param tenantId
	 */
	public void setTenantId(Integer tenantId) {
		THREAD_LOCAL_TENANT.set(tenantId);
	}

	/**
	 * 获取TTL中的租户ID
	 * @return
	 */
	public Integer getTenantId() {
		return THREAD_LOCAL_TENANT.get();
	}

	/**
	 * 清除当前线程中的租户
	 * 慎用
	 */
	public void clear() {
		THREAD_LOCAL_TENANT.remove();
	}

}

5.使用

在具体的Controller,Service,Mapper中,只要是当前请求的线程,都是可以直接使用TenantContextHolder.getTenantId()获取租户ID的。

### Java ThreadLocal 参数传递方法及示例 #### 使用 `InheritableThreadLocal` 实现参数继承 为了使父线程中的 `ThreadLocal` 变量能够被子线程访问,可以使用 `InheritableThreadLocal` 类。这允许子线程继承父线程的 `ThreadLocal` 值。 ```java public class InheritableThreadLocalExample { private static final InheritableThreadLocal<String> inheritableThreadLocal = new InheritableThreadLocal<>(); public static void main(String[] args) throws InterruptedException { inheritableThreadLocal.set("Main Value"); Thread childThread = new Thread(() -> { System.out.println("Child thread value: " + inheritableThreadLocal.get()); }); childThread.start(); childThread.join(); System.out.println("Parent thread value: " + inheritableThreadLocal.get()); } } ``` 这段代码展示了如何设置并让子线程继承来自父线程的 `ThreadLocal` 值[^4]。 #### 利用第三方库增强功能 对于更复杂的场景,特别是当需要跨多个层次的线程传递数据时,推荐使用阿里巴巴开源项目 `transmittable-thread-local` 提供的支持。此库解决了标准 `ThreadLocal` 和 `InheritableThreadLocal` 存在的一些局限性,特别是在异步编程环境中[^5]。 要集成这个库到 Maven 项目中,请添加如下依赖: ```xml <dependency> <groupId>com.alibaba</groupId> <artifactId>transmittable-thread-local</artifactId> <version>2.14.3</version> </dependency> ``` 之后可以通过创建自定义的 `TransmittableThreadLocal` 或者直接利用其提供的 API 进行操作。 #### 清理资源的重要性 无论采用哪种方式来管理 `ThreadLocal` 数据,在不再需要这些值的时候应当调用 `remove()` 方法清除它们,防止潜在的记忆泄漏风险。例如: ```java threadLocal.remove(); ``` 这样做有助于确保每个线程在其生命周期结束前释放关联的对象引用,从而避免不必要的内存占用[^1]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值