深入分析java内存模型(注意和java内存结构的区别)

简介: 最近在更java多线程相关的文章,正好有人问我一些java内存模型的问题,因此花了一些时间,好好地了解一下。本篇文章主要是为了解决以下几个问题?1、java内存模型和java内存结构有什么区别?2、为什么要有内存模型?3、java的内存模型是什么样子的?这篇文章,基本上不会涉及到代码,全是一些概念性的知识,但是也是面试常问和java进阶所需要掌握的必要的基本知识点,所以,希望你耐着性子,慢慢来。

一、java内存模型和java内存结构有什么区别


1、java内存结构

记得是在好几年前研究Android的时候,看的java内存模型,时常和java内存结构分不清,因此,这一小节是针对小白或者是对其概念还不理解的人。

我们都知道,我们的java代码其实是不能直接运行的,他要经过一系列的步骤。看下图:

v2-454219a8002110ddaa25b385a1ce3d63_1440w.jpg我们的java文件,首先要经过编程成为class文件,然后通过类装载器加载到jvm中去执行。这个jvm(红色虚线框起来的这部分)就是java运行时数据区,意思就是java代码在运行的时候,这些数据要存放在不同的内存空间里面。jvm就是指代这个的。当然了上面的运行时数据区jvm是jdk1.7版本的。也就是说不同的jdk版本,这个jvm长得是不一样的。我们可以把java7的内存结构拿出来:

v2-340c6442c9653260e22091827a9a3111_1440w.jpg

我们可以看到一共划分了5个部分,其中java堆区和方法区还是所有线程共享的一片区域。为什么要所有线程共享呢?因为假设一个数据,每个线程都保留一份,那其中有一个线程调皮,把这个数据更改了。其他的线程发现自己的数据没有变,这就出现了问题了。于是设计成了所有线程共享,java内存模型出来了。


2、java内存模型


java内存模型也叫做JMM,但是这个模型可不是像java内存结构一样,是真实存在的。java内存模型是一个抽象出来的概念。意思是把一部分内存区域设计成所有线程共享的,一个线程对数据更改,其他线程就能立刻知道。这种设计的方法叫做内存模型。我们可以提前看一下:java内存模型长什么样。

v2-68e920a8c61825303bfd0fe7b5eaec58_1440w.jpg

这就是java内存模型,也就是多个线程共享同一份数据。现在不知道你理解java内存模型和java内存结构的区别了没有,我们可以这样来总结一下:


(1)java内存结构是解决java中的数据如何存放的问题。

(2)java内存模型是解决java中多个线程共享数据的问题。

OK,到了这基本上就算是把两者的区别介绍完了。下面就来看看为什么要有内存模型吧


二、为什么要有内存模型


深入理解java虚拟机是从硬件的发展来分析的。因此,我也将从这个角度来分析。


阶段一


在计算机发展的第一个阶段,程序是在CPU中运行,数据在主存中保存。随着技术的发展,CPU的速度越来越多高,但是主存的速度却没有提高太多。就好比是下面这种情况:

v2-6a522414bbd68a64a87b98d14b26d74d_1440w.jpg

阶段二


为了解决上面的问题,于是乎出现了缓存,里面存放了一些CPU经常使用的主存数据,缓存的速度和CPU差不多,当CPU查找数据的时候,首先从缓存中查找,没有的话再从主存中查找。写数据的时候,先写缓存的数据,然后再更新到主存中。这样一种机制使得速度提高很多。


阶段三


技术继续发展,在上面缓存的基础上出现了一级二级三级缓存,查找也是逐层的,第一级缓存没有就到第二层,就这样以此类推。这时候CPU也得到了快读发展,由之前的一个核变成了多核CPU(一个CPU变成了多部分)。

v2-7a68998e0200a2b35c75c42018ba419c_1440w.jpg

这时候呢,之前只能同时跑一个线程,现在就能跑很多个线程了。而且从上面我们可以看到,每一个核都有相应的缓存区,但是主内存还是哪一个。既然能同时跑多个线程,那速度肯定杠杠滴就上去了吧,不跑不知道,一跑吓一跳,立马出现了很多个问题。


问题一:缓存一致性问题


也就是说,每一个核都有自己的缓存区,但是这些缓存区保存的数据却不一样。一张图就明白了:

v2-0cf25624ddf41fe329fb4ba8eb3be784_1440w.jpg

问题二:处理器优化和指令重排


