Java内存模型(JMM)

简介: Java内存模型(JMM)是一个抽象概念,用于规范程序中各种变量(实例字段、静态字段及数组元素)的访问方式,确保不同Java虚拟机(JVM)上的并发程序结果一致可靠。JMM定义了主存储器(所有线程共享)与工作存储器(线程私有)的概念,线程间通过主存储器进行通信。JMM具备三大特性:原子性(确保基本读写操作的不可分割)、可见性(确保一个线程对共享变量的修改对其他线程可见)、有序性(防止指令被处理器或编译器重排序影响程序逻辑)。通过这些特性,JMM解决了多线程环境下的数据一致性问题。

什么是Java内存模型

JMM本身只是一个抽象的概念,并不真实存在,它描述的是一种规则或规范;通过这组规范,定义了程序中对各种变量(包括实例字段,静态字段和构成数组对象的元素)的访问方式。需要每个JVM的实现都要遵守这样的规范;有了JMM规范的保障后,并发程序运行在不同虚拟机上时,得到的程序结果才是安全可靠可信赖的,如果没有JMM内存模型来规范,那经过不同JVM翻译之后,就可能出现,运行结果不相同或不正确。

简单说JMM就是屏蔽了各种硬件和操作系统的访问差异,保证Java程序在各种平台下对内存的访问都能保证效果一致的机制规范。

JMM还抽象出主存储器(Main Memory)和工作存储器(Working Memory)两种:

主存储器是实例对象所在的区域,所有实例都存在于主存储器内,主存储器是所有线程共享的

工作存储器是线程所拥有的作业区,每个线程都有其专用的工作存储器;工作存储器存有主存储器中必要部分的拷贝,称为工作拷贝(Working Copy)

所以线程无法直接对主内存进行操作,线程A想要和线程B通信,只能通过主存进行。

三大特性

JMM有三大特性:原子性、可见性、有序性

原子性

JMM保证了对共享变量的读取和写入可以被视为原子操作

为支持JMM,Java定义了8种原子操作,用来控制主存和工作内存之间的交互

  • read读取:作用于主内存,将共享变量从主内存传送到线程的工作内存中
  • load载入:作用于工作内存,把read读取的值放到工作内存中的副本变量中
  • store存储:作用于工作内存,把工作内存中的变量传送到主内存中
  • write写入:作用于主内存,把从工作内存中store传送过来的值写到主内存变量中
  • use使用:作用于工作内存,把工作内存的值传递给执行引擎,当虚拟机遇到需要使用这个变量的指令时,就会执行这个动作
  • assign赋值:作用于工作内存,把执行引擎获取到的值赋值给工作线程中的变量,当虚拟机遇到给变量赋值的指令时,就执行此操作
  • lock锁定:作用于主内存,把变量标记为现场独占状态
  • unlock解锁:作用于主内存,它将释放独占状态

可见性

多个线程访问共享变量时,一个线程如果修改变量值,在刷新到主内存之前,其他线程不一定能立即看到这个修改。

在JVM中,栈负责运行(主要是方法),堆中负责存储(比如new的对象),JVM运行程序的实体是线程,每个线程创建时,JVM都会为其创建一个工作内存,工作内存是每个现场私有数据区域;而Java内存模型中规定,所有变量都存储在主内存中,主内存是共享内存区域,所有线程都可以访问;但线程对变量的操作(读写)必须在自己工作内存中进行,首先要将变量从主内存拷贝到自己的工作内存空间,然后对变量操作,操作完成后,再将变量回写到主内存;由于不能直接操作主内存的变量,各个线程工作内存中存储着主内存变量副本,因此不同线程无法直接访问对方工作内存,线程间通信必须通过主内存完成。

同步的规定

  • 线程解锁前,必须把共享变量的值刷新回主内存
  • 线程加锁前,必须将主内存的最新值读取到自己的工作内存
  • 加锁解锁是同一把锁

可见性问题(缓存一致性问题):指在未加同步锁的多线程环境下,同时修改共享变量,导致结果与预期不符的问题。

代码复现

java

代码解读

复制代码

public class Demo {
    private static int num;
    public static void main(String[] args) throws InterruptedException {
        Thread[] threads = new Thread[100];
        CountDownLatch latch = new CountDownLatch(threads.length);
        for (int i = 0; i < threads.length; i++) {
            threads[i] = new Thread(() -> {
                for (int j = 0; j < 10000; j++) {
                    num++;
                }
                latch.countDown();
            });
        }
        Arrays.stream(threads).forEach(Thread::start);
        latch.await();
        System.out.println("预期值:" + threads.length * 10000 + ",实际值:" + num);
        // 预期值:1000000,实际值:189067
    }
}

