关于ThreadLocal

1.什么是ThreadLocal:

ThreadLocal类用来提供线程内部的局部变量。这种变量在多线程环境下访问(通过get和set方法访问)时能保证各个线程的变量相对独立于其他线程内的变量。ThreadLocal实例通常来说都是private static类型的,用于关联线程和线程上下文。

即ThreadLocal有三个特点:

①线程并发: 在多线程并发的场景下,不会出现在单线程中

②传递数据: 我们可以通过ThreadLocal在同一线程,不同组件中传递公共变量

③线程隔离: 每个线程的变量都是独立的,不会互相影响

2.ThreadLocal的常见的方法

ThreadLocal():创建ThreadLocal对象

public void set( T value):设置当前线程绑定的局部变量

public void get():获取当前线程的局部变量

public void remove():移除当前线程的局部变量

当多线程运行时,他们时并发的,所以会引起线程获取数据混乱的现象,如下:

此时是for循环5次,每次创建一个线程,设置线程名为循环的次数,运行线程并获取在run方法中设置好的数据。可以发现他们的数据混乱了。

而使用ThreadLocal时,可以绑定当前线程与变量

此时数据就与当前线程所绑定。

3.ThreadLocal和synchronized的区别:

ThreadLocal采用’以空间换时间’的方式, 为每一个线程都提供了一份变量的副本,从而实现同时访问而相不干扰。而synchronized同步机制采用’以时间换空间’的方式, 只提供了一份变量,让不同的线程排队访问。ThreadLocal比synchronized的性能更高,synchronized每次只能一个线程执行,降低了效率。

4.ThreadLocal的内部结构:

在JDK8中 ThreadLocal的设计是:每个Thread维护一个ThreadLocal,这个Map的keyThreadLocal实例本身,value才是真正要存储的值Object。这样的话当Thread销毁之后,对应的ThreadLocalMap也会随之销毁,能减少内存的使用。

5.ThreadLocal的内存泄漏:

        在使用ThreadLocal的过程中会发现有内存泄漏的情况发生,由于ThreadLocalMap的生命周期跟Thread一样长,如果没有手动删除对应key就会导致内存泄漏。这主要有两个原因

1.在使用ThreadLocal,没有手动删除使用完的Entry。此时,只要在使用完ThreadLocal,调用其remove方法删除对应的Entry,就能避免内存泄漏。

2.CurrentThread依然运行。而在使用完ThreadLocal之后,如果当前Thread也随之执行结束,ThreadLocalMap自然也会被gc回收,从根源上避免了内存泄漏。

  • 1
    点赞
  • 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、付费专栏及课程。

余额充值