我的编程空间,编程开发者的网络收藏夹
学习永远不晚

jvm中java内存模型的示例分析

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

jvm中java内存模型的示例分析

这篇文章主要介绍了jvm中java内存模型的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。

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

 

1、java内存结构

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

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

jvm中java内存模型的示例分析

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

jvm中java内存模型的示例分析

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

 

2、java内存模型

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

jvm中java内存模型的示例分析

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

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

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

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

 

二、为什么要有内存模型

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

 

阶段一

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

jvm中java内存模型的示例分析

 

阶段二

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

 

阶段三

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

jvm中java内存模型的示例分析

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

问题一:缓存一致性问题

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

jvm中java内存模型的示例分析

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

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

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

问题三:软件问题

(1)原子性问题

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

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

(2)可见性问题

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

jvm中java内存模型的示例分析

(3)有序性

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

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

 

阶段四

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

 

三、java内存模型

 

1、解释

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

jvm中java内存模型的示例分析

从这张图我们分析一下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域的写操作一定要发生在读操作前面。

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

感谢你能够认真阅读完这篇文章,希望小编分享的“jvm中java内存模型的示例分析”这篇文章对大家有帮助,同时也希望大家多多支持编程网,关注编程网行业资讯频道,更多相关知识等着你来学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

jvm中java内存模型的示例分析

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

jvm中java内存模型的示例分析

这篇文章主要介绍了jvm中java内存模型的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、java内存模型和java内存结构有什么区别 1、java内存结构记得是
2023-06-19

JVM系列之内存模型的示例分析

这篇文章主要介绍JVM系列之内存模型的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!1. 内存模型和运行时数据区这一章学习java虚拟机内存模型(Java Virtual machine menory mod
2023-06-15

Java内存模型的示例分析

这篇文章主要为大家展示了“Java内存模型的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Java内存模型的示例分析”这篇文章吧。1. 为什么要有内存模型?要想回答这个问题,我们需要先弄
2023-06-29

java之JVM架构模型的示例分析

小编给大家分享一下java之JVM架构模型的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!Java可以用来干什么Java主要应用于:1. web开发;2.
2023-06-14

java中JVM运行时内存整理的示例分析

这篇文章给大家分享的是有关java中JVM运行时内存整理的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。Java可以用来干什么Java主要应用于:1. web开发;2. Android开发;3. 客户端开
2023-06-14

JVM内存区域的示例分析

这篇文章主要介绍了JVM内存区域的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。JVM内存区域我们在编写程序时,经常会遇到OOM(out of Memory)以及内存
2023-06-05

Java内存模型之重排序的示例分析

小编给大家分享一下Java内存模型之重排序的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!一、数据依赖性如果两个操作访问同一个变量,而且这两个操作中有一个操作为写操作,此时这两个操作之间存在数据依赖性。数据依赖性分
2023-06-15

JVM中内存区域与内存溢出的示例分析

小编给大家分享一下JVM中内存区域与内存溢出的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!Java内存区域与内存溢出异常运行时数据区域程序计数器当前线程
2023-06-17

Java内存模型顺序一致性的示例分析

小编给大家分享一下Java内存模型顺序一致性的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!简介:顺序一致性内存模型是一个理论参考模型,处理器的内存模型和
2023-06-25

解析JVM内存监控流程的示例分析

解析JVM内存监控流程的示例分析,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。你对JVM内存监控是否熟悉,这里向大家简单描述一下,这里是亲自测试的详细配置过程,
2023-06-17

java内存模型讨论及案例分析

常用内存选项 -Xmx: 最大堆大小 -Xms:最小堆大小 -Xss :线程堆栈大小,默认1M 生产环境最好保持 Xms = Xmx java内存研究 内存布局 可见: 堆大小 = 新生代 + 老年代,新生代=E+From Survivo
2023-08-30

java中jvm关闭的示例分析

这篇文章给大家分享的是有关java中jvm关闭的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。Java有哪些集合类Java中的集合主要分为四类:1、List列表:有序的,可重复的;2、Queue队列:有序
2023-06-14

Java原型模式的示例分析

这篇文章主要介绍了Java原型模式的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。1.思考问题现在有一只羊 tom,姓名为: tom,年龄为:1,颜色为:白色,请编写
2023-06-29

Redis缓存IO模型的示例分析

Redis缓存IO模型的示例分析,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。前言redis作为应用最广泛的nosql数据库之一,大大小小也经历过很多次升级。在4.0版本之
2023-06-21

JAVA内存泄漏的示例分析

本篇文章给大家分享的是有关JAVA内存泄漏的示例分析,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。Java内存泄漏是每个Java程序员都会遇到的问题,程序在本地运行一切正常,可
2023-06-03

Java内存模型的并发处理示例

这篇文章给大家分享的是有关Java内存模型的并发处理示例的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。Java的优点是什么1. 简单,只需理解基本的概念,就可以编写适合于各种情况的应用程序;2. 面向对象;3.
2023-06-14

Golang内存模型实例源码分析

这篇文章主要介绍“Golang内存模型实例源码分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Golang内存模型实例源码分析”文章能帮助大家解决问题。1. 简介(Introduction)Go
2023-07-05

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录