并发编程学习(一)Java 内存模型

Java 内存模型

什么是 Java 内存模型(Java Memory Model)

Java 内存模型是一种规范,规范了 Java 虚拟机和计算机内存是如何进行协同工作的。

JMM 规定了一个线程如何和何时看到其它线程修改过后的的共享变量的值,以及在必须时如何同步的访问共享变量。

主内存、高速缓存、寄存器

CPU 包含一系列的寄存器,它们是 CPU 内存的基础。

CPU 在寄存器上操作的速度,远大于在主存上读写的速度。是因为 CPU 对寄存器的运行速度远大于主存。

那如何解决这个问题呢?需要引入高速缓存

计算机的存储设备与处理器的运算速度之间有着几个数量级的差距,所以现在的计算机系统都需要加一层读写速度尽可能和处理器运算速度接近的高速缓存,来作为内存和处理器之间的缓冲。

将运算所需要的数据复制到缓存中。让运算能快速的进行。当运算结束后,再把数据从缓存同步到内存之中。

这样处理器就不需要等待内存中的读写。

线程和主内存的抽象关系

线程之间的共享变量存储在主内存中,每个线程都有一个私有的本地内存,也可以叫做工作内存,本地内存是 Java 内存模型的一个抽象的概念,它并不是真实存在的,它涵盖了高速缓存、写缓冲区、寄存器、其它硬件和编译器优化。本地内存中存储以读或写共享变量的一个拷贝的副本。

从更底层来说,主内存就是硬件的内存,是为了获取更好的运行速度,虚拟机及硬件系统可能会让工作内存优先存储与寄存器和高速缓存中。

Java 内存模型的工作内存是寄存器和高速缓存的一个抽象描述。 
** 
JVM 内存模型是一种对内存的物理划分,只局限在内存,而且也只局限于 JVM 的内存。

如果线程 A 和线程 B 获取主内存的共享变量进行更新操作,就会出现线程不安全的问题。

那就需要增加同步的手段来保证并发时结果的准确性。

同步的八种操作

  • lock(锁定):作用于主内存的变量,把一个变量标识为一条线程独占状态。
  • unlock(解锁):作用于主内存的变量,把一个处于锁定状态的变量释放出来,释放后的变量才可以被其他线程锁定。
  • read(读取):作用于主内存的变量,把一个变量值从主内存传输到线程的工作内存中,以便随后的 load 动作使用。
  • load(载入):作用于工作内存的变量,把 read 操作从主内存中得到的变量值放入工作内存的变量副本中。
  • use(使用):作用于工作内存的变量,把工作内存中的一个变量值传递给执行引擎。
  • assign(赋值):作用于工作内存的变量,把一个从执行引擎接收到的值赋值给工作内存的变量。
  • store(存储):作用于工作内存的变量,把工作内存中的一个变量的值传送到主内存中,以便随后的 write 的操作。
  • write(写入): 作用于主内存的变量,把 store 操作从工作内存中一个变量的值传送到主内存的变量中。

同步规则

  1. 如果要把一个变量从主内存复制到工作内存,就需要按照顺序地执行 read 和 load 操作。如果把变量从工作内存中同步回主内存中,就要按顺序地执行 store 和 write 操作。但 Java 内存模型只要求上述操作必须按顺序执行,而没有保证必须是连续执行。(read 和 load 之间,store 和 write 之间是可以插入其它操作的。)
  2. 不允许 read 和 load、store 和 write 操作之一单独出现。
  3. 不允许一个线程丢弃它的最近 assign 操作,即变量在工作内存中改变了之后必须同步到主内存中。
  4. 不允许一个线程无原因地(没有发生过任何 assign 操作)把数据从工作内存同步回主内存中。
  5. 一个新的变量只能在主内存中诞生,不允许在工作内存中直接使用一个未被初始化(load 或 assign)的变量。即就是对一个变量实施 usd 和 store 之前,必须先执行过了 assign 和 load 操作。
  6. 一个变量在同一时刻只允许一条线程对其进行 lock 操作,但 lock 操作可以被同一线程重复执行好多次,多次执行 lock 后,只有执行相同次数的 unlock 操作,变量才会被解锁。lock 和 unlock 必须成对出现。
  7. 如果对一个变量执行 lock 操作,将会清空工作内存中此变量的值,在执行引擎使用这个变量前需要重新执行 load 或 assign 操作初始化变量的值。
  8. 如果一个变量实现没有被 lock 操作锁定,则不允许对它执行 unlock 操作;也不允许去 unlock 一个被其他线程锁定的变量。
  9. 对一个变量执行 unlock 操作之前,必须先把此变量同步到主内存中(执行 store 和 write 操作)。

