前言

Java代码在编译后会变成Java字节码,字节码被类加载器加载到JVM里,JVM执行字节
码,最终需要转化为汇编指令在CPU上执行,Java中所使用的并发机制依赖于JVM的实现
CPU的指令


第二章:Java并发机制的底层实现原理

2.1 volatile的应用

volatile是轻量级synchronized,能保证共享变量的“可见性”。比synchronized使用和执行成本低,不会引起线程上下文的切换和调度。

下面先来看与其原理相关的CPU术语与说明。

汇编代码中,lock前缀的指令引发了两件事情:

  1. 将当前处理器缓存行的数据写回到系统内存
  2. 这个写回内存的操作使在其他CPU里缓存了该内存地址的数据无效。

    如果对声明了volatile的变量写操作,JVM会向处理器发送一条Lock前缀的指令,将这个变量所在缓存行的数据写回到系统内存。为了保证其他处理器的缓存也是新的,实现缓存一致性协议。。每个处理器嗅探在总线上的数据检查自己缓存是否过期。当处理器发现自己缓存行对应的内存地址被修改,就会将当前缓存行设置成无效状态,会重新从系统内存中把数据读到处理器缓存里。

追加字节能优化性能?

如果队列的头节点和尾节点都不足64字节的话,处理器会将
它们都读到同一个高速缓存行中,在多处理器下每个处理器都会缓存同样的头、尾节点,当一
个处理器试图修改节点时,会将整个缓存行锁定,那么在缓存一致性机制的作用下,会导致
其他处理器不能访问自己高速缓存中的节点,而队列的入队和出队操作则需要不停修改头
节点和尾节点,所以在多处理器的情况下将会严重影响到队列的入队和出队效率。

两种场景下不该使用这种方式:
缓存行非64字节的处理器。
共享变量不会被频繁地写。


2.2 synchronized的实现原理与应用

Java的每一个对象都可以作为锁。
对于普通同步方法,锁是当前实例对象。
对于静态同步方法,锁是当前类的class对象。
对于同步方法块,锁是synchronized括号里配置的对象。
方法同步和代码块同步,两种的实现细节不一样。代码块使用monitorenter和monitorexit指令实现的。
monitorenter指令是在编译后插入到同步代码块的开始位置,而monitorexit是插入到方法结
束处和异常处。


2.2.1 Java对象头
synchronized用的锁是存在Java对象头里的。非数组类型,2Byte存对象头,数组类型3Byte。
HotSpot虚拟机中,对象在内存中存储的布局可以分为三块区域:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。 对象头默认存储对象的HashCode(25bit),分代年龄(4bit)和锁标记位。


2.2.2 锁的升级与对比
Java SE 1.6为了减少获得锁和释放锁带来的性能消耗,引入了“偏向锁”和“轻量级锁”。1.6中,一共有4种状态,由低到高是无锁,偏向锁,轻量锁,重量锁。锁只能升级。
1.BiaseLocking
当一个线程访问同步块并获取锁时,会在对象头和栈帧中的锁记录里存储锁偏向的线程ID,以后该线程在进入和退出
同步块时不需要进行CAS操作来加锁和解锁,只需简单地测试一下对象头的Mark Word里是否
存储着指向当前线程的偏向锁。 如果测试成功,表示线程已经获得了锁。如果测试失败,则需
要再测试一下Mark Word中偏向锁的标识是否设置成1(表示当前是偏向锁):如果没有设置,则
使用CAS竞争锁;如果设置了,则尝试使用CAS将对象头的偏向锁指向当前线程

偏向锁使用了一种等到竞争出现才释放锁的机制,所以当其他线程尝试竞争偏向锁时,
持有偏向锁的线程才会释放锁。偏向锁的撤销,需要等待全局安全点(在这个时间点上没有正
在执行的字节码)。它会首先暂停拥有偏向锁的线程,然后检查持有偏向锁的线程是否活着,
如果线程不处于活动状态,则将对象头设置成无锁状态;如果线程仍然活着,拥有偏向锁的栈
会被执行,遍历偏向对象的锁记录,栈中的锁记录和对象头的Mark Word要么重新偏向于其他
线程,要么恢复到无锁或者标记对象不适合作为偏向锁,最后唤醒暂停的线程。

