ThreadLocal使用及应用场景分析

1. ThreadLocal 定义

  • 定义

    提供线程局部变量;一个线程线程局部变量在多个线程中,分别有独立的值(副本)。

  • 特点

    简单(开箱即用)、快速(无额外开销)、安全(线程安全)。

  • 场景

    多线程场景(资源持有、线程一致性、并发计算、线程安全等场景)。

  • 实现原理

    Java中用哈希表实现。

    image-20220208093618274

  • 应用范围

    几乎所有提供多线程特征的语言。

2. ThreadLocal 基本API

image-20220208093856115

public class ThreadLocalTest {
     public static ThreadLocal<Long> x = new ThreadLocal<Long>(){
         @Override
         protected Long initialValue() {
             System.out.println(Thread.currentThread().getId() + " Initial value run...");
             return Thread.currentThread().getId();
         }
     };

    public static void main(String[] args) {
        new Thread(()->{
            System.out.println(x.get());
        }).start();
        x.set(100L);
        //x.remove();
        System.out.println(x.get());
    }

}

3. ThreadLocal 的4种关键场景

  1. 线程资源持有

    持有线程资源供线程的各个部分使用,全局获取,减少编程难度。

    image-20220208101039021

  2. 线程资源一致性

    帮助需要保持线程一致的资源(如数据库事务)维护一致性,降低编程难度。

    image-20220208101237633

  3. 线程安全

    帮助只考虑了单线程的程序库,无缝向多线程场景迁移。

    image-20220208101328539

  4. 分布式计算

    帮助分布式计算场景的各个线程累积局部计算结果。

    image-20220208101428551

4. ThreadLocal 并发场景分析

  • Apache ab 压力并发测试工具的使用

    可参考:https://www.cnblogs.com/wtcl/p/6795740.html

  • 用synchronize实现线程安全的缺点

    性能得不到满足。

  • 用ThreadLocal实现并发线程安全

    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    @RestController
    public class StatController {
        //static Integer c =0;
        static ThreadLocal<Integer> c = new ThreadLocal<Integer>(){
            @Override
            protected Integer initialValue(){
              return 0;
            }
        };
    
        void _add(){
            try {
                Thread.sleep(100);
                c.set(c.get() + 1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    
        @RequestMapping("/stat")
        public Integer stat(){
            return c.get();
        }
    
        @RequestMapping("/add")
        public Integer add(){
            _add();
            return 1;
        }
    }
    
    

    上面的代码是有问题的,因为获取的值是每个线程自己的变量值,每次累加的也是每个线程自己的累加值。

    正确的写法应该是:

    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    import java.util.HashSet;
    
    @RestController
    public class StatController {
        static HashSet<Val<Integer>> set = new HashSet<Val<Integer>>();
        synchronized  static void addSet(Val<Integer> v){
            set.add(v);
        }
        static ThreadLocal<Val<Integer>> c = new ThreadLocal<Val<Integer>>(){
            @Override
            protected Val<Integer> initialValue(){
                Val<Integer> v = new Val<>();
                v.setV(0);
                addSet(v);
                return v;
            }
        };
    
        void _add(){
            try {
                Thread.sleep(100);
                Val<Integer> v = c.get();
                v.setV(v.getV() + 1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    
        @RequestMapping("/stat")
        public Integer stat(){
            return set.stream().map(x-> x.getV()).reduce((a,x)-> a+x).get();
        }
    
        @RequestMapping("/add")
        public Integer add(){
            _add();
            return 1;
        }
    }
    
    
    public class Val<T> {
        T v;
    
        public T getV() {
            return v;
        }
    
        public void setV(T v) {
            this.v = v;
        }
    }
    
  • 15
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
ThreadLocalJava 中的一个线程局部变量,它为每个线程提供了独立的变量副本,每个线程都可以通过 ThreadLocal 对象来访问自己的变量副本,而不会影响其他线程的副本。ThreadLocal 在多线程编程中非常有用,可以解决线程安全问题。 使用 ThreadLocal 的主要场景包括: 1. 线程上下文信息传递:有些情况下,我们需要在多个方法之间传递一些上下文信息,例如用户认证信息、数据库连接等。使用 ThreadLocal 可以避免在方法参数中传递这些上下文信息,每个线程都可以独立地访问自己的上下文信息。 2. 线程安全的对象:有些对象在多线程环境下是不安全的,如果每个线程都持有一个对象的副本,就可以避免多线程竞争访问导致的安全问题。例如 SimpleDateFormat 是非线程安全的,可以使用 ThreadLocal 来为每个线程提供一个独立的 SimpleDateFormat 对象。 3. 隐式参数传递:有些方法需要依赖某些参数,但是这些参数对于调用方来说并不是必须的。使用 ThreadLocal 可以将这些参数设置为 ThreadLocal 变量,在方法中直接获取这些参数,而不需要显式传递。 需要注意的是,使用 ThreadLocal 时要及时清理资源,避免内存泄漏。在使用完毕后,应该调用 ThreadLocal 的 remove() 方法来清理当前线程持有的变量副本。 总之,ThreadLocal 可以用于在多线程环境下实现线程安全、线程间数据隔离和传递上下文信息等功能。但是过度使用 ThreadLocal 也会导致代码可读性变差,增加了代码的复杂性,需要根据具体场景进行合理使用

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小灰灰-58

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值