这问题的意思是,既然CPU有这么多内核,肯定是想让资源得到充分利用,于是把我们写的程序拆分,对一些代码进行乱序处理,这就是处理器优化。而且,java虚拟机一看CPU的这个操作真的强,于是就模仿了一下,创建了即时编译器(JIT),这个编译器也会做指令重排的操作。很明显,我们的代码顺序被打乱,指令被重排,就可能不会按照我们的意愿去执行了。


上面出现的这些问题,好像都是从硬件的角度来分析的,《深入理解java虚拟机》一书于是引出了软件的问题,也就是说,上面的这些问题如果转化到软件层次会带来什么问题呢?


问题三:软件问题


(1)原子性问题


首先缓存一致性问题在程序中会带来原子性问题,原子性问题是什么意思呢?你首先就要先理解原子。在生物里面原子叫做不可再分的物质。在软件里面,原子叫做不可再分的程序操作。而原子性问题肯定就是打破了这个规则,也就是说在这个操作中又进行了拆分。

在缓存一致性问题中,两个CPU内核中a的数据不一致,也就是说两个CPU内核读取主存a的值是不一样的。那么对于a的更改这个操作肯定就不是原子性,在A更改的过程中,两个CPU内核同时进行了更改。


(2)可见性问题


上面在介绍原子性问题的时候说了,两个线程(CPU内核)访问同一个变量时,线程2修改了这个变量的值,但是线程1却没有看到其修改,所以读的仍然是旧数据。

v2-0cf25624ddf41fe329fb4ba8eb3be784_1440w (1).jpg

(3)有序性


也就是程序没有按照指定的顺序去执行。

可见性问题和有序性问题就是由于处理器优化和执行重排造成的。


阶段四


针对于这么多问题,于是java虚拟机提出了一个java内存模型。有效地解决了上面出现的这三个问题:


三、java内存模型


1、解释


为了和开头进行对照,我们再给出以此他的内存模型图:

v2-68e920a8c61825303bfd0fe7b5eaec58_1440w (1).jpg

从这张图我们分析一下java内存模型是如何解决上面的三个问题的。


规则一:所有的数据都在主内存中。

规则二:每个线程都保留一份共享变量的副本。线程对变量的所有操作都必须在这个副本内存中进行,而不能直接读写主内存。

规则三:不同的线程之间也无法直接访问对方工作内存中的变量,线程间变量的传递均需要自己的工作内存和主存之间进行数据同步进行。


看文字有点乱,我们举个例子,这个例子我觉得不太恰当,但是结合着上面的概念,相信你会明白的。在古代的时候经常发生旱灾,朝廷去赈灾,于是拿了一个大铁锅煮粥,这个大铁锅就是主内存,里面的粥就是数据。而每个难民就代表了不同的线程。


(1)难民都有一个碗,盛放一碗粥。就好比是每个线程都有一个本地内存,有一个主内存的副本。

(2)难民喝粥就只能在自己碗里,不能直接爬到锅中喝粥,就好比线程只能在自己的本地内存中操作数据,不能直接到主内存读写数据。

(3)其中一个难民想要把粥分给伙伴,怎么办呢?这个难民要先把粥倒进锅里,同伴再去锅里盛。就好比线程不能直接访问对方的内存,他们之间的数据传递都是通过主内存。

这个例子希望你能明白。现在这个模型算是出来了,还有一个问题没有解决,那就是java提供了什么东西来实现的这三个规则呢?


由于提供的机制太多,我们可以简单的例举几个,比如说synchronized关键字保证了原子性,volatile关键字保证了可见性。synchronized关键字和volatile关键字保证了有序性。当然还有很多的Lock机制,并发包里面等等都是为了解决这三个问题提出来的。


2、happens-before原则


其中有一条很重要的规则叫做happens-before原则,这条原则是为了解决可见性提出来的。什么意思呢?


如果一个操作的执行结果需要对另一个操作可见,那么这两个操作之间必须要存在happens-before关系。举个例子:


(1)程序顺序规则:一个线程中的每个操作发生在后一个操作之前,这就是happens-before。

(2)锁规则:对于锁机制,一定要先加锁,才能解锁,这也是happens-before。

(3)volatile域规则:对一个volatile域的写操作一定要发生在读操作前面。

上面是在程序角度来看的,举一个最简单不过的例子,你必须要做饭,才能够吃到饭。

这篇文章当然不是最详细的最全面的,只是希望大家有所收获,如有问题欢迎批评指正

