- 1、本文档共53页,可阅读全部内容。
- 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
- 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
查看更多
深入理解Java 内存模型
juapkAndroid开发者社区【整理】
深入理解Java 内存模型(一)——基础
并发编程模型的分类
在并发编程中,我们需要处理两个关键问题:线程之间如何通信及线程之间如何同步(这里的线程是指并发执行的活动实体)。通信是
指线程之间以何种机制来交换信息。在命令式编程中,线程之间的通信机制有两种:共享内存和消息传递。
在共享内存的并发模型里,线程之间共享程序的公共状态,线程之间通过写-读内存中的公共状态来隐式进行通信。在消息传递的并发模
型里,线程之间没有公共状态,线程之间必须通过明确的发送消息来显式进行通信。
同步是指程序用于控制不同线程之间操作发生相对顺序的机制。在共享内存并发模型里,同步是显式进行的。程序员必须显式指定某个
方法或某段代码需要在线程之间互斥执行。在消息传递的并发模型里,由于消息的发送必须在消息的接收之前,因此同步是隐式进行的。
Java 的并发采用的是共享内存模型,Java 线程之间的通信总是隐式进行,整个通信过程对程序员完全透明。如果编写多线程程序的Java
程序员不理解隐式进行的线程之间通信的工作机制,很可能会遇到各种奇怪的内存可见性问题。
Java 内存模型的抽象
在java 中,所有实例域、静态域和数组元素存储在堆内存中,堆内存在线程之间共享(本文使用“共享变量”这个术语代指实例域,静态
域和数组元素)。局部变量(Local variables),方法定义参数(java 语言规范称之为formal method parameters )和异常处理器参数
(exception handler parameters )不会在线程之间共享,它们不会有内存可见性问题,也不受内存模型的影响。
Java 线程之间的通信由Java 内存模型(本文简称为JMM )控制,JMM 决定一个线程对共享变量的写入何时对另一个线程可见。从抽象
的角度来看,JMM 定义了线程和主内存之间的抽象关系:线程之间的共享变量存储在主内存(main memory)中,每个线程都有一个私
有的本地内存(local memory),本地内存中存储了该线程以读/写共享变量的副本。本地内存是JMM 的一个抽象概念,并不真实存在。
它涵盖了缓存,写缓冲区,寄存器以及其他的硬件和编译器优化。Java 内存模型的抽象示意图如下:
从上图来看,线程A 与线程B 之间如要通信的话,必须要经历下面2 个步骤:
1. 首先,线程A 把本地内存A 中更新过的共享变量刷新到主内存中去。
2. 然后,线程B 到主内存中去读取线程A 之前已更新过的共享变量。
下面通过示意图来说明这两个步骤:
如上图所示,本地内存A 和B 有主内存中共享变量x 的副本。假设初始时,这三个内存中的x 值都为0。线程A 在执行时,把更新后的x
值(假设值为 1)临时存放在自己的本地内存A 中。当线程A 和线程B 需要通信时,线程A 首先会把自己本地内存中修改后的x 值刷新
到主内存中,此时主内存中的x 值变为了 1。随后,线程B 到主内存中去读取线程A 更新后的x 值,此时线程B 的本地内存的x 值也变
为了1。
从整体来看,这两个步骤实质上是线程A 在向线程B 发送消息,而且这个通信过程必须要经过主内存。JMM 通过控制主内存与每个线程
的本地内存之间的交互,来为java 程序员提供内存可见性保证。
重排序
在执行程序时为了提高性能,编译器和处理器常常会对指令做重排序。重排序分三种类型:
1. 编译器优化的重排序。编译器在不改变单线程程序语义的前提下,可以重新安排语句的执行顺序。
2. 指令级并行的重排序。现代处理器采用了指令级并行技术(Instruction-Level Parallelism, ILP)来将多条指令重叠执行。如
果不存在数据依赖性,处理器可以改变语句对应机器指令的执行顺序。
3. 内存系统的重排序。由于处理器使用缓存和读/写缓冲区,这使得加载和存储操作看上去可能是在乱序执行。
从java 源代码到最终实际执行的指令序列,会分别经历下面三种重排序:
上述的1 属于编译器重排序,2 和3 属于处理器重排序。这些重排序都可能会导致多线程程序出现内存可见性问题。对于编译器,JMM
的编译器重排序规则会禁止特定类型的编译器重排序(不是所有的编译器重排序都要禁止)。对于处理器重排序,JMM 的处理器重排序
规则会要求java 编译器在生成指令序列时,插入特定类型的内存屏障(memory
您可能关注的文档
- 第八章 Linux命令.pptx
- 第八章 MHS-5200A函数信号发生器说明书.pdf
- 第八章 modaris-v5.pdf
- 第八章 MTK手机基带电路工作原理.ppt
- 第八章 netEasy自动化系统网络.pdf
- 第八章 -Origin绘图.第2讲.ppt
- 第八章 OS05存储管理.ppt
- 第八章 RSM数据采集系统建立.pdf
- 第八章 ssh&scp.pdf
- 第八章 Talend使用步骤.pdf
- 2024高考物理一轮复习规范演练7共点力的平衡含解析新人教版.doc
- 高中语文第5课苏轼词两首学案3新人教版必修4.doc
- 2024_2025学年高中英语课时分层作业9Unit3LifeinthefutureSectionⅢⅣ含解析新人教版必修5.doc
- 2024_2025学年新教材高中英语模块素养检测含解析译林版必修第一册.doc
- 2024_2025学年新教材高中英语单元综合检测5含解析外研版选择性必修第一册.doc
- 2024高考政治一轮复习第1单元生活与消费第三课多彩的消费练习含解析新人教版必修1.doc
- 2024_2025学年新教材高中英语WELCOMEUNITSectionⅡReadingandThi.doc
- 2024_2025学年高中历史专题九当今世界政治格局的多极化趋势测评含解析人民版必修1.docx
- 2024高考生物一轮复习第9单元生物与环境第29讲生态系统的结构和功能教案.docx
- 2024_2025学年新教材高中英语UNIT5LANGUAGESAROUNDTHEWORLDSect.doc
文档评论(0)