java 并发模型总类_java并发编程系列-内存模型基础

java线程之间的通信对程序开发人员是完全透明的,内存的可见性问题很容易困扰很多开发人员。本篇博文将揭开java内存模型的神秘面纱,来看看内存模型到底是怎样的。

并发编程模型的分类

并发编程中需要处理的两个关键问题:线程之间如何通信

线程之间如何同步

所谓通信是指线程之间以何种机制来交换信息,在命令式编程中,线程的通信机制有两种:共享内存(隐式通信:通过共享程序的公共状态,读-写内存中的公共状态实现)

消息传递(显示通信:线程间发送消息实现 ,比较典型的就是wait()和notify())

所谓同步,就是控制不同线程间操作发生相对顺序的机制:共享内存(同步是显示的,由程序开发人员显示的指定某段代码或者某个方法需要在线程之间互斥执行)。

消息传递(同步是隐式的,消息的发送必须在消息接收之前)。

java的并发采用的是共享内存模型,线程之间的通信是隐式执行的,同步需要开发人员显示进行控制。

JAVA内存模型(JMM)的抽象

JMM把java虚拟机内部划分为线程栈和堆。逻辑视图如下:

AAffA0nNPuCLAAAAAElFTkSuQmCC

JMM逻辑视图.png

java中所有的实例域、静态域,数组元素都是存储在堆内存,堆内存在线程之间共享。而对象引用,局部变量、方法参数和异常处理器参数都是存在在栈内存,也就是线程栈中,线程栈中的变量仅对自己可见,对其他线程不可见。不同线程之间的通信由java内存模型(java memory model ,简称JMM)控制。JMM的抽象结构图,如下:

AAffA0nNPuCLAAAAAElFTkSuQmCC

JMM内存模型抽象图.png

线程之间的共享变量存储在堆内存,每个线程都有私有的本地内存(线程栈),私有本地内存中存储了主内存中共享变量的拷贝,本地内存只是JMM的一个抽象概念,并不真实存在。

上图中线程A要与线程B通信的话,由于线程本地变量的不可见性,首先要将线程A中变量的更改,刷新到主内存中,然后线程B本地私有的共享变量副本失效,从新读取刷新的新值,才能完成。从上面的描述看,线程A向线程B通信,必须要经过主内存,JMM控制主内存与每个线程的本地变量的交互,来为java程序员提供内存的可见性。

硬件内存架构

软件最终还要运行在硬件上,看一下现代计算机硬件内存架构的简单图示:

AAffA0nNPuCLAAAAAElFTkSuQmCC

硬件模型.png

现在的计算机一般都有两个或者多个CPU,其中有些还是多核心实现。

每个CPU都包含一系列的寄存器,它们是CPU内内存的基础。CPU在寄存器上执行操作的速度远大于在主存上执行的速度。这是因为CPU访问寄存器的速度远大于主存。

每个CPU可能还有一个CPU缓存层。实际上,绝大多数的现代CPU都有一定大小的缓存层。CPU访问缓存层的速度快于访问主存的速度,但通常比访问内部寄存器的速度还要慢一点。一些CPU还有多层缓存,但这些对理解Java内存模型如何和内存交互不是那么重要。只要知道CPU中可以有一个缓存层就可以了。

一个计算机还包含一个主存。所有的CPU都可以访问主存。主存通常比CPU中的缓存大得多。

CPU的高速缓存虽然解决了效率的问题,但是又带来了一个新的问题:数据一致性。当一个CPU需要读取主存时,它会将主存的部分读到CPU缓存中。它甚至可能将缓存中的部分内容读到它的内部寄存器中,然后在寄存器中执行操作,这样就不会使CPU直接与内存相连。当CPU需要将结果写回到主存中去时,它会将内部寄存器的值刷新到缓存中,然后在某个时间点将值刷新回主存。

Java内存模型和硬件内存架构之间的桥接

上面已经提到,Java内存模型与硬件内存架构之间存在差异。硬件内存架构没有区分线程栈和堆。对于硬件,所有的线程栈和堆都分布在主内中。部分线程栈和堆可能有时候会出现在CPU缓存中和CPU内部的寄存器中。如下图所示:

