关于ThreadLocal的浅谈

本文通过具体示例,深入探讨了ThreadLocal在多线程环境下的应用与特性,包括线程隔离、对象复用及内存回收等关键知识点。

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

以前用ThreadLocal用的比较少,今天在看以前的代码的时候,发现有个类里面的方法里面使用了公共的静态变量,而且有很多类都引用了这个方法。由于这个变量不是final的,是动态赋值的,所以在多线程的情况下可能会引起问题。因为要改造这个方法的这个变量问题,所以我想到了ThreadLocal,并对ThreadLocal进行了一次实验,通过实验,对ThreadLocal有了一次比较深刻的认识。在此把代码贴出来,跟各位共同学习一下。

public class Test {
    public static ThreadLocal<String> threadLocal = new ThreadLocal<String>();
    private static final long waitTime = 3000L;
    public static void main(String[] args) throws Exception {
        Thread.sleep(60000);
        Thread thread1 = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());
                threadLocal.set(Thread.currentThread().getName()+"----111111111");
                try {
                    Thread.sleep(waitTime);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                User user = new User("张三");
                user.say();
            }
        });

        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());
                threadLocal.set(Thread.currentThread().getName()+"----22222222");
                try {
                    Thread.sleep(waitTime);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                User user = new User("李四");
                user.say();
            }
        });

        Thread thread3 = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());
                threadLocal.set(Thread.currentThread().getName()+"----3333333333");
                try {
                    Thread.sleep(waitTime);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                User user1 = new User("王五");
                user1.say();
                User user2 = new User("赵六");
                user2.say();
            }
        });

        thread1.start();
        thread2.start();
        thread3.start();


        Thread thread4 = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());
                threadLocal.set(Thread.currentThread().getName()+"----4444444444");
                try {
                    Thread.sleep(waitTime);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                new Thread(new Runnable() {
                    @Override
                    public void run() {
                        User user1 = new User("宋八1");
                        user1.say();
                        threadLocal.set(Thread.currentThread().getName()+"----5555555555");
                        User user2 = new User("宋八2");
                        user2.say();
                    }
                }).start();
                new Thread(new Runnable() {
                    @Override
                    public void run() {
                        User user1 = new User("钱九1");
                        user1.say();
                        threadLocal.set(Thread.currentThread().getName()+"----666666666666666");
                        User user2 = new User("钱九2");
                        user2.say();
                    }
                }).start();
            }
        });
        thread4.start();

        Thread.sleep(60000);
    }

}

public class User {
    private String name;

    public User(){
        super();
    }
    public User(String name){
        this.name = name;
    }
    public void say(){
            System.out.println(name + " say:" + Test.threadLocal.get());
    }

}

结果输出如下:

Thread-2
Thread-3
Thread-4
Thread-5
张三 say:Thread-2----111111111
李四 say:Thread-3----22222222
王五 say:Thread-4----3333333333
赵六 say:Thread-4----3333333333
钱九1 say:null
钱九2 say:Thread-7----666666666666666
宋八1 say:null
宋八2 say:Thread-6----5555555555

 

做一下说明:

我创建了四个主线程,每个线程内部对ThreadLocal变量进行赋值,并在主线程内创建User对象,User对象会输出ThreadLocal的值。通过结果可以看出:

1、各个线程内部的ThreadLocal是不相互影响的,并且主线程与子线程是不相互影响的。 

2、同一个线程内容,不同对象获取的同一个ThreadLocal对象是一样的。

3、子线程是无法获取到主线程的ThreadLocal值的

通过以上试验,我还进行一下内存的监测,发现一个线程调用结束之后,会进行对象的回收。如图:

 

通过

 

以上的试验,希望对大家理解和使用ThreadLocal能有所帮助。

 

 

修改ThreadLocal的值可以通过以下步骤实现: 1. 首先,创建一个ThreadLocal对象,并将其初始化为需要修改的值。可以使用ThreadLocal的`set()`方法来设置值。例如,`threadLocal.set(value)`,其中value是要设置的新值。 2. 接下来,可以在需要修改ThreadLocal值的线程中使用上述设置的ThreadLocal对象。可以通过`threadLocal.get()`方法来获取该线程的ThreadLocal值。 3. 然后,对获取的ThreadLocal值进行修改,可以使用任何适当的方法来更改值。 4. 最后,使用ThreadLocal的`set()`方法将修改后的值设置回ThreadLocal对象中。例如,`threadLocal.set(modifiedValue)`,其中modifiedValue是修改后的值。 需要注意的是,每个线程都有自己独立的ThreadLocal副本,因此在不同的线程中修改ThreadLocal值不会相互影响。这是ThreadLocal的特性之一,使得每个线程可以独立地管理自己的数据。 总结起来,修改ThreadLocal的值需要创建ThreadLocal对象、设置新值、获取当前线程的ThreadLocal值、修改该值,最后将修改后的值设置回ThreadLocal对象中。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [浅谈ThreadLocal](https://blog.youkuaiyun.com/qq_36609994/article/details/121163994)[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%"] - *3* [ThreadLocal](https://blog.youkuaiyun.com/weixin_43573186/article/details/125068926)[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、付费专栏及课程。

余额充值