同步锁

java

代码解读

复制代码

public class Demo {
    private static int num;
    public static void main(String[] args) throws InterruptedException {
        Thread[] threads = new Thread[100];
        CountDownLatch latch = new CountDownLatch(threads.length);
        ReentrantLock lock = new ReentrantLock();
        for (int i = 0; i < threads.length; i++) {
            threads[i] = new Thread(() -> {
                for (int j = 0; j < 10000; j++) {
                    lock.lock();
                    num++;
                    lock.unlock();
                }
                latch.countDown();
            });
        }
        Arrays.stream(threads).forEach(Thread::start);
        latch.await();
        System.out.println("预期值:" + threads.length * 10000 + ",实际值:" + num);
        // 预期值:1000000,实际值:1000000
    }
}

有序性

在本(单)线程内执行顺序按照代码的先后顺序来执行,所有的操作都是有序的,线程内似表现为串行;但在多线程内,所有的操作都是无序的。

重排序:处理器为提高程序运行效率,提高并行效率,可能会对代码进行优化,编译器认为重排序后程序的执行效率更优,这样一来代码执行顺序就未必是编写代码时候的顺序,在多线程情况下就可能会出错;但它也需要满足以下两个条件

  • 在的单线程环境下不能改变程序运行的结果
  • 存在数据依赖关系的不允许重排序

数据依赖性:如果两个操作访问同一个变量,且这两个操作中有一个为写,此时这两个操作存在数据依赖性;分为以下列三种类型,下面三种情况,只要重排两个操作执行顺序,程序的执行结果就会发生改变;所以编译器和处理器不会改变单线程或单处理器环境下存在数据依赖性操作的执行顺序;在多处理器或多线程之间的数据依赖性不被编译器和处理器考虑。

名称 代码示例 说明
写后读 a = 1;b = a; 写一个变量之后,再读这个变量
写后写 a = 1;a = 2; 写一个变量之后,再写这个变量
读后写 a = b;b = 1; 读一个变量之后,再写这个变量

有序性问题(指令重排序) :指在多线程环境下,由于执行语句重排序后,重排序代码块没有执行完,就切换到其他线程,导致计算结果与预期不符的问题;这就是编译器的编译优化给并发编程带来的有序性问题。

代码复现

java

代码解读

复制代码

public class Demo {
    private static int a, b, x, y;
    public static void main(String[] args) throws InterruptedException {
        for (int i = 0; i < 1_0000_0000; i++) {
            a = 0;
            b = 0;
            x = 0;
            y = 0;
            CountDownLatch latch = new CountDownLatch(2);
            Thread t1 = new Thread(() -> {
                a = 1;
                x = b;
                latch.countDown();
            });
            Thread t2 = new Thread(() -> {
                b = 1;
                y = a;
                latch.countDown();
            });
            t1.start();
            t2.start();
            latch.await();
            if (x == 0 && y == 0) {
                System.err.println("第" + i + "次出现(x=0,y=0)");
                break;
            }
            // 第144654次出现(x=0,y=0)
        }
    }
}

禁止指令重排

java

代码解读

复制代码

public class Demo {
    private static volatile int a, b, x, y;
    public static void main(String[] args) throws InterruptedException {
        for (int i = 0; i < 1_0000_0000; i++) {
            a = 0;
            b = 0;
            x = 0;
            y = 0;
            CountDownLatch latch = new CountDownLatch(2);
            Thread t1 = new Thread(() -> {
                a = 1;
                x = b;
                latch.countDown();
            });
            Thread t2 = new Thread(() -> {
                b = 1;
                y = a;
                latch.countDown();
            });
            t1.start();
            t2.start();
            latch.await();
            if (x == 0 && y == 0) {
                System.err.println("第" + i + "次出现(x=0,y=0)");
                break;
            }
        }
    }
}


转载来源:https://juejin.cn/post/7392481848284463167

