Java并发(七)之 内存模型

16人阅读 评论(0) 收藏 举报
分类:

概要

  • Java 内存模型视图屏蔽各种硬件和操作系统的内存访问差异,以实现让 Java 程序在各种平台下都能达到一致的内存访问效果。

主内存与工作内存

  • 处理器上的寄存器的读写的速度比内存快几个数量级,为了解决这种速度矛盾,在它们之间加入了高速缓存。
  • 加入高速缓存带来了一个新的问题:缓存一致性。如果多个缓存共享同一块主内存区域,那么多个缓存的数据可能会不一致。CPU 使用一致性协议来解决一致性问题。
    这里写图片描述
  • 所有的变量都存储在主内存中,每个线程还有自己的工作内存,工作内存存储在高速缓存或者寄存器中,保存了该线程使用的变量的主内存副本拷贝。
  • 线程只能直接操作工作内存中的变量,不同线程之间的变量值传递需要通过主内存来完成。
    这里写图片描述

内存间交互操作

  • Java 内存模型定义了 8 个操作来完成主内存和工作内存的交互操作。
    这里写图片描述
  • read:把一个变量的值从主内存传输到工作内存中
  • load:在 read 之后执行,把 read 得到的值放入工作内存的变量副本中
  • use:把工作内存中一个变量的值传递给执行引擎
  • assign:把一个从执行引擎接收到的值赋给工作内存的变量
  • store:把工作内存的一个变量的值传送到主内存中
  • write:在 store 之后执行,把 store 得到的值放入主内存的变量中
  • lock:作用于主内存的变量
  • unlock

内存模型三大特性

1. 原子性

  • Java 内存模型保证了 read、load、use、assign、store、write、lock 和 unlock
    操作具有原子性,例如对一个 int 类型的变量执行 assign 赋值操作,这个操作就是原子性的。但是 Java
    内存模型允许虚拟机将没有被 volatile 修饰的 64 位数据(long,double)的读写操作划分为两次 32 位的操作来进行,即
    load、store、read 和 write 操作可以不具备原子性。
  • 有一个错误认识就是,int 等原子性的变量在多线程环境中不会出现线程安全问题。前面的线程不安全示例代码中,cnt 变量属于 int 类型变量,1000 个线程对它进行自增操作之后,得到的值为 997 而不是 1000。

2. 可见性

  • 可见性指当一个线程修改了共享变量的值,其它线程能够立即得知这个修改。Java内存模型是通过在变量修改后将新值同步回主内存,在变量读取前从主内存刷新变量值来实现可见性的。
  • volatile 可保证可见性。synchronized 也能够保证可见性,对一个变量执行 unlock 操作之前,必须把变量值同步回主内存。final 关键字也能保证可见性:被 final 关键字修饰的字段在构造器中一旦初始化完成,并且没有发生 this 逃逸(其它线程可以通过 this 引用访问到初始化了一般的对象),那么其它线程就能看见 final 字段的值。
  • 对前面的线程不安全示例中的 cnt 变量用 volatile 修饰,不能解决线程不安全问题。因为 volatile 并不能保证操作的原子性。

3. 有序性

  • 有序性是指:在本线程内观察,所有操作都是有序的。在一个线程观察另一个线程,所有操作都是无序的,无序是因为发生了指令重排序。
  • 在 Java 内存模型中,允许编译器和处理器对指令进行重排序,重排序过程不会影响到单线程程序的执行,却会影响到多线程并发执行的正确性。
  • volatile 关键字通过添加内存屏障的方式来禁止指令重排,即重排序时不能把后面的指令放到内存屏障之前。
  • 也可以通过 synchronized 来保证有序性,它保证每个时刻只有一个线程执行同步代码,相当于是让线程顺序执行同步代码。
查看评论

JVM之内存构成(二)--JAVA内存模型与并发

物理机中的并发硬件效率与一致性 Java线程执行的内存模型 工作内存 主内存 内存间交互 long和double的非源自性协定 Volatile类型变量的特殊规则和语义 保证可见性 禁止指令重排优化 ...
  • qq_33938256
  • qq_33938256
  • 2016-09-19 13:05:29
  • 1059

【死磕Java并发】-----Java内存模型之总结

经过四篇博客阐述,我相信各位对Java内存模型有了最基本认识了,下面LZ就做一个比较简单的总结。总结JMM规定了线程的工作内存和主内存的交互关系,以及线程之间的可见性和程序的执行顺序。一方面,要为程序...
  • chenssy
  • chenssy
  • 2017-03-03 12:58:24
  • 4960

根据java内存模型理解并发出现的问题

原子性1、某些读写共享变量的操作如果不是原子操作,多线程并发的情况下会出现并发问题。 2、原子性实现了多个线程并发访问某段代码的时候,使这些线程能够有序访问。因为实现原子操作代码的一旦被执行,就不能...
  • bh_xiaoxinba
  • bh_xiaoxinba
  • 2016-09-12 22:21:27
  • 249

【死磕Java并发】-----Java内存模型之重排序

在执行程序时,为了提供性能,处理器和编译器常常会对指令进行重排序,但是不能随意重排序,不是你想怎么排序就怎么排序,它需要满足以下两个条件: 1. 在单线程环境下不能改变程序运行的结果; 2. 存在...
  • chenssy
  • chenssy
  • 2017-02-20 08:44:36
  • 4010

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

”你永远都不知道一个线程何时在运行!“ 在上一篇博客JAVA并发编程1_多线程的实现方式中后面看到多线程中程序运行结果往往不确定,和我们预期结果不一致。这就是线程的不安全。线程的安全性是非常复杂的,...
  • cauchyweierstrass
  • cauchyweierstrass
  • 2015-05-15 23:16:42
  • 2623

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

此前我们讲到了线程、同步以及volatile关键字,对于Java的并发编程我们有必要了解下Java的内存模型,因为Java线程之间的通信对于工程师来言是完全透明的,内存可见性问题很容易使工程师们觉得困...
  • itachi85
  • itachi85
  • 2016-05-29 10:39:11
  • 6509

并发基础二:Java内存模型

1. Java内存模型 1. 定义 Java内存模型规定所有变量都存储在主内存中(Main memory),每个线程还有自己的工作内存(working memory)。 线程的工作内存中保存...
  • huanglu20125
  • huanglu20125
  • 2018-02-24 16:11:30
  • 41

《Java并发编程实战》第十六章 Java内存模型 读书笔记

Java内存模型是保障多线程安全的根基,这里仅仅是认识型的理解总结并未深入研究。 一、什么是内存模型,为什么需要它 Java内存模型(Java Memory Model)并发相关的安全...
  • love_world_
  • love_world_
  • 2014-06-05 07:58:40
  • 1595

《Java并发编程的艺术》读书笔记——Java内存模型

第三章 Java内存模型3.1 内存模型基础3.1.1 并发编程的两个关键问题 线程之间如何通信 java采用共享内存模型隐式通信 线程之间如何同步 共享内存模型模型需要显式指定同步 3.1.2 ...
  • shuxiangxingkong
  • shuxiangxingkong
  • 2016-09-13 10:49:17
  • 501

【死磕Java并发】-----Java内存模型之分析volatile

前篇博客【死磕Java并发】—–深入分析volatile的实现原理 中已经阐述了volatile的特性了: volatile可见性;对一个volatile的读,总可以看到对这个变量最终的写; vola...
  • chenssy
  • chenssy
  • 2017-02-23 20:27:42
  • 3345
    个人资料
    专栏达人 持之以恒
    等级:
    访问量: 1万+
    积分: 1793
    排名: 2万+
    博客专栏
    最新评论