JMM内存模型--重排序、可见性、原子性

本文详细介绍了Java内存模型(JMM)中的重排序、可见性和原子性三大要点。通过实例分析了重排序可能导致的问题及其解决办法,探讨了CPU缓存引起的可见性问题以及volatile关键字的作用。同时,文章还讲解了原子性概念,并以单例模式为例说明了如何保证原子操作。

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

为什么需要JMM(Java Memory Model)?

  • C语言不存在内存模型的概念,很多行为依赖于处理器,不同处理器的操作会导致运行结果不一样,所以不能保证并发安全
  • JMM是一组规范,也是工具类和关键字的原理
  • JMM最重要的三点内容:重排序、可见性、原子性

一、重排序

1、重排序例子

  • 下面是一个演示重排序的例子,一个线程执行a = 1; x = b,一个线程执行b = 1; x = b

    public class OutOfOrderExecution {
         
        private static int x = 0, y = 0;
        private static int a = 0, b = 0;
    
        public static void main(String[] args) throws InterruptedException {
         
            //计数器
            int i = 0;
            for (;;) {
         
                i++;
                x = 0;
                y = 0;
                a = 0;
                b = 0;
    
                //保证两个线程同时开始执行
                CountDownLatch latch = new CountDownLatch(3);
    
                Thread one = new Thread(new Runnable() {
         
                    @Override
                    public void run() {
         
                        try {
         
                            latch.countDown();
                            latch.await();
                        } catch (InterruptedException e) {
         
                            e.printStackTrace();
                        }
    
                        a = 1;
                        x = b;
                    }
                });
    
                Thread two = new Thread(new Runnable() {
         
                    @Override
                    public void run() {
         
                        try {
         
                            latch.countDown();
                            latch.await();
                        } catch (InterruptedException e) {
         
                            e.printStackTrace();
                        }
    
                        b = 1;
                        y = a;
                    }
                });
    
                two.start();
                one.start();
                latch.countDown();
                one.join();
                two.join();
    
                String result = "第" + i + "次(" + x + "," + y + ")";
                //四种情况:0,1/1,0/1,1/0,0
                if (x == 0 && y == 0
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值