关闭偏向锁,程序默认进轻量级锁状态。

2.轻量级锁
线程在执行同步块之前,JVM会先在当前线程的栈桢中创建用于存储锁记录的空间,并
将对象头中的Mark Word复制到锁记录中,官方称为Displaced Mark Word。然后线程尝试使用
CAS将对象头中的Mark Word替换为指向锁记录的指针。如果成功,当前线程获得锁,如果失
败,表示其他线程竞争锁,当前线程便尝试使用自旋来获取锁。

轻量级解锁时,会使用原子的CAS操作将Displaced Mark Word替换回到对象头,如果失败,表示当前锁存在竞争,锁就会膨胀成重量级锁。

3.锁的优缺点:


2.3 原子操作的实现原理
不可中断的一个或一系列操作。
首先处理器会自动保存“基本“的内存操作的原子性。当一个处理器读取一个字节时,其他处理器不能访问这个字节的内存地址。
但复杂的内存操作是不能自动保证其原子性的,比如跨总线宽度,跨多个缓存行和跨页表的访问。但是,处理器提供“总线锁定““缓存锁定“来保证复杂内存操作的原子性。

  1. 总线锁就是使用处理器提供的一个LOCK#信号,当一个CPU在总线输出此信号时,其他处理器的请求将被阻塞住,那么该CPU可以独占该内存。
  2. 缓存锁定是指内存区域如果被缓存在CPU的缓存行中,并且在Lock操作期间被锁定,那么当它执行锁操作回写到内存时,CPU不在总线上声明LOCK#信号,而是修改内部的内存地址,并允许它的缓存一致性机制来保证操作的原子性。因为缓存一致性机制能阻止同时修改两个以上CPU缓存的内存区域数据。

    有两种情况CPU不会使用缓存锁定:
    1 .操作的数据不能被缓存在CPU内部。或操作的数据跨多个缓存行
    2.有些处理器不支持缓存锁定

Java中如何实现原子操作

  1. 使用循环CAS实现
  2. CAS实现原子性操作的三大问题

    ABA问题:JDK1.5里,AtomicStampedReference可以解决。这个类的compareAndSet方法的作用是首先检查当前引用是否为预期引用,并检查当前标志是否为预期标志,若全部相等,以原子方式将该引用和该标志的值设置为给定的更新值。

循环时间长开销大:JVM支持处理器提供的pause指令。

只能保证一个共享变量的原子操作:对多个共享变量,循环CAS无法保证操作的原子性。一个取巧的办法,把多个共享变量合并成一个对象:i=2,j=a,合并为ij=2a。由AtomicReference支持

3.使用锁机制实现原子操作
除了偏向锁,JVM实现锁都用了循环CAS,当一个线程想进入同步块时,循环CAS获取锁,退出同步块时,循环CAS释放锁。

