并发编程|第一篇:Java内存模型
1.进程与线程区别
-
进程
操作系统中运行的exe程序即可理解为进程,如电脑中独立运行的QQ、WPS等应用程序,一个进程可包含多个线程
-
线程
线程为进程中独立运行的执行单元,可理解为QQ中的传文件、开视频、聊天等均为QQ进程中的执行单元
2.Java内存区域
区域 | 用途 | 说明 |
---|---|---|
程序计数器 | 线程执行时的行号指示器 | 1.记录下一次要执行的字节码指令地址 2.唯一一个不存在内存溢出区域 3.线程私有 4.执行Native方法时,计数器值为空(undefined) |
虚拟机栈 | 存放对象引用,局部变量表,方法出口等信息 | 1.线程私有 2.存入 StackOverflowError 和OutOfmemoryError 异常 |
本地方法栈 | 虚拟机执行native方法用到的区域 | 1.线程私有 2.存入 StackOverflowError 和OutOfmemoryError 异常 |
堆 | 主要用于创建实例对象区域,垃圾回收主要区域 | 1.线程共有 2.会抛出 OutOfmemoryError
|
方法区 | 存放类信息、常量、静态变量、即时编译后的代码 | 1.包含运行时常量池 2.线程共有 3.会抛出 OutOfmemoryError
|
直接内存 | 直接操作native函数库的内存,其引用存放在jav堆中的DirectByteBuffer 对象中 |
1.会抛出OutOfmemoryError
|
3.Java内存模型
-
内存模型概念
我们都知道在同一进程中的子线程间是完全隔离的,互不影响,那么在多个线程间如何进行通信,即线程间数据如何共享,在命令式编程中线程间通信机制有两种:共享内存和消息传递
- 共享内存: 即线程间存在公共状态,假设存在两条线程,线程A和线程B,线程A将数据写入公共状态中,线程B从公共状态中获取线程A写入的数据,通过写-读公共状态数据从而达到隐式通信
- 消息传递: 多条线程间没有公共状态,必须通过发送消息来显式进行通信
java并发中采用的是共享内存模型,线程间通信都是隐式进行的,对程序员完全透明。
-
Java内存模型结构
Java线程间通信由Java内存模型(JMM)控制,JMM是一抽象概念,决定一个线程对共享变量的写入何时对另一线程可见。Jmm定义了线程与主内存间的抽象关系如下:
- 主内存:存入共享变量区域
- 本地内存:每个线程私有,存入共享变量副本
-
线程模型中数据通信
-
线程A要修改主内存中变量X的值,线程A将主内存中变量X拷贝至本地内存中
-
线程A修改本地内存中变量X->X1
-
线程A将本地内存中修改后的值 X1刷新至主内存中
-
线程B从主内存中读取线程A刷新至主内存中的变量X1
私有信息、基本数据类型直接分配至工作内存中,引用数据类型地址存放在工作内存中,对象存放至堆内存中
-
java内存区域与java内存模型区别与联系
1) java内存区域: 从物理上对java内存进行划分,分为堆、栈、方法区,java内存区域解决了不同类型数据存储问题
2) java内存模型: 从概念上对java内存进行划分,分为主内存、本地内存,jmm解决了不同线程间数据通信问题
4.硬件内存架构与Java内存模型
由于cpu处理速度非常快,而内存速度与cpu速度相关巨大,为了不影响内存运行的拖累,因此设计者们在cpu与内存间增加缓存,数据处理步骤如下:
-
cpu先从寄存器中获取数据,如果获取到数据则直接使用,否则执行下一步
-
cpu从缓存中获取数据,获取到则返回,否则继续执行下一步
-
cpu从寄存器、缓存中均未获取到数据,则才从内存中获取数据
- cpu缓存一致性问题
-
总线加锁:一次只允许一个cpu使用数据,减低了cpu的吞吐量
-
缓存一致性协议(MESI):当cpu在缓存中操作数据时,会将数据读取至寄存器中,并进行修改更新内存数据,同时将消息线(cacheLine) 置无效,其它cpu要操作该数据时,当读到该数据的cacheLine无效时,直接到内存中读取该数据,从而保证了数据一致性问题
5.Java线程与内核硬件关系
平时我们只关注创建线程并将线程交给线程池管理执行,而后续的线程池如何管理线程,线程如何执行,为何java创建的线程能在操作系统中执行?这些问题我们都没有关注,我们通过下图进行分析
-
当执行Java代码时,一段Java代码分多个线程,这些线程交给线程池管理
-
每个线程对应每个内核线程进行处理
-
内核线程由操作系统管理
-
操作系统进行cpu切换从而执行内核线程中代码逻辑,从而实现了java线程与内核间转换