并发编程系列教程(03) - Volatile

简介: 并发编程系列教程(03) - Volatile

代码已上传到Github:https://github.com/ylw-github/Java-ThreadDemo

1. 什么是Volatile(英: [ˈvɒlətaɪl] 美: [ˈvɑːlətl])

可见性也就是说一旦某个线程修改了该被volatile修饰的变量,它会保证修改的值会立即被更新到主存,当有其他线程需要读取时,可以立即获取修改之后的值。

在Java中为了加快程序的运行效率,对一些变量的操作通常是在该线程的寄存器或是CPU缓存上进行的,之后才会同步到主存中,而加了volatile修饰符的变量则是直接读写主存。

Volatile 保证了线程间共享变量的及时可见性,但不能保证原子性。

public class VolatileDemo {
    public static class ThreadVolatileDemo extends Thread {
        public boolean flag = true;
        @Override
        public void run() {
            super.run();
            System.out.println("开始执行子线程....");
            while (flag) {
            }
            System.out.println("线程停止");
        }
        public void setRunning(boolean flag){
            this.flag = flag;
        }
    }
    public static void main(String[] args) throws InterruptedException {
        ThreadVolatileDemo threadVolatileDemo = new ThreadVolatileDemo();
        threadVolatileDemo.start();
        Thread.sleep(3000);
        threadVolatileDemo.setRunning(false);
        System.out.println("flag 已经设置成false");
        Thread.sleep(1000);
        System.out.println(threadVolatileDemo.flag);
    }
}

运行结果:

已经将结果设置为fasle为什么?还一直在运行呢。

原因:线程之间是不可见的,读取的是副本,没有及时读取到主内存结果。

解决办法使用Volatile关键字将解决线程之间可见性, 强制线程每次读取该值的时候都去“主内存”中取值。

2. Volatile特性

1.保证此变量对所有的线程的可见性,这里的“可见性”,如本文开头所述,当一个线程修改了这个变量的值,volatile 保证了新值能立即同步到主内存,以及每次使用前立即从主内存刷新。但普通变量做不到这点,普通变量的值在线程间传递均需要通过主内存(详见:Java内存模型)来完成。

2.禁止指令重排序优化。有volatile修饰的变量,赋值后多执行了一个“load addl $0x0, (%esp)”操作,这个操作相当于一个内存屏障(指令重排序时不能把后面的指令重排序到内存屏障之前的位置),只有一个CPU访问内存时,并不需要内存屏障;(什么是指令重排序:是指CPU采用了允许将多条指令不按程序规定的顺序分开发送给各相应电路单元处理)。

volatile 性能:

volatile 的读性能消耗与普通变量几乎相同,但是写操作稍慢,因为它需要在本地代码中插入许多内存屏障指令来保证处理器不发生乱序执行。

3. Volatile与Synchronized区别

1)从而我们可以看出volatile虽然具有可见性但是并不能保证原子性。

2)性能方面,synchronized关键字是防止多个线程同时执行一段代码,就会影响程序执行效率,而volatile关键字在某些情况下性能要优于synchronized。

但是要注意volatile关键字是无法替代synchronized关键字的,因为volatile关键字无法保证操作的原子性。

总结

目录
相关文章
|
5月前
|
存储 缓存 Java
volatile底层原理详解
volatile底层原理详解
51 0
|
11月前
|
安全 Java
并发编程系列教程(02) - 多线程安全
并发编程系列教程(02) - 多线程安全
26 0
|
2月前
|
缓存 Java
【多线程面试题二十三】、 说说你对读写锁的了解volatile关键字有什么用?
这篇文章讨论了Java中的`volatile`关键字,解释了它如何保证变量的可见性和禁止指令重排,以及它不能保证复合操作的原子性。
|
5月前
|
缓存 安全 Java
JUC并发编程之volatile详解
Java内存模型是Java虚拟机(JVM)规范中定义的一组规则,用于屏蔽各种硬件和操作系统的内存访问差异,保证多线程情况下程序的正确执行。Java内存模型规定了线程之间如何交互以及线程和内存之间的关系。它主要解决的问题是可见性、原子性和有序性。
|
11月前
|
安全 Java
并发编程系列教程(11) - 原子类
并发编程系列教程(11) - 原子类
39 0
|
11月前
|
存储 缓存 Java
并发编程系列教程(05) - Java内存模型
并发编程系列教程(05) - Java内存模型
48 0
|
11月前
|
缓存 编译器 程序员
并发编程系列教程(04) - 重排序
并发编程系列教程(04) - 重排序
39 0
|
缓存 Java 编译器
JUC并发编程学习(十七) -5分钟搞懂volatile
JUC并发编程学习(十七) -5分钟搞懂volatile
JUC并发编程学习(十七) -5分钟搞懂volatile
|
SQL 缓存 安全
Java并发编程学习系列七:深入了解volatile关键字
Java并发编程学习系列七:深入了解volatile关键字
112 0
Java并发编程学习系列七:深入了解volatile关键字
|
缓存 Java
Java并发编程系列2 - volatile
主要讲解volatile的相关知识,以及容易遇到的坑。
156 1
Java并发编程系列2 - volatile