ThreadLocal和Volatile

本文探讨了ThreadLocal如何实现线程数据隔离,确保多线程环境中的数据安全性,以及volatile关键字的作用,包括保证共享变量的可见性和防止指令重排。尽管volatile无法提供原子性,但在特定场景下仍是有用的。

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

一、ThreadLocal

  线程绑定,辅助一个线程持有自己的数据,这个数据与其他线程不共享。把数据绑定到线程,线程当做一条流水线,来传递数据。多线程并行时,数据是安全的。
线程绑定
例:

public class Test {
	static ThreadLocal<Double> threadLocal = new ThreadLocal<>();
	
	public static void main(String[] args) {
		new Thread() {
			@Override
			public void run() {
				a();
				b();
				c();
				removeData();
			}
		}.start();
		new Thread() {
			@Override
			public void run() {
				a();
				b();
				c();
				removeData();
			}
		}.start();
		new Thread() {
			@Override
			public void run() {
				a();
				b();
				c();
				removeData();
			}
		}.start();
	}
	
	static void a() {
		double d = getData();
		String n = Thread.currentThread().getName();
		System.out.println(n+" - "+d);
	}
	static void b() {
		double d = getData();
		String n = Thread.currentThread().getName();
		System.out.println(n+" - "+d);
	}
	static void c() {
		double d = getData();
		String n = Thread.currentThread().getName();
		System.out.println(n+" - "+d);
	}
	
	static double getData() {
		Double d = threadLocal.get();//从当前线程获取绑定的数据
		if (d == null) {//如果线程上没有绑定过数据
			//产生一个数据,绑定到当前线程
			d = Math.random();
			threadLocal.set(d);
		}
		return d;
	}
	
	static void removeData() {
		threadLocal.remove();
	}
	
}

二、 volatile

  1. 共享变量的可见性
  2. 禁止指令重排的优化
    cpu 为了提高运算效率,可能根据一定规则,对运算指令重新排序
     a = 6;       flag = true
   flag = true;     a = 6;
  1. 不能包装原子性, 只能靠锁来解决
  2. 什么时候使用volatile
 - volatile 易变,不稳定
 - 多个线程频繁访问,修改变量

volatile
例:

public class Test {
	 /*
     * volatile 保证数据的可见性
     *
     * volatile 涉及一些底层cpu指令
     * 可以让一个cpu缓存,监视其他cpu的数据变化,
     * 发现其他cpu修改了数据,会把高速缓存数据标记成废弃,
     * 会重新从内存复制新数据
     */
    public static volatile boolean flag = false;//共享的变量

    public static void main(String[] args) throws InterruptedException {
        T1 t1 = new T1();
        T2 t2 = new T2();
        new Thread(t1, "T1").start();
        Thread.sleep(1000);//为了保证t1比t2先启动,sleep一下
        new Thread(t2, "T2").start();
    }

    static class T1 extends Thread {
        public void run() {
            while (true) {
                if (flag) {
                    System.out.println(Thread.currentThread().getName() + " - flag : " + flag);
                    break;
                }
            }
        }
    }

    static class T2 extends Thread {
        public void run() {
            flag = true;
            System.out.println(Thread.currentThread().getName() + " - flag : " + flag);
        }
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值