这些同步操作和对应的基本规则是 Java 中并发相关的类在设计时都必须准守的。

这里必须注意的是,Java 内存模型还有特殊的规则,我们实际用到的时候再进行介绍。比如 final 修饰的值不允许修改,类似于这种的。

总结

我们回顾一下前面所讲的内容,Java 内存模型是一个规范,规定了一个线程如何和何时可以看到由其它线程修改过后的共享变量的值,以及在必须时如何同步的访问共享变量。

线程之间的通信必须经过主内存。同时 Java 内存模型定义了同步的八个操作及基本规则。这个为我们处理并发问题提供了理论基础。

Java 内存模型

什么是 Java 内存模型(Java Memory Model)

Java 内存模型是一种规范,规范了 Java 虚拟机和计算机内存是如何进行协同工作的。

JMM 规定了一个线程如何和何时看到其它线程修改过后的的共享变量的值,以及在必须时如何同步的访问共享变量。

主内存、高速缓存、寄存器

CPU 包含一系列的寄存器,它们是 CPU 内存的基础。

CPU 在寄存器上操作的速度,远大于在主存上读写的速度。是因为 CPU 对寄存器的运行速度远大于主存。

那如何解决这个问题呢?需要引入高速缓存

计算机的存储设备与处理器的运算速度之间有着几个数量级的差距,所以现在的计算机系统都需要加一层读写速度尽可能和处理器运算速度接近的高速缓存,来作为内存和处理器之间的缓冲。

将运算所需要的数据复制到缓存中。让运算能快速的进行。当运算结束后,再把数据从缓存同步到内存之中。

这样处理器就不需要等待内存中的读写。

线程和主内存的抽象关系

线程之间的共享变量存储在主内存中,每个线程都有一个私有的本地内存,也可以叫做工作内存,本地内存是 Java 内存模型的一个抽象的概念,它并不是真实存在的,它涵盖了高速缓存、写缓冲区、寄存器、其它硬件和编译器优化。本地内存中存储以读或写共享变量的一个拷贝的副本。

从更底层来说,主内存就是硬件的内存,是为了获取更好的运行速度,虚拟机及硬件系统可能会让工作内存优先存储与寄存器和高速缓存中。

Java 内存模型的工作内存是寄存器和高速缓存的一个抽象描述。

**

JVM 内存模型是一种对内存的物理划分,只局限在内存,而且也只局限于 JVM 的内存。

如果线程 A 和线程 B 获取主内存的共享变量进行更新操作, 就会出现线程不安全的问题 。

那就需要增加同步的手段来保证并发时结果的准确性。

同步的八种操作

  • lock(锁定):作用于主内存的变量,把一个变量标识为一条线程独占状态。
  • unlock(解锁):作用于主内存的变量,把一个处于锁定状态的变量释放出来,释放后的变量才可以被其他线程锁定。
  • read(读取):作用于主内存的变量,把一个变量值从主内存传输到线程的工作内存中,以便随后的 load 动作使用。
  • load(载入):作用于工作内存的变量,把 read 操作从主内存中得到的变量值放入工作内存的变量副本中。
  • use(使用):作用于工作内存的变量,把工作内存中的一个变量值传递给执行引擎。
  • assign(赋值):作用于工作内存的变量,把一个从执行引擎接收到的值赋值给工作内存的变量。
  • store(存储):作用于工作内存的变量,把工作内存中的一个变量的值传送到主内存中,以便随后的 write 的操作。
  • write(写入): 作用于主内存的变量,把 store 操作从工作内存中一个变量的值传送到主内存的变量中。

