volatile 关键字

本文详细介绍了Java中volatile关键字的作用,包括变量可见性和禁止指令重排序。volatile确保多线程环境下变量的可见性,但不保证原子性,适用于状态独立的共享变量。示例代码展示了volatile在并发计数场景中的不足,以及作为线程退出标志的应用。总结了volatile使用条件,并与synchronized进行比较。
摘要由CSDN通过智能技术生成


一、volatile 关键字的作用(变量可见性、禁止重排序)

Java 语言提供了一种稍弱的同步机制,即 volatile 变量,用来确保将变量的更新操作通知到其他
线程。volatile 变量具备两种特性,volatile 变量不会被缓存在寄存器或者对其他处理器不可见的
地方,因此在读取 volatile 类型的变量时总会返回最新写入的值。

1. 变量可见性
其一是保证该变量对所有线程可见,这里的可见性指的是当一个线程修改了变量的值,那么新的
值对于其他线程是可以立即获取的。
3. 禁止重排序
volatile 禁止了指令重排。
4. 比 sychronized 更轻量级的同步锁
在访问 volatile 变量时不会执行加锁操作,因此也就不会使执行线程阻塞,因此 volatile 变量是一种比 sychronized 关键字更轻量级的同步机制。volatile 适合这种场景:一个变量被多个线程共享,线程直接给这个变量赋值。当对非 volatile 变量进行读写的时候,每个线程先从内存拷贝变量到 CPU 缓存中。如果计算机有多个 CPU,每个线程可能在不同的 CPU 上被处理,这意味着每个线程可以拷贝到不同的 CPU cache 中。而声明变量是 volatile 的,JVM 保证了每次读变量都从内存中读,跳过 CPU cache
这一步。
6. 适用场景
值得说明的是对 volatile 变量的单次读/写操作可以保证原子性的,如 long 和 double 类型变量,但是并不能保证 i++这种操作的原子性,因为本质上 i++是读、写两次操作。在某些场景下可以代替 Synchronized。但是,volatile 的不能完全取代 Synchronized 的位置,只有在一些特殊的场景下,才能适用 volatile。总的来说,必须同时满足下面两个条件才能保证在并发环境的线程安全:
(1)对变量的写操作不依赖于当前值(比如 i++),或者说是单纯的变量赋值(boolean
flag = true)

(2)该变量没有包含在具有其他变量的不变式中,也就是说,不同的 volatile 变量之间,不
能互相依赖
。只有在状态真正独立于程序内其他内容时才能使用 volatile。
代码示例:

  private  static  volatile  int a=0;

    public static void main(String[] args) {
        CountDownLatch count=new CountDownLatch(10000);

        for (int j = 0; j < 10000; j++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    for (int i = 0; i < 10; i++) {
                        a++;
                    }
                    count.countDown();
                }
            }).start();
        }

        System.out.println("等待10000个任务执行完成");
        try {
            count.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(a);

    }

结果为:
在这里插入图片描述
a的值小于10000,说明volatile并不能保证 i++这种操作的原子性,因为本质上 i++是读、写两次操作。
代码示例:

private  static  volatile  boolean flag=false;

    public static void main(String[] args) {

        new Thread(new Runnable() {
            @SneakyThrows
            @Override
            public void run() {

                Thread.sleep(10000L);
                flag=true;
            }
        }).start();
        new Thread(new Runnable() {
            @SneakyThrows
            @Override
            public void run() {
              while (!flag){
                  Thread.sleep(3000L);
                  System.out.println("flag线程标志为false");
              }
                System.out.println("flag线程标志变为true,退出循环");
            }
        }).start();

    }

flag变量作为一个线程退出循环体的标识。


总结

以上为个人学习过程中对java的一些学习总结,如有错误,欢迎各位批评指导,如有侵权,请联系本人删除,谢谢!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值