AAffA0nNPuCLAAAAAElFTkSuQmCC

image.png

Java内存模型的基础原理

从源代码到指令序列的重排序

在程序执行时,为了提高程序的执行性能,编译器和处理器常常会对指令做重排序,换句话说程序的执行顺序和程序开发人员编写的顺序可能会存在差异,这是编译器和处理器对源代码做了优化。但是JMM的编译器重排序规则会禁止特定类型的编译器重排序,对于处理器的重排序,JMM的处理器重排序规则会要求java编译器在生成指令时,插入特定的内存屏障(Memory Barriers)指令,来禁止特定类型的处理器重排序。换句话说编译器和处理器的重排序都是可控的。

重排序分为三类:编译器重排序:编译器在不改变单线程程序语义的前提下,可以重新安排语句的执行顺序。

指令级重排序:现在的处理器都采用了并行技术,可以将多条执行重叠执行,如果不存在数据依赖性,可以改变语句对应机器语句的执行顺序。之所以存在数据依赖的语句不做重排序是因为改变顺序后将导致执行结果发生变化。

内存系统重排序:由于处理器使用缓存和读/写缓冲区,这使得加载和存储操作看上去可能是在乱序执行。

JMM属于语言级的内存模型,它确保在不同的编译器和不同的处理器平台上,通过禁止特定类型的编译器和处理器重排序,为程序员提供一致性的内存可见性保证。

重排序与内存屏障

为了保证内存可见性,Java编译器在生成指令序列的适当位置会插入内存屏障指令来禁止特定类型的处理器重排序。内存屏障又称为内存栅栏,是一个CPU指令:保证特定的操作顺序

影响某些数据的内存可见性

例如: volatile关键字 就是通过内存屏障实现的。

happens-before

JSP-133(内存模型)使用happens-before来阐述操作之间的内存可见性。在JMM中如果一个操作执行结果需要对另一个操作可见,那么这两个操作之间必须要存在happens-before关系。两个操作具有happens-before关系,并不意味着前一个操作必须要在后一个操作之间执行,happens-before仅仅要求前一个操作的执行结果对后一个操作可见。且前一个操作按顺序排在第二个操作的前面。

happens-before规则如下:程序顺序规则:一个线程中的每个操作,happens-before于线程中任意后续操作

监视器锁规则:对一个锁的解锁,happens-before于随后对这个锁的加锁

volatil变量规则:对一个volatile域的写,happens-before与任意后续对这个volatile域的读

传递性:如果A happens-before B ,且B happens-before C , 那么A happens-before C .

重排序

数据依赖

如果两个操作访问同一个变量,且这两个操作中有一个为写操作,此时这两个操作之间就存在数据的依赖性。写后读 a=1;b=a

写后写 a=1 ; a =2

读后写 a=b ; b =1

上面三种情况,只要重排序两个操作的执行顺序,程序的执行结果就会发生改变。编译器和处理器可能会对操作做重排序。做重排序时,会遵守数据依赖性,编译器和处理器不会改变存在数据依赖关系的两个操作的执行顺序,只针对单个处理器和单个线程而言。

as-if-serial语义

它的意思是不管怎么重排序,单线程执行的结果都不会发生变化。为了遵守as-if-serial语义,编译器和处理器都不会对存在数据依赖的语句执行重排序。

顺序一致性

数据竞争与顺序一致性保证

当程序未正确同步时,就可能存在数据竞争。在一个线程中写一个变量

在另一个线程中读同一个变量

而且读写没有通过同步来排序

顺序一致性内存模型

两大特性:一个线程中所有操作必须按照程序的顺序来执行

所有线程都只能看到一个单一的操作执行顺序。在顺序一致性内存模型中,每个操作都必须原子执行且立刻对所有线程可见。

同步程序的顺序一致性效果

同步程序的顺序一致性效果将于程序在顺序一致性模型中的执行结果相同。

未同步程序的执行特性

