Java内存模型与线程
本文主要内容
主内存与工作内存
volatile关键字
线程状态转换
与线程相关的内容,本博已经说过不少,本文着重阐述以前没有提及的内容
主内存与工作内存
在物理机上,“高效并发”并没有那么容易实现。因为任务不可能只依赖于处理器计算完成,至少与内存的交互很难消除。而内存设备与处理器的运算速度之间有着几个数据级的差距,所以现代计算机都添加高速缓存作为内存和处理器之间的缓冲。
内存将数据复制到缓存当中,当运算结束后再从缓存中同步回内存中。
值得一提的是,为了使处理器的运算单元被充分利用,处理器可能会对输入代码进行乱序执行,不过处理器会保证该结果与顺序执行的结果是一致的。
Java虚拟机为了屏蔽硬件差异,也设计了一套内存模型。
主内存:Java模型规定了所有的变量都存储在主内存中(与物理机中的内存对应,但仅是虚拟机内存的一部分)
工作内存:每条线程有自己的工作内存(可以物理机中的高速缓存类比),工作线程保存了变量的主内存副本拷贝
线程对变量的所有操作都必须在工作内存中进行,而不能直接读写主内存的变量,不同线程间也无法直接访问对方工作内存中的变量,线程间变量值的传递均需要通过主内存来完成
Java内存模型定义了以下八种操作:
lock(锁定):作用于主内存变量,它把一个变量标识为一条线程独占状态
unlock(解锁):作用于主内存变量,释放处于锁定状态的变量,释放后的变量才可被其它线程使用
read(读取):作用于主内存的变量,它把变量值从主内存传输到工作内存中
load(载入):作用于工作内存的变量,它把read操作的变量值放入工作内存的副本中
use(使用):作用于工作内存的变量,它把工作内存中变量的值传递给执行引擎
assign(赋值):作用于工作内存的变量,它把从执行引擎收到的值赋值给工作内存的变量
store(存储):作用于工作内存变量,它把工作内存中变量值传递到主内存中
write(写入):作用于主内存的变量,它把store操作从工作内存中得到的变量值放入主内存中
volatile关键字
当一个变量被定义成volatile时,它具备两个特性,第1是保证此变量对所有线程的可见性(这里的可见性是指当一条线程修改了这个变量的值,新值对于其它线程立即可见,普通变量不行,因为要通过主内存才能知道),第2是禁止指令重排序优化,意思就是代码执行顺序和代码本身顺序一致。
虽然volatile变量可保证可见性,但它并不一定都是并发安全的。
public static volatile int race = 0; public static void increase(){ race++; } public static void main(String[] args) { Thread[] threads = new Thread[20]; for (int i = 0; i < threads.length; i++) { threads[i] = new Thread(){ public void run() { for (int j = 0; j < 10000; j++) { increase(); } } }; threads[i].start(); } while (Thread.activeCount() > 1) { Thread.yield(); } System.out.println(race); }
上述代码运行结果始终少于200000,得不到正确的运行结果。
我们直接查看javap内容,有助于理解
当getstatic指令把race值取到栈顶时,volatile关键字保证了race值此时是正确的,但在执行iconst_1、add这些指令的时候,其它线程可能已经把race值加大了,而在操作栈顶的值就变成了过期的数据,所以putstatic指令执行后就可能把较小的race值同步回主内存当中了。
volatile在如下两种情况下适合使用:
运算结果并不依赖变量的当前值,或者能确保只有单一线程修改变量的值
变量不需要与其它的状态变量共同参与不变约束
线程状态转换
Java中一共定义了5种线程状态:
新建(new):创建后尚未启动的线程处理这种状态
运行(runnable):runnable包括了操作线程线程状态中的Running和Ready,也就是处理此状态的线程有可能正在执行,也有可能正在等待CPU为它分配执行时间
无限期等待(waiting):处理这种状态的线程不会被分配CPU执行时间,它们要等待被其它线程显示地唤醒,没有设置timeout参数的 Object.wait()和Object.join()方法,会让线程陷入无限等待状态
期限等待(timed waiting):处理这种状态的线程不会被分配CPU执行时间,不过无须等待其它线程唤醒,一定时间后它们会由系统自动唤醒
阻塞(blocked):线程被阻塞,阻塞和等待的区别是:阻塞状态在等待着获取到一个排它锁,这个事件在另外一个线程放弃这外锁的时候发生。而等待状态则是在等待一段时间,或者唤醒动作的发生。
结束:已终止线程的状态,线程已经执行结束
作者:某昆
链接:https://www.jianshu.com/p/4b186593991a
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
使用Python实现一个大数据搜索引擎
搜索是大数据领域里常见的需求。Splunk和ELK分别是该领域在非开源和开源领域里的领导者。本文利用很少的Python代码实现了一个基本的数据搜索功能,试图让大家理解大数据搜索的基本原理。 布隆过滤器 (Bloom Filter) 第一步我们先要实现一个布隆过滤器。 布隆过滤器是大数据领域的一个常见算法,它的目的是过滤掉那些不是目标的元素。也就是说如果一个要搜索的词并不存在与我的数据中,那么它可以以很快的速度返回目标不存在。 让我们看看以下布隆过滤器的代码: classBloomfilter(object): """ ABloomfilterisaprobabilisticdata-structurethattradesspaceforaccuracy whendeterminingifavalueisinaset.Itcantellyouifavaluewaspossibly added,orifitwasdefinitelynotadded,butitcan'ttellyouforcertainthat itwasadded. """ def__init__(self,size):...
- 下一篇
虚拟与真实--vsphere篇ESXI配置与连接
样例项目实战请参考视频:http://www.roncoo.com/course/view/ec2dc65e93da408ab70cf9ead80777f8 关注微信:
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Docker安装Oracle12C,快速搭建Oracle学习环境
- Red5直播服务器,属于Java语言的直播服务器
- CentOS7,CentOS8安装Elasticsearch6.8.6
- CentOS8编译安装MySQL8.0.19
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- CentOS6,CentOS7官方镜像安装Oracle11G
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- SpringBoot2更换Tomcat为Jetty,小型站点的福音
- CentOS8安装Docker,最新的服务器搭配容器使用