ThreadLocal的使用

ThreadLocal 的使用

首先我们来说一下,ThreadLocal的作用,它可以解决多线程的数据安全问题

ThreadLocal的使用非常简单,它可以为当前线程存储一个数据(注意:只是一个数据,如果是多个数据,那么需要多个ThreadLocal来进行实例化),存储的数据只有指定的线程可以得到。

这边是官方文档的解释

这个类提供线程局部变量。 这些变量与其正常的对应方式不同,因为访问一个的每个线程(通过其get或set方法)都有自己独立初始化的变量副本。 ThreadLocal实例通常是希望将状态与线程关联的类中的私有静态字段(例如,用户ID或事务ID)。 

大致意思就是,ThreadLocal类似于一个Map集合,Key就是ThreadLocal本身,value就是数据,通过get()取出数据,通过set()来存储数据。

这边我们用map模仿一下ThreadLocal的作用


public class ThreadLocalTest {
    //定义一个Map模仿ThreadLocal的作用  Hashtable是线程安全的
    public final static Map<String, Object> map = new Hashtable<>();

    //定义一个随机数用于存放数据
    public static Random random = new Random();

    //定义一个线程内部类
    public static class temp implements Runnable{
        @Override
        public void run() {
            //获取当前线程名字
            String name = Thread.currentThread().getName();
            //生成随机数
            int i = random.nextInt(1000);
            //输出一下当前线程生成的随机数
            System.out.println("线程["+name+"]生成的随机数是"+i);
            //存储当前线程的数据
            map.put(name,i);

            try {
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("在线程["+name+"]结束时取出的随机数是"+map.get(name));

        }
    }

测试一下

        //这边测试一下
        public static void main(String[] args) {
            //生成三个线程
            for (int i = 0; i < 3; i++) {
                //调用该线程
                new Thread(new temp()).start();
            }
        }

来看一下输出的结果

线程[Thread-0]生成的随机数是417
线程[Thread-2]生成的随机数是339
线程[Thread-1]生成的随机数是380
在线程[Thread-2]结束时取出的随机数是339
在线程[Thread-0]结束时取出的随机数是417
在线程[Thread-1]结束时取出的随机数是380

我们改用ThreadLocal来用一下


    //定义一个ThreadLocal
    public static ThreadLocal threadLocal = new ThreadLocal();


    public void run() {
            //获取当前线程名字
            String name = Thread.currentThread().getName();
            //生成随机数
            int i = random.nextInt(1000);
            //输出一下当前线程生成的随机数
            System.out.println("线程["+name+"]生成的随机数是"+i);
            //存储当前线程的数据
            //map.put(name,i);
            //改用threadLocal
            threadLocal.set(i);

            try {
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            //输出改用threadLocal
            System.out.println("在线程["+name+"]结束时取出的随机数是"+threadLocal.get());

        }

输出结果

线程[Thread-0]生成的随机数是890
线程[Thread-1]生成的随机数是778
线程[Thread-2]生成的随机数是657
在线程[Thread-2]结束时取出的随机数是657
在线程[Thread-1]结束时取出的随机数是778
在线程[Thread-0]结束时取出的随机数是890

不难发现,set()、get()方法就是吧线程本身作为key来储存值

试一下单个实例化存储多个数据

在这里插入图片描述
来看一下输出结果
在这里插入图片描述

会发现后来的值会将原来赋的值覆盖掉,要想储存多个值,必须将ThreadLocal多个实例化

这时候想一下ThreadLocal的作用是解决多线程数据安全问题,是否与synchronize一样。分析一下ThreadLocal的特性

ThreadLocal和Synchronized都是为了解决多线程中相同变量的访问冲突问题,不同的点是

  • Synchronized是通过线程等待,牺牲时间来解决访问冲突
  • ThreadLocal是通过每个线程单独一份存储空间,牺牲空间来解决冲突,并且相比于Synchronized,ThreadLocal具有线程隔离的效果,只有在线程内才能获取到对应的值,线程外则不能访问到想要的值。

正因为ThreadLocal的线程隔离特性,使他的应用场景相对来说更为特殊一些。在android中Looper、ActivityThread以及AMS中都用到了ThreadLocal。当某些数据是以线程为作用域并且不同线程具有不同的数据副本的时候,就可以考虑采用ThreadLocal

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ThreadLocal是一个Java类,用于在多线程环境下保存线程本地变量的副本。通过创建ThreadLocal对象,每个线程都可以拥有自己独立的变量副本,互不干扰。 使用ThreadLocal的过程如下: 1. 创建一个ThreadLocal对象,可以指定泛型类型。 2. 在每个线程中,通过ThreadLocal对象的`get()`方法获取当前线程的变量副本。如果当前线程没有设置过变量值,则会使用默认值进行初始化。 3. 在每个线程中,通过ThreadLocal对象的`set(value)`方法设置当前线程的变量值。 4. 在每个线程中,通过ThreadLocal对象的`remove()`方法移除当前线程的变量副本。 需要注意的是,ThreadLocal对象的生命周期与Thread对象的生命周期一样长。当ThreadLocal对象被垃圾回收时,关联的变量副本也会被回收。 在ThreadLocal内部,使用ThreadLocalMap来存储每个线程的变量副本。ThreadLocal的实例作为key,变量值作为value。ThreadLocalMap可以使用强引用或弱引用来引用ThreadLocal对象。如果使用强引用,当ThreadLocal对象被回收时,如果没有手动删除对应的变量副本,会导致内存泄漏。如果使用弱引用,当ThreadLocal对象被回收时,对应的变量副本也会被回收。 总结来说,ThreadLocal是一种线程本地变量,通过保存每个线程的变量副本,实现了多线程环境下的线程隔离。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [ThreadLocal使用与原理](https://blog.csdn.net/qq_35190492/article/details/116431270)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [ThreadLocal使用详解](https://blog.csdn.net/LJJZJ/article/details/88763666)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值