对于未同步或未正确同步的多线程程序,JMM只提供最小安全性。JMM不保证未同步的程序的执行结果与该程序在顺序一致性模型中的执行结果一致。

作者:起个名忒难

链接:https://www.jianshu.com/p/de47a2e49e5d

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/260322.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ATS读小文件(内存命中)

一个资源根据其大小可能会存在多个存储对象中。如果足够小(连同doc结构的大小小于一个fragment的size),连同这个资源的meta信息一起存储在一个doc中。如果比较大,第一个存储对象保存资源的meta信息,后面跟着若干个frag…

fatal error C1902: 程序数据库管理器不匹配;请检查安装解决

终于找到原因了,原来是我安装的字体渲染,并且采用注册表的加载方式!改掉就好了!上天哪,这是怎么影响到的 卸载MacType程序后,进行尝试! VS2008 和 VS2010 又能用了! 我想求教育。。。…

[译]多线程网络服务模型

2019独角兽企业重金招聘Python工程师标准>>> 多线程网络服务模型 /*** 谨献给Yoyo** 原文出处&#xff1a;https://www.toptal.com/software/guide-to-multi-processing-network-server-models* author dogstar.huang <chanzonghuanggmail.com> 2016-04-02*/作…

java mvc引擎_SpringMvc+JavaConfig+Idea 搭建项目

1.介绍之前搭建SpringMvc项目要配置一系列的配置文件&#xff0c;比如web.xml,applicationContext.xml,dispatcher.xml。Spring 3.X之后推出了基于JavaConfig方式以及注解的形式的配置。在一定程度上简化了Spring项目的配置。近几年特别火的SpringBoot&#xff0c;大大的简化了…

极域电子书包课堂管理系统_【君莲微讯】君莲学校(小学部)开展电子书包第13共同体数学研讨活动...

借 助 媒 体 技 术丰 富 图 形 认 识君莲学校(小学部)开展电子书包共同体 数学研讨活动 2020年12月2日下午&#xff0c;君莲学校(小学部)开展了以“借助媒体技术 丰富图形认识”为主题的闵行区电子书包第13共同体的数学研讨活动。共同体学校教师代表、学校电子书包项目组主管朱…

python批量改动指定文件夹文件名称

这小样例仅仅要是说明用python怎么批量改动指定文件夹的文件名称&#xff1a; 记得要把脚本跟改动的文件放在同一个文件夹下 #encoding:utf-8 import os import sys files os.listdir(D:\\1) #路径能够自己for name in files:a os.path.splitext(name)if a[1] .txt: #txt能够…

Linux vmstat命令实战详解

vmstat命令是最常见的Linux/Unix监控工具&#xff0c;可以展现给定时间间隔的服务器的状态值,包括服务器的CPU使用率&#xff0c;内存使用&#xff0c;虚拟内存交换情况,IO读写情况。这个命令是我查看Linux/Unix最喜爱的命令&#xff0c;一个是Linux/Unix都支持&#xff0c;二是…

java 入门 博客园_javaweb入门

复习&#xff1a;css的常用样式&#xff1a;borderbackgroundpaddingmarginfloatposition 定位top left确定div在页面中的位置&#xff0c;这两个值可以为负数。cssdiv 布局方式cssdivtable 先由div划分大块儿&#xff0c;再由table进行整齐布局。下拉列表&#xff1a;层叠的布…

以ThreadStart方式实现多线程

3.1 使用ThreadStart委托 这里先以一个例子体现一下多线程带来的好处&#xff0c;首先在Message类中建立一个方法ShowMessage()&#xff0c;里面显示了当前运行线程的Id&#xff0c;并使用Thread.Sleep&#xff08;int ) 方法模拟部分工作。在main()中通过ThreadStart委托绑定M…

我的atom插件

atom插件实在是太多了&#xff0c;下面就说说我的插件 1.minimap 右边的小地图&#xff0c;和sublime里面的差不多&#xff1b; 2.open-in-browser 右击默认浏览器打开&#xff1b; 3.emmet 这个不用多说吧&#xff0c;html快速编译 4.git-plus 直接在atom提交代码&#xff0…

