关于ThreadLocal的那点事

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

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"
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ThreadLocalJava中一个非常重要的线程封闭技术。它可以让每个线程都拥有自己的变量副本,避免了线程间的竞争和数据泄露问题。在本文中,我们将详细介绍ThreadLocal的定义、用法及其优点。 1. ThreadLocal的定义 ThreadLocalJava中一个用来实现线程封闭技术的类。它提供了一个本地线程变量,可以在多线程环境下使每个线程都拥有自己的变量副本。每个线程都可以独立地改变自己的副本,而不会影响到其他线程的副本。ThreadLocal的实现是基于ThreadLocalMap的,每个ThreadLocal对象都对应一个ThreadLocalMap,其中存储了线程本地变量的值。 2. ThreadLocal的用法 使用ThreadLocal非常简单,只需要创建一个ThreadLocal对象,然后调用其get()和set()方法即可。get()方法用来获取当前线程的变量副本,如果当前线程还没有变量副本,则会创建一个新的副本并返回。set()方法用来设置当前线程的变量副本,如果当前线程已经有了变量副本,则会覆盖原来的副本。 下面是一个简单的例子,演示了如何使用ThreadLocal来实现线程封闭: ```java public class ThreadLocalTest { private static ThreadLocal<String> threadLocal = new ThreadLocal<>(); public static void main(String[] args) throws InterruptedException { new Thread(() -> { threadLocal.set("Thread A"); System.out.println("Thread A: " + threadLocal.get()); }).start(); new Thread(() -> { threadLocal.set("Thread B"); System.out.println("Thread B: " + threadLocal.get()); }).start(); Thread.sleep(1000); System.out.println("Main: " + threadLocal.get()); } } ``` 运行结果如下: ``` Thread A: Thread A Thread B: Thread B Main: null ``` 从输出结果可以看出,每个线程都拥有自己的变量副本,互不影响。而在主线程中,由于没有设置过变量副本,所以返回null。 3. ThreadLocal的优点 ThreadLocal的优点主要体现在以下几个方面: (1)线程安全:ThreadLocal可以避免线程间的竞争和数据泄露问题,每个线程都可以独立地修改自己的变量副本,不会影响其他线程。 (2)高效性:ThreadLocal使用起来非常简单,而且性能也非常高,比如在Web开发中,可以将用户信息存储在ThreadLocal中,从而避免在每个方法中都去查询数据库。 (3)易用性:ThreadLocal的使用非常灵活,可以根据实际需要自由地定义数据类型和访问方式。 总的来说,ThreadLocalJava中一个非常重要的线程封闭技术,可以帮助开发人员避免线程间的竞争和数据泄露问题,提高程序的安全性和性能。在实际开发中,我们应该充分利用ThreadLocal的优点,合理地运用它来解决各种线程安全问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值