相关文章
|
18天前
|
缓存 安全 Java
Java并发编程进阶:深入理解Java内存模型
Java并发编程进阶:深入理解Java内存模型
29 0
|
17天前
|
存储 算法 Java
深入浅出Java内存管理
【8月更文挑战第28天】Java的内存管理是每个Java开发者都绕不过去的技术话题。本文将通过生动的比喻和直观的例子,带你走进Java内存管理的奇妙世界。我们将一起探索对象在Java虚拟机中的生命周期,了解栈与堆的区别,以及垃圾回收机制如何默默守护着我们的应用程序。准备好,我们即将启程!
43 14
|
9天前
|
算法 安全 Java
Java内存管理:深入理解垃圾收集器
在Java的世界里,内存管理是一块基石,它支撑着应用程序的稳定运行。本文将带你走进Java的垃圾收集器(GC),探索它是如何默默守护着我们的内存安全。我们将从垃圾收集的基本概念出发,逐步深入到不同垃圾收集器的工作机制,并通过实例分析它们在实际应用中的表现。文章不仅旨在提升你对Java内存管理的认识,更希望你能通过这些知识优化你的代码,让程序运行更加高效。
30 3
|
17天前
|
监控 算法 Java
Java内存管理:垃圾收集器的工作原理与调优实践
在Java的世界里,内存管理是一块神秘的领域。它像是一位默默无闻的守护者,确保程序顺畅运行而不被无用对象所困扰。本文将带你一探究竟,了解垃圾收集器如何在后台无声地工作,以及如何通过调优来提升系统性能。让我们一起走进Java内存管理的迷宫,寻找提高应用性能的秘诀。
|
15天前
|
Kubernetes Cloud Native Java
云原生之旅:从容器到微服务的演进之路Java 内存管理:垃圾收集器与性能调优
【8月更文挑战第30天】在数字化时代的浪潮中,企业如何乘风破浪?云原生技术提供了一个强有力的桨。本文将带你从容器技术的基石出发,探索微服务架构的奥秘,最终实现在云端自由翱翔的梦想。我们将一起见证代码如何转化为业务的翅膀,让你的应用在云海中高飞。
|
7天前
|
安全 Java API
【性能与安全的双重飞跃】JDK 22外部函数与内存API:JNI的继任者,引领Java新潮流!
【9月更文挑战第7天】JDK 22外部函数与内存API的发布,标志着Java在性能与安全性方面实现了双重飞跃。作为JNI的继任者,这一新特性不仅简化了Java与本地代码的交互过程,还提升了程序的性能和安全性。我们有理由相信,在外部函数与内存API的引领下,Java将开启一个全新的编程时代,为开发者们带来更加高效、更加安全的编程体验。让我们共同期待Java在未来的辉煌成就!
34 11
|
8天前
|
安全 Java API
【本地与Java无缝对接】JDK 22外部函数和内存API:JNI终结者,性能与安全双提升!
【9月更文挑战第6天】JDK 22的外部函数和内存API无疑是Java编程语言发展史上的一个重要里程碑。它不仅解决了JNI的诸多局限和挑战,还为Java与本地代码的互操作提供了更加高效、安全和简洁的解决方案。随着FFM API的逐渐成熟和完善,我们有理由相信,Java将在更多领域展现出其强大的生命力和竞争力。让我们共同期待Java编程新纪元的到来!
32 11
|
5天前
|
监控 Java 大数据
【Java内存管理新突破】JDK 22:细粒度内存管理API,精准控制每一块内存!
【9月更文挑战第9天】虽然目前JDK 22的确切内容尚未公布,但我们可以根据Java语言的发展趋势和社区的需求,预测细粒度内存管理API可能成为未来Java内存管理领域的新突破。这套API将为开发者提供前所未有的内存控制能力,助力Java应用在更多领域发挥更大作用。我们期待JDK 22的发布,期待Java语言在内存管理领域的持续创新和发展。
|
1天前
|
存储 缓存 算法
Java中的内存管理:理解垃圾回收机制
本文将深入探讨Java中的内存管理,特别是垃圾回收机制。我们将从基本的内存分配开始,逐步解析垃圾回收的原理和过程,以及它对Java应用程序性能的影响。通过实例演示,我们会展示如何在Java中有效地管理和优化内存使用。最后,我们将讨论一些常见的内存泄漏问题及其解决方案。
|
16天前
|
缓存 Java
Java内存管理秘籍:掌握强软弱幻四大引用,让代码效率翻倍!
【8月更文挑战第29天】在Java中,引用是连接对象与内存的桥梁,主要分为强引用、软引用、弱引用和幻象引用。强引用确保对象生命周期由引用控制,适用于普通对象;软引用在内存不足时可被回收,适合用于内存敏感的缓存;弱引用在无强引用时即可被回收,适用于弱关联如监听器列表;幻象引用需与引用队列配合使用,用于跟踪对象回收状态,适用于执行清理工作。合理使用不同类型的引用车可以提升程序性能和资源管理效率。
37 4