volatile底层原理的再次理解

volatile

volatile作用于变量。能够保证多个线程对该变量修改的可见,一个线程修改了该变量后,另一个线程能够立即感知到。因为volatile修饰的变量存放在主内存中,没有在工作内存中存放副本,变量的读写都是直接操作主内存。什么是主内存和工作内存,看java内存模型(JMM)规定。

JMM java内存模型简述

java内存模型分为两部分:

  • 主内存: 多个线程共享的区域,也就是堆和方法区。主内存存放共享变量。
  • 工作内存:线程的工作内存是线程独有的区域,也就是程序计数器,虚拟机栈,和本地方法栈。工作内存方法共享变量的副本。
    JMM规定线程对变量的读写在工作内存中进行,工作内存中存放了主内存中的一个副本,线程在工作内存中修改了数据后会将数据数据同步到主内存中去,供其他线程读取。但是这过程有时间差,可能导致其他线程读取到的数据不是最新的,导致「可见性」问题。volatile修饰的变量存放在主内存中,不会在工作内存中存放副本。

可见性和有序性保证

变量的可见性: volatile修饰的变量保存在主内存中,不会存放副本在工作内存中,解决了可见性问题

有序性: volatile通过硬件层面的指令屏障,读写屏障来禁止指令排序。在读取数据之前加一个读屏障保证高速缓存中的数据能从主内存中获取最新的数据。在写数据之后,用一个写屏障保证高速缓存中的数据能够及时同步到工作内存中,让其他线程可见。通过读写屏障的配合来保证禁止局部指令的重排,从而保证有序性。

不能保证原子性

不能保证原子性,比如多个线程对一个volatile修饰的变量执行i++操作,最后执行的结果和预期不符合。

java 复制代码
import java.util.concurrent.CountDownLatch;

/**
 * volatile不具有原子性
 * 多个线程i++操作得到的结果和预期不符
 */
public class Volatile {
    private static int cnt;
    public static void add(){
        for (int i = 0; i < 10000; i++) {
            cnt++;
        }
    }

    public static void main(String[] args) {
        CountDownLatch countDownLatch = new CountDownLatch(2);// 将执行完的线程阻塞,当计数器为0,线程唤醒
        for (int i = 0; i < 2; i++) {
            new Thread(()->{
                add();
                countDownLatch.countDown();
            }, i+"").start();
        }

        try {
            countDownLatch.await(); // 等待所有线程执行完(计数器为0),线程才会唤醒,否则会一直阻塞
            System.out.println(cnt);// 预计结果是2000
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

参考

一篇文章彻底搞懂volatile关键字底层实现原理

相关推荐
golang学习记8 分钟前
IDEA 2026.1官宣:AI 建议免费了!
java·ide·intellij-idea
light blue bird20 分钟前
原生控件GDI完成作业协同界面
jvm·数据库·.net·winform·gdi+界面
cccccc语言我来了28 分钟前
Linux(9)操作系统
android·java·linux
东离与糖宝34 分钟前
金三银四Java校招面经:从双非到大厂Offer,我只准备了这些
java·面试
禾小西1 小时前
Spring AI :Spring AI的介绍
java·人工智能·spring
愤豆1 小时前
05-Java语言核心-语法特性--模块化系统详解
java·开发语言·python
bksczm1 小时前
文件流(fstream)
java·开发语言
NGC_66111 小时前
Java 线程池阻塞队列与拒绝策略
java·开发语言
小碗羊肉1 小时前
【从零开始学Java | 第二十二篇】List集合
java·开发语言
m0_716765231 小时前
C++提高编程--STL常用容器(set/multiset、map/multimap容器)详解
java·开发语言·c++·经验分享·学习·青少年编程·visual studio