并发编程|第一篇:Java内存模型

1.进程与线程区别

  • 进程

    操作系统中运行的exe程序即可理解为进程,如电脑中独立运行的QQ、WPS等应用程序,一个进程可包含多个线程

  • 线程

    线程为进程中独立运行的执行单元,可理解为QQ中的传文件、开视频、聊天等均为QQ进程中的执行单元

2.Java内存区域

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kuvPUXrE-1570933481968)(C:\Users\zycao\AppData\Roaming\Typora\typora-user-images\1570717171714.png)]

区域用途说明
程序计数器线程执行时的行号指示器1.记录下一次要执行的字节码指令地址
2.唯一一个不存在内存溢出区域
3.线程私有
4.执行Native方法时,计数器值为空(undefined)
虚拟机栈存放对象引用,局部变量表,方法出口等信息1.线程私有
2.存入StackOverflowErrorOutOfmemoryError异常
本地方法栈虚拟机执行native方法用到的区域1.线程私有
2.存入StackOverflowErrorOutOfmemoryError异常
主要用于创建实例对象区域,垃圾回收主要区域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定义了线程与主内存间的抽象关系如下:

    • 主内存:存入共享变量区域
    • 本地内存:每个线程私有,存入共享变量副本
      [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-c7xwzOrI-1572749850775)(C:\Users\zycao\AppData\Roaming\Typora\typora-user-images\1572746139405.png)]
  • 线程模型中数据通信

  1. 线程A要修改主内存中变量X的值,线程A将主内存中变量X拷贝至本地内存中

  2. 线程A修改本地内存中变量X->X1

  3. 线程A将本地内存中修改后的值 X1刷新至主内存中

  4. 线程B从主内存中读取线程A刷新至主内存中的变量X1

私有信息、基本数据类型直接分配至工作内存中,引用数据类型地址存放在工作内存中,对象存放至堆内存中

  • java内存区域与java内存模型区别与联系

    1) java内存区域: 从物理上对java内存进行划分,分为堆、栈、方法区,java内存区域解决了不同类型数据存储问题

    2) java内存模型: 从概念上对java内存进行划分,分为主内存、本地内存,jmm解决了不同线程间数据通信问题

4.硬件内存架构与Java内存模型

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-eq1KK8T8-1572749850777)(C:\Users\zycao\AppData\Roaming\Typora\typora-user-images\1572748095821.png)]

由于cpu处理速度非常快,而内存速度与cpu速度相关巨大,为了不影响内存运行的拖累,因此设计者们在cpu与内存间增加缓存,数据处理步骤如下:

  1. cpu先从寄存器中获取数据,如果获取到数据则直接使用,否则执行下一步

  2. cpu从缓存中获取数据,获取到则返回,否则继续执行下一步

  3. cpu从寄存器、缓存中均未获取到数据,则才从内存中获取数据

  • cpu缓存一致性问题
  1. 总线加锁:一次只允许一个cpu使用数据,减低了cpu的吞吐量

  2. 缓存一致性协议(MESI):当cpu在缓存中操作数据时,会将数据读取至寄存器中,并进行修改更新内存数据,同时将消息线(cacheLine) 置无效,其它cpu要操作该数据时,当读到该数据的cacheLine无效时,直接到内存中读取该数据,从而保证了数据一致性问题

5.Java线程与内核硬件关系

平时我们只关注创建线程并将线程交给线程池管理执行,而后续的线程池如何管理线程,线程如何执行,为何java创建的线程能在操作系统中执行?这些问题我们都没有关注,我们通过下图进行分析

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4FP05D5z-1572749850777)(C:\Users\zycao\AppData\Roaming\Typora\typora-user-images\1572749129067.png)]

  1. 当执行Java代码时,一段Java代码分多个线程,这些线程交给线程池管理

  2. 每个线程对应每个内核线程进行处理

  3. 内核线程由操作系统管理

  4. 操作系统进行cpu切换从而执行内核线程中代码逻辑,从而实现了java线程与内核间转换

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值