上一篇: Java并发之AQS原理解读(一)

前言

本文从源码角度分析AQS独占锁工作原理,并介绍ReentranLock如何应用。

独占锁工作原理

独占锁即每次只有一个线程可以获得同一个锁资源。

获取锁

  1. 尝试获取资源(修改state),成功则返回
  2. 资源不足的情况下,线程会被封装成Node写入阻塞队列,然后以CAS自旋地方式循环重试获取锁(当插入的结点是head的直接后继时尝试获取锁,否则进入阻塞,只有当其他线程释放锁或者调用当前节点线程的中断方法时,才会重试获取锁)
  3. 自旋获取锁成功后,会将当前节点设为队列头结点
  4. 如果自旋阶段发生了线程中断,在获取锁成功之后,会补偿主动调用一次 interrupt 方法。因为自旋时调用的是interrupted方法返回中断标识,调用完后会清除状态

源码分析:

/* 获取独占锁
* 1. tryAcquire 先尝试获取锁,如果成功直接返回;
* 2. 否则 addWaiter 初始化辅助头结点,并将新节点添加到阻塞队列;
* acquireQueued 如果新节点是 head 的直接后继则尝试获取锁,否则 LockSupport 阻塞当前线程,
* 直到被释放锁的线程唤醒或者发生线程中断,才会重新尝试获取锁(CAS自旋阶段);
* 3. 获取锁成功后,如果之前循环重试阶段发生线程中断,则会通过 selfInterrupt 将线程中断标志设为 true
*/
public final void acquire(int arg) {
if (!tryAcquire(arg) && acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
selfInterrupt();
} // 将新节点插入到队尾
private Node addWaiter(Node mode) {
Node node = new Node(Thread.currentThread(), mode);
Node pred = tail;
// 如果tail节点不为null时,直接尝试插入
if (pred != null) {
node.prev = pred;
// 修改tail变量的值为插入节点的地址,即让tail指向新插入的节点
// pred的值不变,还是原tail的地址
if (compareAndSetTail(pred, node)) {
pred.next = node;
return node;
}
}
// tail节点为null时,先初始化辅助头节点,再插入新节点
enq(node);
return node;
} // 初始化辅助头节点,循环地将新节点插入到队尾,直至成功
private Node enq(final Node node) {
for (;;) {
Node t = tail;
if (t == null) {
// 初始化辅助头节点
if (compareAndSetHead(new Node()))
tail = head;
} else {
// 插入新节点
node.prev = t;
if (compareAndSetTail(t, node)) {
t.next = node;
return t;
}
}
}
} // CAS自旋阶段
// 循环重试获取锁,不成功就阻塞,直到被其他释放锁线程唤醒或发生线程中断,方法返回自旋阶段是否发生线程中断
final boolean acquireQueued(final Node node, int arg) {
boolean failed = true;
try {
boolean interrupted = false;
for (;;) {
final Node p = node.predecessor();
/*
* 如果新插入结点是 head 的直接后继,则尝试获取锁
* 获取成功,则将当前节点设为head,并改成 dummy node(假结点)
*/
if (p == head && tryAcquire(arg)) {
setHead(node);
p.next = null; // help GC,丢弃原先的 dummy node
failed = false;
return interrupted;
}
/*
* 获取锁失败后阻塞当前节点,直到其他线程释放锁或调用当前线程的线程中断
* 发生线程中断的情况时,会将 interrupted 设为 true,表示自旋阶段发生了线程中断
* shouldParkAfterFailedAcquire方法在前驱节点状态不为SIGNAL的情况下都会循环重试获取锁
*/
if (shouldParkAfterFailedAcquire(p, node) && parkAndCheckInterrupt())
interrupted = true;
}
} finally {
if (failed)
cancelAcquire(node);
}
} // 根据前驱节点等待状态判断是否要阻塞当前线程
private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {
int ws = pred.waitStatus;
if (ws == Node.SIGNAL)
/*
* 前驱节点等待状态为SIGNAL时,在释放锁的时候会唤醒后继节点,
* 所以当前节点的线程可以阻塞自己
*/
return true;
if (ws > 0) {
/*
* 前驱节点等待状态为CANCELLED时,向前遍历
* 断开对 CANCELLED 状态结点引用,help gc
* 之后会回到循环重试获取锁
*/
do {
node.prev = pred = pred.prev;
} while (pred.waitStatus > 0);
pred.next = node;
} else {
/**
* 等待状态为0或者PROPAGATE(-3)时,设置前驱节点等待状态为SIGNAL,
* 之后会回到循环重试获取锁
*/
compareAndSetWaitStatus(pred, ws, Node.SIGNAL);
}
return false;
} // 阻塞当前线程并返回线程的中断标识
private final boolean parkAndCheckInterrupt() {
// 阻塞当前线程,可以通过 LockSupport.unpark 或 currentThread.interrupt 唤醒
LockSupport.park(this);
return Thread.interrupted();
}

释放锁

  1. 尝试释放资源(修改state),如果失败直接返回
  2. 成功的话,再唤醒阻塞队列中的下一个结点的线程。当前节点后继不符合时,会从队尾往前找

源码分析:

// 释放锁
public final Boolean release(int arg) {
// 释放资源
if (tryRelease(arg)) {
Node h = head;
if (h != null && h.waitStatus != 0)
unparkSuccessor(h);
return true;
}
return false;
} // 获取并唤醒头结点后的一个待唤醒结点(waitStatus<=0)
private void unparkSuccessor(Node node) {
int ws = node.waitStatus;
if (ws < 0)
compareAndSetWaitStatus(node, ws, 0);
Node s = node.next;
if (s == null || s.waitStatus > 0) {
s = null;
// 结点直接后继不符合,则从队尾向前找
for (Node t = tail; t != null && t != node; t = t.prev)
if (t.waitStatus <= 0)
s = t;
}
if (s != null)
// 唤醒阻塞结点的线程
LockSupport.unpark(s.thread);
}

ReentranLock实现

非公平锁含义在于拥有锁的线程释放锁的时候,当前尝试获取锁资源的线程可以和队列中的第一个等待线程竞争;而已经进入队列的线程只能按照先进先出的顺序获取锁,也就是公平锁的逻辑。

第一次获取锁时,只要锁还未被占用,非公平锁会先直接通过CAS抢占,而公平锁则会判断是否有其他结点先进入队列,没有的话才会尝试获取锁。

ReentranLock是可重入锁,可重入即获得锁的线程可以重复进入临界区。通过实现AbstractOwnableSynchronizer接口记录获取锁的线程,当线程重复进入临界区时state+1,退出临界区时state-1

非公平锁

// 先尝试 CAS 抢占锁,失败后再通过 acquire 尝试
final void lock() {
if (compareAndSetState(0, 1))
setExclusiveOwnerThread(Thread.currentThread());
else
acquire(1);
} protected final boolean tryAcquire(int acquires) {
return nonfairTryAcquire(acquires);
} final boolean nonfairTryAcquire(int acquires) {
final Thread current = Thread.currentThread();
int c = getState();
if (c == 0) {
// 当锁未被占用时,尝试获取锁
if (compareAndSetState(0, acquires)) {
setExclusiveOwnerThread(current);
return true;
}
}
else if (current == getExclusiveOwnerThread()) {
// 已获得锁的线程支持重入
int nextc = c + acquires;
if (nextc < 0) // overflow
throw new Error("Maximum lock count exceeded");
setState(nextc);
return true;
}
return false;
}

公平锁

// 不抢占,通过 acquire 尝试
final void lock() {
acquire(1);
} protected final boolean tryAcquire(int acquires) {
final Thread current = Thread.currentThread();
int c = getState();
if (c == 0) {
/**
* 当锁未被占用时,且阻塞队列中没有结点比当前节点更早开始等待时,才尝试获取锁
*/
if (!hasQueuedPredecessors() &&
compareAndSetState(0, acquires)) {
setExclusiveOwnerThread(current);
return true;
}
}
else if (current == getExclusiveOwnerThread()) {
// 已获得锁的线程支持重入
int nextc = c + acquires;
if (nextc < 0)
throw new Error("Maximum lock count exceeded");
setState(nextc);
return true;
}
return false;
}

下一篇 Java并发之AQS原理解读(三) 将介绍共享锁工作原理。

参考:

Java AQS源码解读

hasQueuedPredecessors源码分析

Java并发之AQS原理解读(二)的更多相关文章

  1. Java并发之AQS原理解读(三)

    上一篇:Java并发之AQS原理解读(二) 前言 本文从源码角度分析AQS共享锁工作原理,并介绍下使用共享锁的子类如何工作的. 共享锁工作原理 共享锁与独占锁的不同之处在于,获取锁和释放锁成功后,都会 ...

  2. Java并发之AQS原理解读(一)

    前言 本文简要介绍AQS以及其中两个重要概念:state和Node. AQS 抽象队列同步器AQS是java.util.concurrent.locks包下比较核心的类之一,包括AbstractQue ...

  3. Java并发之AQS原理剖析

    概述: AbstractQueuedSynchronizer,可以称为抽象队列同步器. AQS有独占模式和共享模式两种: 独占模式: 公平锁: 非公平锁: 共享模式: 数据结构: 基本属性: /** ...

  4. 并发之AQS原理(二) CLH队列与Node解析

    并发之AQS原理(二) CLH队列与Node解析 1.CLH队列与Node节点 就像通常医院看病排队一样,医生一次能看的病人数量有限,那么超出医生看病速度之外的病人就要排队. 一条队列是队列中每一个人 ...

  5. 并发之AQS原理(三) 如何保证并发

    并发之AQS原理(三) 如何保证并发 1. 如何保证并发 AbstractQueuedSynchronizer 维护了一个state(代表了共享资源)和一个FIFO线程等待队列(多线程竞争资源被阻塞时 ...

  6. 并发之AQS原理(一) 原理介绍简单使用

    并发之AQS原理(一) 如果说每一个同步的工具各有各的强大,那么这个强大背后是一个相同的动力,它就是AQS. AQS是什么 AQS是指java.util.concurrent.locks包里的Abst ...

  7. 深入理解Java并发框架AQS系列(二):AQS框架简介及锁概念

    深入理解Java并发框架AQS系列(一):线程 深入理解Java并发框架AQS系列(二):AQS框架简介及锁概念 一.AQS框架简介 AQS诞生于Jdk1.5,在当时低效且功能单一的synchroni ...

  8. Java并发之AQS详解

    一.概述 谈到并发,不得不谈ReentrantLock:而谈到ReentrantLock,不得不谈AbstractQueuedSynchronizer(AQS)! 类如其名,抽象的队列式的同步器,AQ ...

  9. Java并发之AQS详解(转)

    一.概述 谈到并发,不得不谈ReentrantLock:而谈到ReentrantLock,不得不谈AbstractQueuedSynchronized(AQS)! 类如其名,抽象的队列式的同步器,AQ ...

随机推荐

  1. 【洛谷 P2388 阶乘之乘】模拟

    分析 求因数5的个数 AC代码 #include<iostream> using namespace std; int main() { long long n,t,ans=0,s=0; ...

  2. LUSE: 无监督数据预训练短文本编码模型

    LUSE: 无监督数据预训练短文本编码模型 1 前言 本博文本应写之前立的Flag:基于加密技术编译一个自己的Python解释器,经过半个多月尝试已经成功,但考虑到安全性问题就不公开了,有兴趣的朋友私 ...

  3. JS 之循环 应用案例1

    应用场景:将el-select下拉框的lab值,显示在下面的详情text框,见下图 用到了 el-select 传值,js循环判断options.code是否等于传进来的值,等于就break; tex ...

  4. 01.泛型Generic

    1. 基本了解 1.1 什么是泛型? 字面意思:不确定的类型 泛型常用:泛型方法,泛型类,泛型接口,泛型委托 1.2 泛型 T(熟悉) T 的作用,其实就是一个通用的容器,制造它的人开始不指定它是用来 ...

  5. centos ansible常用命令

    ansible在日常运维中经常使用,特别是批量执行多台服务器的时候,有效减小重复的操作成本,以下从安装到使用仅讲解工作中常用的几种方式,模块很多功能很强大,但不做全面讨论. ansible安装 在ce ...

  6. csredis-in-asp.net core理论实战-哨兵模式-使用示例

    csredis 开源地址 https://github.com/2881099/csredis 续上篇 csredis-in-asp.net core理论实战-主从配置.哨兵模式 示例源码 https ...

  7. QGIS打印布局中绘制多个子图

    QGIS如何绘制多图 数据准备 这是一份英国大曼彻斯特地区的数据,里面包含了教育.收入.人口密度.绿地比例.城市比例等数据,我们准备把这些数据在地图上呈现出来,为此,我们需要做在地图中绘制6幅子图,这 ...

  8. Sqlserver 关于varchar(max) 笔记

    看SQL server的版本,SQLserver2005以上 的nvarchar(max) 可以存放2G的内容,所以要是 SQL2005以上的nvarchar(max)足够你用的了.用nvarchar ...

  9. C++员工管理系统(封装+多态+继承+分类化+函数调用+读写文件+指针+升序降序算法等一系列知识结合)

    1 C++职工管理系统 2 该项目实现 八个 功能 3 1-增加功能 2-显示功能 3-删除功能 4-修改功能 4 5-查找功能 6-排序功能 7-清空功能 8-退出功能 5 实现多个功能使用了多个C ...

  10. 三年Android开发,月薪一万二,不敢跳槽,每天都很焦虑

    在我们的身边,存在一个普遍现象:很多人从事Android开发工作多年,走过的弯和坎,不计其数,经历的心酸难与外人道也.可是技术确难以提升.止步不前,薪资也只能看着别人水涨船高,自己却没有什么起色. 虽 ...