《Java并发编程的艺术》:第2章 Java并发机制的底层实现原理相关推荐

  1. 【Java并发编程的艺术】读书笔记——Java并发编程基础

    学习参考资料:<Java并发编程的艺术> 文章目录 1.线程的几种状态 2.如何安全的终止线程 3.线程间通信(重要) 3.1共享内存 3.2消息传递 1.线程的几种状态 线程在运行的生命 ...

  2. 并发编程的艺术:第二章

    第二章:Java并发机制的底层实现原理 Java代码在编译之后会生成字节码文件,也就是.class文件.class文件经过类加载器的加载到JVM中,JVM解释字节码文件,最终转化为汇编指令在CPU上执 ...

  3. 【Java并发编程的艺术】第二章读书笔记之原子操作

    前言 今天的笔记来了解一下原子操作以及Java中如何实现原子操作. 概念 原子(atomic)本意是"不能被进一步分割的最小粒子",而原子操作(atomic operation)意 ...

  4. Java 并发编程的艺术 pdf 下载

    并发编程领域的扛鼎之作,作者是阿里和1号店的资深Java技术专家,对并发编程有非常深入的研究,<Java并发编程的艺术>是他们多年一线开发经验的结晶.本书的部分内容在出版早期发表在Java ...

  5. # Java 并发编程的艺术(一)

    Java 并发编程的艺术(一) 文章目录 Java 并发编程的艺术(一) Java中的线程池 线程池的实现原理 线程池的处理流程 ThreadPoolExecutor执行流程 线程池队列 线程池拒绝策 ...

  6. Java并发编程的艺术pdf

    下载地址:网盘下载 并发编程领域的扛鼎之作,作者是阿里和1号店的资深Java技术专家,对并发编程有非常深入的研究,<Java并发编程的艺术>是他们多年一线开发经验的结晶.本书的部分内容在出 ...

  7. [转] 《Java并发编程的艺术》笔记

    转自https://gitee.com/Corvey/note 作者:Corvey 第一章 并发编程的挑战 略 第二章 Java并发机制的底层实现原理 volatile的两条实现原则: Lock前缀指 ...

  8. 多线程知识梳理(2) - 并发编程的艺术笔记

    layout: post title: <Java并发编程的艺术>笔记 categories: Java excerpt: The Art of Java Concurrency Prog ...

  9. 《Java并发编程的艺术》一一第2章Java并发机制的底层实现原理

    第2章Java并发机制的底层实现原理 2.1 volatile的应用 Java代码在编译后会变成Java字节码,字节码被类加载器加载到JVM里,JVM执行字节码,最终需要转化为汇编指令在CPU上执行, ...

最新文章

  1. VS2015配置并运行汇编(一步一步照图做)【vs2017的链接在最后】
  2. 人工智能、网络空间对国家核安全的影响
  3. Python 数据类型--Bytes类型
  4. Python 字符串 String 内建函数大全(1)
  5. 征稿 | ​2020年全国知识图谱与语义计算大会
  6. Spring properties定义bean
  7. Windows Server 2016 AD中新建组织单位、组、用户
  8. LR录制脚本后,中文显示的是乱码,怎么解决?
  9. Leetcode95. Unique Binary Search Trees II不同的二叉搜索树2
  10. 数学建模学习之模糊评价法
  11. Thunderbird 邮件签名三个实现方式
  12. 【广告计算】互联网控制舆论的三个方法
  13. StarRocks 企业行|走进 58 同城,探索极速统一 3.0 时代的企业实践
  14. MYSQL中AS(取别名)
  15. Extjs——初步学习
  16. 如何有效实现软件的需求管理 - 3
  17. 安全多方计算-入门学习笔记(三)
  18. Python 蜻蜓fm有声书批量下载 支持账号登录 原创源码
  19. Python面向对象,从农药到吃鸡 bilibili视频笔记
  20. org.apache.poi.ss.usermodel.Cell.getCellType()Lorg/apache/poi/ss/usermodel/CellType(百分比解决你问题)

热门文章

  1. android与ndk交互,NDK-JNI与Java的交互 hello-world
  2. 现代ups电源及电路图集_2020山特UPS电源自动开机200KVA实力
  3. 微服务乱码_本地正常服务器乱码
  4. python计数循环,python - Python中的密码求解器循环计数 - SO中文参考 - www.soinside.com...
  5. dump的文件 查看pg_PostgreSQL 逻辑复制异常引发Pg_wal目录WAL文件膨胀一例
  6. php单进程锁定,强制PHP命令行脚本单进程运行的方法
  7. HALCON 20.11:标定助手使用注意事项
  8. QString转HTuple
  9. AgileEAS.NET SOA 中间件平台5.2版本下载、配置学习(四):开源的Silverlight运行容器的编译、配置...
  10. 网络对抗技术实验二,第一部分,第二部分