同步规则

  1. 如果要把一个变量从主内存复制到工作内存,就需要按照顺序地执行 read 和 load 操作。如果把变量从工作内存中同步回主内存中,就要按顺序地执行 store 和 write 操作。但 Java 内存模型只要求上述操作必须按顺序执行,而 没有保证必须是连续执行 。( read 和 load 之间,store 和 write 之间是可以插入其它操作的。 )
  2. 不允许 read 和 load、store 和 write 操作之一单独出现。
  3. 不允许一个线程丢弃它的最近 assign 操作,即变量在工作内存中改变了之后必须同步到主内存中。
  4. 不允许一个线程无原因地(没有发生过任何 assign 操作)把数据从工作内存同步回主内存中。
  5. 一个新的变量只能在主内存中诞生,不允许在工作内存中直接使用一个未被初始化(load 或 assign)的变量。即就是对一个变量实施 usd 和 store 之前,必须先执行过了 assign 和 load 操作。
  6. 一个变量在同一时刻只允许一条线程对其进行 lock 操作,但 lock 操作可以被同一线程重复执行好多次,多次执行 lock 后,只有执行相同次数的 unlock 操作,变量才会被解锁。lock 和 unlock 必须成对出现。
  7. 如果对一个变量执行 lock 操作,将会清空工作内存中此变量的值,在执行引擎使用这个变量前需要重新执行 load 或 assign 操作初始化变量的值。
  8. 如果一个变量实现没有被 lock 操作锁定,则不允许对它执行 unlock 操作;也不允许去 unlock 一个被其他线程锁定的变量。
  9. 对一个变量执行 unlock 操作之前,必须先把此变量同步到主内存中(执行 store 和 write 操作)。

这些同步操作和对应的基本规则是 Java 中并发相关的类在设计时都必须准守的。

这里必须注意的是,Java 内存模型还有特殊的规则,我们实际用到的时候再进行介绍。比如 final 修饰的值不允许修改,类似于这种的。

总结

我们回顾一下前面所讲的内容,Java 内存模型是一个规范,规定了一个线程如何和何时可以看到由其它线程修改过后的共享变量的值,以及在必须时如何同步的访问共享变量。

线程之间的通信必须经过主内存。同时 Java 内存模型定义了同步的八个操作及基本规则。这个为我们处理并发问题提供了理论基础。

原文地址:https://www.cnblogs.com/Java-JJ/p/12622449.html

时间: 2024-10-15 01:23:07

并发编程学习(一)Java 内存模型的相关文章

JAVA并发编程2_线程安全&内存模型

"你永远都不知道一个线程何时在运行!" 在上一篇博客JAVA并发编程1_多线程的实现方式中后面看到多线程中程序运行结果往往不确定,和我们预期结果不一致.这就是线程的不安全.线程的安全性是非常复杂的,没有任何同步的情况下,多线程的执行顺序是不可预测的.当多个线程访问同一个资源时就会出现线程安全问题.例如有一个银行账户,一个线程往里面打钱,一个线程取钱,要是得到不确定的结果那是多么可怕的事情. 引入: 例如下面的程序,在单线程下,执行两次i++理论上i的最终值是12,但是在多线程环境下则不

JVM学习记录-Java内存模型(二)

对于volatile型变量的特殊规则 关键字volatile可以说是Java虚拟机提供的最轻量级的同步机制. 在处理多线程数据竞争问题时,不仅仅是可以使用synchronized关键字来实现,使用volatile也可以实现. Java内存模型对volatitle专门定义了一些特殊的访问规则,当一个变量被定义为volatile时,它将具备以下两个特性: 第一个是保证此变量对所有线程的可见性,这里的“可见性”是指当一条线程修改了这个变量的值,新值对于其他线程来说是可以立即得知的.而普通变量不能做到这

深入理解Java虚拟机- 学习笔记 - Java内存模型与线程

