JMM 内存模型

前言

本文介绍java的内存模型

java并发编程中,需要处理两个关键的问题:线程间的同步和通信

一、什么是Java内存模型(JMM)

在Java虚拟机规范中试图定义一种Java内存模型(Java Memory Model,JMM)来屏蔽各个硬件平台和操作系统的内存访问差异,以实现让Java程序在各种平台下都能达到一致的内存访问效果。那么Java内存模型规定了哪些东西呢,它定义了程序中变量的访问规则,往大一点说是定义了程序执行的次序。(其实也就是在多线程情况下,对于共享可变变量,线程是如何通信的)

java内存模型抽象示意图

线程上的交互实际是线程A向线程B发消息,而这个通信需要经过主存,JMM通过控制主存与每个线程本地内存间的交互,来为java程序员提供内存可见性保证

二、指令重排序

程序在执行时,为了提高性能,编译器和处理器常会对指令进行重排序

会进行3种类型的重排序

通过内存屏障指令来禁止特定类型的处理器重排序。(内存屏障:就是一系列指令保证来控制程序执行的顺序)

内存屏障的类型

屏障类型指令解释
LoadLoad BarriersLoad1;LoadLoad;Load2确保Load1的数据的装载先于Load2及所有后续装载指令的装载
StoreStoreBarriersStore1;StoreStore;Store2确保Store1数据对其他处理器可见(刷新到内存)先于Store2及所有后续存储指令的存储
LoadStore BarriersLoad1;LoadStore;Store2确保Load1的数据的装载先于Store2及所有后续存储指令的存储
StoreLoad BarriersStore1;StoreLoad;Load2确保Store1的数据对其他处理器可见(刷新到内存)先于Load2及所有后续的装载指令的装载

StoreLoad Barriers是一个“全能型”的屏障,它同时具有其他3个屏障的效果

三、happens-before

JSR-133规范使用happens-before的概念来阐述操作之间的内存可见性。在JMM中,如果一个操作执行的结果要对另外一个操作可见,那么这两个操作必须存在happens-before关系。
基本规则:

  • 程序顺序规则: 一个线程中的每个操作,happens-before于该线程中的任意后续操作
  • 监视器锁规则:一个锁的解锁,happens-before于加锁
  • volatile: 一个volatile域的写,happens-before于volatile域的读
  • 传递性:如果A happens-before B ,B happens-before C 那么 A happens-before C

说明:两个操作直接具有happens-before关系,并不意味着前一个操作必须要在后一个操作前执行,只是要求前一个操作的的执行结果对后一个操作可见

四、小结

As-if-serial 语义保证单线程内程序的执行结果不被改变,hapens-before保证正确同步的多线程程序的执行结果不被改变.其二者的本质都是在不改变程序执行结果前提下,尽可能的提高程序的并行度。
参考资料

《JAVA 并发编程实战》
《JAVA并发编程的艺术》
《深入理解Java虚拟机》

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值