相关文章
|
1天前
|
存储 IDE Java
java设置栈内存大小
在Java应用中合理设置栈内存大小是确保程序稳定性和性能的重要措施。通过JVM参数 `-Xss`,可以灵活调整栈内存大小,以适应不同的应用场景。本文介绍了设置栈内存大小的方法、应用场景和注意事项,希望能帮助开发者更好地管理Java应用的内存资源。
14 1
|
6天前
|
存储 缓存 安全
阿里云服务器计算型c7/c8y/c8i,通用型g7/g8y/g8i,内存型r7/r8y/r8i区别及选择参考
为了满足不同企业级用户的多样化需求,阿里云在当下的活动中推出了多款计算型、通用型和内存型的云服务器实例,包括计算型c7/c8y/c8i、通用型g7/g8y/g8i以及内存型r7/r8y/r8i等。这些实例各具特色,适用于不同的应用场景和业务需求。本文将为您详细解析这些实例的区别,以及选择参考,帮助您根据自己的需求选择合适的阿里云服务器实例。
|
2天前
|
存储 固态存储
磁盘和内存的区别
存储特性: • 磁盘:非易失性存储,数据在断电后不会丢失,适合长期存储数据。 • 内存:易失性存储,数据在断电后会丢失,适合临时存储当前运行的程序和数据。 容量: • 磁盘:容量通常较大,从几百GB到数TB不等,适合存储大量的文件和数据。 • 内存:容量相对较小,一般在几GB到几十GB之间,用于提供快速的临时存储空间。 速度: • 磁盘:读写速度较慢,HDD一般在几十MB/s,SSD可以达到几百MB/s甚至数GB/s。 • 内存:读写速度非常快,通常在几十纳秒到几百纳秒之间,能够快速响应CPU的指令。
13 2
|
1月前
|
Java 程序员 调度
Java 高级面试技巧:yield() 与 sleep() 方法的使用场景和区别
本文详细解析了 Java 中 `Thread` 类的 `yield()` 和 `sleep()` 方法,解释了它们的作用、区别及为什么是静态方法。`yield()` 让当前线程释放 CPU 时间片,给其他同等优先级线程运行机会,但不保证暂停;`sleep()` 则让线程进入休眠状态,指定时间后继续执行。两者都是静态方法,因为它们影响线程调度机制而非单一线程行为。这些知识点在面试中常被提及,掌握它们有助于更好地应对多线程编程问题。
66 9
|
7天前
|
Java Shell 数据库
【YashanDB 知识库】kettle 同步大表提示 java 内存溢出
【问题分类】数据导入导出 【关键字】数据同步,kettle,数据迁移,java 内存溢出 【问题描述】kettle 同步大表提示 ERROR:could not create the java virtual machine! 【问题原因分析】java 内存溢出 【解决/规避方法】 ①增加 JVM 的堆内存大小。编辑 Spoon.bat,增加堆大小到 2GB,如: if "%PENTAHO_DI_JAVA_OPTIONS%"=="" set PENTAHO_DI_JAVA_OPTIONS="-Xms512m" "-Xmx512m" "-XX:MaxPermSize=256m" "-
|
1月前
|
安全 Java 程序员
Java面试必问!run() 和 start() 方法到底有啥区别?
在多线程编程中,run和 start方法常常让开发者感到困惑。为什么调用 start 才能启动线程,而直接调用 run只是普通方法调用?这篇文章将通过一个简单的例子,详细解析这两者的区别,帮助你在面试中脱颖而出,理解多线程背后的机制和原理。
70 12
|
2月前
|
存储 缓存 资源调度
阿里云服务器经济型、通用算力型、计算型、通用型、内存型实例区别与选择指南
在我们通过阿里云的活动选购云服务器的时候会发现,相同配置的云服务器往往有多个不同的实例可选,而且价格差别也比较大,这会是因为不同实例规格的由于采用的处理器不同,底层架构也有所不同(例如X86 计算架构与Arm 计算架构),因此不同实例的云服务器其性能与适用场景是有所不同。本文将详细解析阿里云的经济型、通用算力型、计算型、通用型和内存型实例的性能特点及适用场景,帮助用户根据自己的业务需求做出明智的选择。
|
2月前
|
Java
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
今日分享的主题是如何区分&和&&的区别,提高自身面试的能力。主要分为以下四部分。 1、自我面试经历 2、&amp和&amp&amp的不同之处 3、&对&&的不同用回答逻辑解释 4、彩蛋
|
1天前
|
存储 监控 Java
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
88 60
【Java并发】【线程池】带你从0-1入门线程池
|
13天前
|
Java 程序员 开发者
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
72 14