ThreadLocal和synchronized关键字

ThreadLocal是Java中用于在多线程环境下实现线程局部变量的工具类,它能保证每个线程都拥有独立的变量副本,避免数据冲突。通过set和get方法,线程可以存储和获取自己的局部变量,而在不同线程间,这些变量互不影响。相比synchronized的同步机制,ThreadLocal通过为每个线程提供独立副本的方式提高了并发性能。
摘要由CSDN通过智能技术生成

ThreadLocal基本介绍:

  • 线程并发:在多线程并发场景下
  • 传递数据:我们可以通过ThreadLocal在同一线程、不同组件中传递公共变量
  • 线程隔离:每个线程的变量都是独立的,不会互相影响

常用方法

方法声明描述
ThreadLocal()创建ThreadLocal对象
public void set(T value)设置 当前线程绑定的局部变量
public T get()获取 当前线程绑定的局部变量
public void remove()删除 当前线程绑定的局部变量

代码例子:

public class MyDemo01 {
    private String content;

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }

    public static void main(String[] args) {
        MyDemo01 demo = new MyDemo01();
        for (int i = 0; i < 5; i++) {
            Thread thread = new Thread(new Runnable() {
                @Override
                public void run() {
                    demo.setContent(Thread.currentThread().getName() + "的数据");
                    System.out.println("-------------------");
                    System.out.println(Thread.currentThread().getName() + "--->" + demo.getContent());
                }
            });
            thread.setName("线程" + i);
            thread.start();
        }
    }
}

运行结果:

-------------------
-------------------
-------------------
-------------------
-------------------
线程0--->线程3的数据
线程1--->线程1的数据
线程2--->线程3的数据
线程3--->线程3的数据
线程4--->线程3的数据

使用普通的线程进行设置和获取时,可能会出现线程1访问到了线程2的数据。使用ThreadLocal即可解决:

public class MyDemo01 {
    ThreadLocal<String> t1= new ThreadLocal<>();
    private String content;

    public String getContent() {
//        return content;
        return t1.get();
    }

    public void setContent(String content) {
        t1.set(content);
//        this.content = content;
    }
……………………省略后续代码………………

变量被绑定到ThreadLocal对象上,每个线程拿到的值都是自己线程的,不会串。

ThreadLocal类与synchronized关键字

以上例子的另一种改法是:

    public static void main(String[] args) {
        MyDemo02 demo = new MyDemo02();
        for (int i = 0; i < 5; i++) {
            Thread thread = new Thread(new Runnable() {
                @Override
                public void run() {
                    synchronized (MyDemo02.class){
                        demo.setContent(Thread.currentThread().getName() + "的数据");
                        System.out.println("-------------------");
                        System.out.println(Thread.currentThread().getName() + "--->" + demo.getContent());
                    }
                }
            });
            thread.setName("线程" + i);
            thread.start();
        }
    }

使用synchronized关键字锁住代码

区别:

synchronizedThreadLocal
原理同步机制采用“以时间换空间”的方式,只提供了一份变量,让不同的线程排队访问Thread Local采用“以空间换时间”的方式,为每一个线程都提供了一份变量的副本,从而实现同时访问而互不干扰
侧重点多个线程之间访问资源的同步多线程中让每个线程之间的数据相互隔离
总结:在刚刚的案例中,虽然使用ThreadLocal和synchronized都能解决问题,但是ThreadLocal可以使程序拥有更高的并发性
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值