除了在硬件上增加告诉缓存之外,为了使得处理器内部的运算单元能尽量被充分利用,处理器可能会对输入代码进行乱序执行(Out-Of-Order Execution)优化,处理器会在计算之后将乱序执行的结果重组,保证该结果与顺序执行的结果一致,但并不保证程序中各个语句计算的先后顺序与输入代码中的顺序一致,因此,如果存在一个计算任务依赖另外一个计算任务的中间结果,那么其顺序性并不能靠代码的先后顺序来保证.与处理器的乱序优化执行类似,Java虚拟机的即时编译器中也有类似的指令重排序(Instruction

Java内存模型---并发编程网 - ifeve.com

Java内存模型 转自:http://ifeve.com/java-memory-model-6/ 原文地址  作者:Jakob Jenkov 译者:张坤 Java内存模型规范了Java虚拟机与计算机内存是如何协同工作的.Java虚拟机是一个完整的计算机的一个模型,因此这个模型自然也包含一个内存模型——又称为Java内存模型. 如果你想设计表现良好的并发程序,理解Java内存模型是非常重要的.Java内存模型规定了如何和何时可以看到由其他线程修改过后的共享变量的值,以及在必须时如何同步的访问共享

Java并发编程(四)-- Java内存模型

Java内存模型 前面讲到了Java线程之间的通信采用的是共享内存模型,这里提到的共享内存模型指的就是Java内存模型(简称JMM),JMM决定一个线程对共享变量的写入何时对另一个线程可见.从抽象的角度来看,JMM定义了线程和主内存之间的抽象关系:线程之间的共享变量存储在主内存(main memory)中,每个线程都有一个私有的本地内存(local memory),本地内存中存储了该线程以读/写共享变量的副本.本地内存是JMM的一个抽象概念,并不真实存在.它涵盖了缓存,写缓冲区,寄存器以及其他的

史上最全的并发编程学习目录

目录 一:线程基础知识 1.并发编程的基本概念 2. 线程的基本操作 3.线程之间的通信wait和notify 4.join和yield以及sleep详解 5. synchronized关键字讲解 6. volatile原理 7. 线程组 8.线程优先级 9.守护线程 10.ThreadLocal 二:JDK并发包 1.jdk并发工具类 2.jdk并发容器 3.jdk并发框架 三:Java内存模型 Java内存模型的抽象结构 深入理解synchronized关键字 四:线程池 线程池 Futur

Java多线程-Java内存模型

以下内容转自http://ifeve.com/java-memory-model-6/: Java内存模型规范了Java虚拟机与计算机内存是如何协同工作的.Java虚拟机是一个完整的计算机的一个模型,因此这个模型自然也包含一个内存模型——又称为Java内存模型. 如果你想设计表现良好的并发程序,理解Java内存模型是非常重要的.Java内存模型规定了如何和何时可以看到由其他线程修改过后的共享变量的值,以及在必须时如何同步的访问共享变量. 原始的Java内存模型存在一些不足,因此Java内存模型在

Java内存模型深度解读

Java内存模型规范了Java虚拟机与计算机内存是如何协同工作的.Java虚拟机是一个完整的计算机的一个模型,因此这个模型自然也包含一个内存模型——又称为Java内存模型. 如果你想设计表现良好的并发程序,理解Java内存模型是非常重要的.Java内存模型规定了如何和何时可以看到由其他线程修改过后的共享变量的值,以及在必须时如何同步的访问共享变量. 原始的Java内存模型存在一些不足,因此Java内存模型在Java1.5时被重新修订.这个版本的Java内存模型在Java8中人在使用. Java内

并发编程-Java内存模型到底是什么

内存模型 在计算机CPU,内存,IO三者之间速度差异,为了提高系统性能,对这三者速度进行平衡. CPU 增加了缓存,以均衡与内存的速度差异: 操作系统增加了进程.线程,以分时复用 CPU,进而均衡 CPU 与 I/O 设备的速度差异: 编译程序优化指令执行次序,使得缓存能够得到更加合理地利用. 以上三种系统优化,对于硬件的效率有了显著的提升,但是他们同时也带来了可见性,原子性以及顺序性等问题.基于Cpu高速缓存的存储交互很好得解决了CPU和内存得速度矛盾,但是也提高了计算机系统得复杂度,引入了新