第八章 深入理解Java内存模型.pdf

  1. 1、本文档共53页,可阅读全部内容。
  2. 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
  3. 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  4. 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

文档评论(0)

kehan123 + 关注
实名认证
内容提供者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档