MonoRail - 简介 [基础知识篇]

MonoRail - 简介 起源 MonoRail是一个.NET的MVC web开发框架, 原名Castle On Rails, 是CastleProject的一个子项目. 作者hammett在使用过Ruby On Rails后, 觉得非常棒, 他希望在享受ror的开发模式的同时能使用大量现有的资源, 于是就用.NET写出了一个Castle On Rails. 后来ror那…

结对编程(黄金点游戏)

我扮演的角色是驾驶员 一、结对伙伴 领航员&#xff1a;赵峻 作业地址见我的博客。 二、代码地址 https://coding.net/u/k2048/p/huangjindian/git/blob/master/main.c 三、总结 1、个人总结 本次作业我扮演驾驶员&#xff0c;赵峻扮演领航员&#xff0c;我负责算法实现以及代码…

qtgl 鼠标平移 c++_罗技真爱粉的MX Master 3无线鼠标体验

​这是一篇关于罗技MX Master3的晒单&#xff0c;顺带也翻出我的库存清洁整理一下吧。在决定购买一款新鼠标的时候&#xff0c;我的第一目标其实是MX Vertical垂直鼠标&#xff0c;不过MX Vertical目前优势只在外形上&#xff0c;在MX系列中明显属于低配&#xff0c;自由滚轮、…

下载网页中的图片到本地

简单的一个下载如下 &#xff1a; string url "http://avatar.csdn.net/A/2/6/2_yefengzhixia.jpg";string filepath "D:\\pic.jpg";WebClient mywebclient new WebClient();mywebclient.DownloadFile(url, filepath);MessageBox.Show("OK");…

nacos linux启动_微服务系列之Nacos配置中心之一:Nacos介绍与安装

一、Nacos 介绍Nacos 是 Alibaba 公司推出的开源工具&#xff0c;用于实现分布式系统的服务发现与配置管理。英文全称 Dynamic Naming and Configuration Service&#xff0c;Na 为 Naming/NameServer 即注册中心&#xff0c;co 为 Configuration 即配置中心&#xff0c;Servic…

演练:在组件设计器中创建 Windows 服务应用程序

http://msdn.microsoft.com/zh-cn/library/zt39148a(vvs.80).aspx 演练&#xff1a;在组件设计器中创建 Windows 服务应用程序 .NET Framework 2.0其他版本5&#xff08;共 5&#xff09;对本文的评价是有帮助 - 评价此主题注意 Visual Studio 标准版中不提供“Windows 服务”模…

绝对震撼 7款HTML5动画应用及源码

除非特别声明&#xff0c;PHP100新闻均为原创或投稿报道&#xff0c;转载请注明作者及原文链接 原文地址&#xff1a; http://www.php100.com/html/it/mobile/2014/0702/7030.html [导读] 关于HTML5动画&#xff0c;我们已经分享太多了&#xff0c;当然也有很多利用纯CSS3实现的…

maven打包指定main函数

2019独角兽企业重金招聘Python工程师标准>>> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-jar-plugin</artifactId> <version>2.4</version> <configuration> <archive> …

简易贪吃蛇小游戏java版_用GUI实现java版贪吃蛇小游戏

本文实例为大家分享了java版贪吃蛇小游戏的具体代码&#xff0c;供大家参考&#xff0c;具体内容如下项目结构新建一个JFrame窗口,作为程序入口public class GameStart{public static void main(String[] args) {JFrame jFrame new JFrame();jFrame.setBounds(100,100,900,720…

java切入式编程显示屏_C语言嵌入式系统编程修炼之四:屏幕操作

C语言嵌入式系统编程修炼之四:屏幕操作作者:宋宝华 更新日期:2005-07-22汉字处理现在要解决的问题是&#xff0c;嵌入式系统中经常要使用的并非是完整的汉字库&#xff0c;往往只是需要提供数量有限的汉字供必要的显示功能。例如&#xff0c;一个微波炉的LCD上没有必要提供显…