ThreadLocal是什么
ThreadLocal类提供线程内部的局部变量,在多线程环境(即多线程并发场景)访问时能够保证变量相对独立于其他线程内的变量(即线程隔离),用于关联线程和线程上下文(意味着可以通过ThreadLocal在同一线程下,不同的组件里传递共用变量)。
@Data
public class ThreadDemo2 {
private ThreadLocal<String> tl = new ThreadLocal<>();
public static void main(String[] args) {
ThreadDemo2 demo = new ThreadDemo2();
for (int i = 0; i < 5; i++) {
Thread thread = new Thread(() -> {
demo.tl.set(Thread.currentThread().getName() + "的数据");
System.out.println("------------------------------");
System.out.println(Thread.currentThread().getName() + "---->" + demo.getTl().get());
});
thread.setName("线程" + i);
thread.start();
}
}
}
ThreadLocal应用场景
主要抓住ThreadLocal的特性:多线程并发、线程隔离。举一个场景:当一个调用链很长,其中某个链需要使用到某种信息,你可以通过方法参数的方式进行传递。但是,如果某个方法不允许修改入参,这种情况是不是就可以在调用方法前给信息塞到ThreadLocal里,然后其他地方get。
Spring使用ThreadLocal举例
Spring使用ThreadLocal去保证单个线程中的数据库操作使用的是同一个数据库连接,TransactionSynchronizationManager内部使用ThreadLocal对象存储资源,存储的是一个Map,这个Map的Key是DataSource生成的actualKey,Value是ConntcionHolder。
在第一次获取连接的时候,把连接绑定到ThreadLocal,等这个线程下次访问直接拿出来用。在事务结束的时候,调用releaseConnection()把这个ConnectionHolder移除。没有人用的时候,直接close掉这个连接。
相关源码:
JdbcTemplate通过DatasourceUtils.getConnection获取数据库连接:
// JdbcTemplate,通过DatasourceUtils.getConnection获取数据库连接
@Nullable
public <T> T execute(ConnectionCallback<T> action) throws DataAccessException {
Assert.notNull(action, "Callback object must not be null");
Connection con = DataSourceUtils.getConnection(this.obtainDataSource());
// ....
finally {
// 释放资源
DataSourceUtils.releaseConnection(con, getDataSource());
}
}
getConnection()内部调用TransactionSynchronizationManager.getResource()去获取:
public static Connection getConnection(DataSource dataSource) throws CannotGetJdbcConnectionException {
try {
return doGetConnection(dataSource);
} catch (SQLException var2) {
throw new CannotGetJdbcConnectionException("Failed to obtain JDBC Connection", var2)

ThreadLocal为线程提供隔离的局部变量,常用于线程间数据隔离,如Spring的数据库连接管理。其底层通过ThreadLocalMap实现,使用弱引用避免内存泄漏,但需注意手动清理避免内存问题。InheritableThreadLocal支持子线程共享数据。
1926

被折叠的 条评论
为什么被折叠?



