aqs原理初探以及公平锁和非公平锁实现

article/2025/9/27 6:13:37

深入理解AQS

  • 一,AQS
    • 1,ReentrantLock
    • 2,CAS
    • 3,AbstractQueuedSynchronizer
      • 3.1,FairSync
      • 3.2,NofairSync
      • 3.3,AQS中几个重要的相关参数
      • 3.4,Node

一,AQS

AbstractQueuedSynchronizer,定义了一套多线程访问共享资源的同步器框架,依赖于状态的同步器

1,ReentrantLock

一种基于AQS框架的应用实现,类似于synchronized是一种互斥锁,可以保证线程安全。它支持手动加锁与解锁,支持加锁的公平性。主要是Lock锁的实现

public class ReentrantLock implements Lock, Serializable

接下来可以手动的猜想一下这个reentrantLock的实现

ReentrantLock lock = new ReentrantLock(true);
HashSet hashSet = new HashSet();
3个线程 T0,T1,T2
lock.lock(); //加锁while(true){ //循环,轮询获取锁if(加锁成功){ //synchronized,cas  cas:compare and swapbreak;//跳出循环}//Thread.yeild() //让出cpu使用权//Thread.sleep(1000); //睡眠hashSet.add(thread); //将当前线程加入到set中//阻塞LockSupprot.park();}T0获取锁xxxxx业务逻辑xxxxx业务逻辑   
lock.unlock();
Thread thread = hashSet.get();
//唤醒当前线程,notify是唤醒随机的线程
LockSupport.unPark(thread);

三大核心:自旋,加锁,LockSupport,队列(LinkQueue),为了解决这个公平锁和非公平锁,因此优先考虑这个队列。

2,CAS

compare and swap,比较与交换
在这里插入图片描述

如在jmm模型中,两个工作内存都去修改主内存的值。主内存中存在一个a = 0,线程A和线程B的工作内存同时获取到这个值,如果线程A先修改这个值,则线程A会和主内存比较,如果线程A的值a和主内存的值一致,那么就会直接进行修改,如改成a = 1,那么线程B也要改这个值,线程B中的a = 0,那么会和主内存a比较,发现不一致,主内存a=1,那么就会优先将线程B中的值修改成a = 1,再对a进行修改。就是说相等直接修改,不相等需要重新读取,再进行修改。即在一个原子操作里面进行比较和替换

主要通过这个unsafe类实现,里面的实现也是原子类操作,主要是通过以下三个类实现。

//对象类型
public final native boolean compareAndSwapObject(Object var1, long var2, Object var4, Object var5);
//整型值
public final native boolean compareAndSwapInt(Object var1, long var2, int var4, int var5);
//Long型值
public final native boolean compareAndSwapLong(Object var1, long var2, long var4, long var6);

3,AbstractQueuedSynchronizer

该类是ReentrantLock里面的一个抽象内部类。ctrl + alt + shift + u看所有子类,Ctrl + T,看所有的继承类,可以发现很多地方都继承或者实现了这个抽象类

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4dzYpLnW-1657115375965)(C:\Users\HULOUBO\AppData\Roaming\Typora\typora-user-images\1656947493769.png)]
Sync是ReentrantLock的一个抽象的静态内部类,根据图也可以发现这个Sync继承了这个AQS

abstract static class Sync extends AbstractQueuedSynchronizer

通过实现Sync这个接口得到了FairSync公平锁类和NofairSync非公平锁这个类

3.1,FairSync

实现了公平锁,需要排队获取锁,如存在线程t1,t2,t3依次获取锁,需要依次排队执行,突然来了一个线程t4,也是需要排在线程t3后面

ReentrantLock reentrantLock = new ReentrantLock(true);
public ReentrantLock(boolean fair) {//入参,用于判断是公平锁还是非公平锁sync = fair ? new FairSync() : new NonfairSync();
}
//公平锁的实现
static final class FairSync extends ReentrantLock.Sync{...}

公平锁获取锁的方式如下

final void lock() {acquire(1);
}public final void acquire(int arg) {//tryAcquire:尝试去获取锁//addWaiter:线程入队,一个同步等待队列,基于双向列表实现//链表中的每一个结点为一个Node结点,if (!tryAcquire(arg) &&acquireQueued(addWaiter(Node.EXCLUSIVE), arg))selfInterrupt();
}//线程入队操作
private Node addWaiter(Node mode) {//获取当前线程结点Node node = new Node(Thread.currentThread(), mode);// Try the fast path of enq; backup to full enq on failureNode pred = tail;if (pred != null) {node.prev = pred;if (compareAndSetTail(pred, node)) {pred.next = node;return node;}}//结点入队enq(node);return node;
}//判断是否获取锁
protected final boolean tryAcquire(int acquires) {//获取当前获取锁的线程final Thread current = Thread.currentThread();//获取当前同步器状态,被volatile修饰的整型值,默认为0int c = getState();//如果同步状态器的值为0,说明外面的线程可以进行加锁操作if (c == 0) {//hasQueuedPredecessors:判断队列中是否还有在排队的节点//compareAndSetState:原子操作,比较与交换,进行加锁的操作,将state变量将0变为1//setExclusiveOwnerThread:设置获取锁的的线程拥有者if (!hasQueuedPredecessors() &&compareAndSetState(0, acquires)) {setExclusiveOwnerThread(current);return true;}}//如果状态同步器不为0,可能由自己持有,也可能由别的线程持有锁//重复加锁,如定义一个全局锁,出现了这个可重入锁的问题else if (current == getExclusiveOwnerThread()) {//如果自己持有锁的话,state+1即可,反正不等于0就可以int nextc = c + acquires;if (nextc < 0)throw new Error("Maximum lock count exceeded");setState(nextc);return true;}//别的线程获有锁,直接返回return false;
}

总而言之就是说,公平锁就是就是通过一个队列实现,需要进行排队的去获取锁资源。主要是通过这个state的资源状态器来控制获取锁的拥有者,如果state为0,则表示队列中的下一个线程可以去获取锁,并且通过cas的方式来保证锁的安全并发问题。通过队列的思想,来保证这个获取锁的公平性和有序性。

3.2,NofairSync

实现了非公平锁,默认为非公平锁,会存在抢锁的情况

ReentrantLock reentrantLock = new ReentrantLock(flase);
//如果不传入参数,默认是非公平锁
public ReentrantLock() {sync = new NonfairSync();
}
public ReentrantLock(boolean fair) {//入参,用于判断是公平锁还是非公平锁sync = fair ? new FairSync() : new NonfairSync();
}
//非公平锁的实现
static final class NonfairSync extends ReentrantLock.Sync{...}

非公平锁获取锁的方法和公平锁类似,只是少了几步使用队列的几个方法

3.3,AQS中几个重要的相关参数

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-12BMqIQA-1657115375967)(C:\Users\HULOUBO\AppData\Roaming\Typora\typora-user-images\1657029597278.png)]

exclusiveOwnerThread:用于记录当前独占模式下,获取锁的线程是谁

state:同步状态器,默认为0,表示当前没有线程获取锁,外面的线程可以来获取锁了

Node:双向链表结构,是一个同步等待队列,head:队头,tail:队尾,prev前躯指针,next,后继指针

waiteState:结点的什生命状态

Lock锁和synchronized锁都是可重入锁

3.4,Node

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Zfr5p0Ru-1657115375970)(C:\Users\HULOUBO\AppData\Roaming\Typora\typora-user-images\1657110491185.png)]
pre:前驱指针
next:后继指针
waitStatues:每个结点都存在很多状态,这个主要是存储结点的生命状态

结点的几个生命状态如下

SIGNAL:-1     //可被唤醒
CANCELLED:1	  //代表异常,中断引起的,需要被废弃
CONDITION:-2  //条件等待
PROPAGATE:-3  //传播
Init:0初始状态

结点入队顺序如下,入队时,将入队结点得前驱指针指向链表的tail结点,将tail节点的next节点指向当前节点,并将当前结点设置为tail尾指针结点。

private Node enq(final Node node) {//自旋for (;;) {Node t = tail;if (t == null) { // Must initialize//如果队列为空,要防止出现多个线程的并发问题,结点直接放在队头if (compareAndSetHead(new Node()))tail = head;} else {node.prev = t;//保证这个入队的安全性,防止入队出现并发问题//结点入队到队尾if (compareAndSetTail(t, node)) {t.next = node;return t;}}}
}

当前结点前面有结点获取锁,当前节点需要进行阻塞park,线程要开始排队等待。
结点在阻塞之前,还得尝试获取一次锁。
a,如果结点可以获取到锁,即当前节点为头结点的下一个结点,头结点即将锁被释放,则把当前结点作为头结点。之前的头结点就可以被GC回收了
b,如果结点不能获取到锁,那么当前结点就要等待被唤醒
​ (1),第一轮循环会去修改head状态,并且将waitState修改为sinal = -1可被唤醒状态
​ (2),第二轮,阻塞线程

final boolean acquireQueued(final Node node, int arg) {boolean failed = true;try {boolean interrupted = false;for (;;) {//当前节点的前驱指针指向的节点final Node p = node.predecessor();//如果当前结点指向的该结点为头部结点,则不需要进行阻塞if (p == head && tryAcquire(arg)) {//将当前结点作为头部结点setHead(node);p.next = null; // help GCfailed = false;return interrupted;}if (shouldParkAfterFailedAcquire(p, node) &&parkAndCheckInterrupt())interrupted = true;}} finally {if (failed)cancelAcquire(node);}
}

每个结点的生命状态的变换如下。默认的结点状态为0,需要将节点状态(waitState)转化为-1可唤醒状态。前一个节点中的waitStatus状态记录着后一个节点的生命状态。

//pred:前驱节点 node:当前节点
private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {//默认为初始状态0int ws = pred.waitStatus;//SIGNAL为-1,可被唤醒状态if (ws == Node.SIGNAL)return true;if (ws > 0) {do {node.prev = pred = pred.prev;} while (pred.waitStatus > 0);pred.next = node;} else {//将头结点初始状态转化为可唤醒状态compareAndSetWaitStatus(pred, ws, Node.SIGNAL);}return false;
}

在修改成可被唤醒的状态之后,就进行阻塞操作。

private final boolean parkAndCheckInterrupt() {//阻塞线程,并且判断该线程是否是被中断的LockSupport.park(this);return Thread.interrupted();
}

在头结点释放锁的时候,也会发一个通知去告知下一个需要获取锁的线程来抢锁,即唤醒队列中的下一个被阻塞的线程

//真正的释放锁
public void unlock() {sync.release(1);
}
public final boolean release(int arg) {//尝试释放锁if (tryRelease(arg)) {Node h = head;//因为前一个结点中存放后一个节点中的声明状态,因此//如果头结点的waitStatus不为0,就说明后一个结点是一个可被唤醒的状态if (h != null && h.waitStatus != 0)//唤醒unparkSuccessor(h);return true;}return false;
}
//尝试去释放锁
protected final boolean tryRelease(int releases) {//修改同步状态器stateint c = getState() - releases;if (Thread.currentThread() != getExclusiveOwnerThread())throw new IllegalMonitorStateException();boolean free = false;//将同步状态器state 变为 0,说明当前同步状态器可以进行锁的获取了if (c == 0) {free = true;//置空当前线程setExclusiveOwnerThread(null);}setState(c);return free;
}

最后在这个unparkSuccessor方法中,也有这个具体的unpark唤醒操作

if (s != null)LockSupport.unpark(s.thread)

通过上述代码描述,也验证了一开始的猜想:自旋,加锁,LockSupport,队列(LinkQueue)


http://chatgpt.dhexx.cn/article/OmhkX5pi.shtml

相关文章

图解ReentrantLock底层公平锁和非公平锁实现原理

&#x1f4bb;在面试或者日常开发当中&#xff0c;经常会遇到公平锁和非公平锁的概念。 两者最大的区别如下&#x1f447; 1️⃣ 公平锁&#xff1a;N个线程去申请锁时&#xff0c;会按照先后顺序进入一个队列当中去排队&#xff0c;依次按照先后顺序获取锁。就像下图描述的上…

ReentrantLock之公平锁和非公平锁详解

ReentrantLock是一个互斥锁&#xff0c;它具有synchronized相同的能力&#xff1b;但相比之下&#xff0c;ReentrantLock扩展性更强&#xff0c;比如实现了公平锁。 下面详细拆解下ReentrantLock的公平锁和非公平锁的实现。 JDK版本&#xff1a;1.8.0_40 公平锁 先看Reentr…

ReentrantLock中公平锁和非公平锁的区别

目录 背景知识 ReentrantLock的组成 概述 公平锁示意图 非公平锁示意图 源码解读 非公平锁 公平锁 代码对比 问题 知识扩展 tryLock方法 参考资料 背景知识 ReentrantLock的组成 首先看下ReentrantLock的组成结构。 公平锁和非公平锁主要是通过内部类FairSync和…

公平锁和非公平锁

Reentrant Re entrant&#xff0c;Re是重复、又、再的意思&#xff0c;entrant是enter的名词或者形容词形式&#xff0c;翻译为进入者或者可进入的&#xff0c;所以Reentrant翻译为可重复进入的、可再次进入的&#xff0c;因此ReentrantLock翻译为重入锁或者再入锁。 公平锁…

阿里面试官:说一下公平锁和非公平锁的区别?

点赞再看&#xff0c;养成习惯&#xff0c;微信搜索【三太子敖丙】关注这个互联网苟且偷生的工具人。 本文 GitHub https://github.com/JavaFamily 已收录&#xff0c;有一线大厂面试完整考点、资料以及我的系列文章。 前言 上次我们提到了乐观锁和悲观锁&#xff0c;那我们知道…

Ubuntu 手动安装 JDK8

文章目录 1. 下载2. 解压安装3. 配置环境变量 1. 下载 先去官网下载合适的版本&#xff0c;官网&#xff1a;https://www.oracle.com/java/technologies/downloads/archive/ 通过下载页面获取到下载链接后&#xff0c;可以直接在Ubuntu上使用wget下载&#xff0c;也可以先下载…

centos8安装jdk教程

文章目录 一、安装二、配置环境变量三.验证 一、安装 1、查看JDK软件包列表 yum search java | grep -i --color jdk2、选择版本安装 yum install -y java-1.8.0-openjdk java-1.8.0-openjdk-devel或者如下命令安装jdk8所有文件 yum install -y java-1.8.0-openjdk*二、配置…

Java - JDK8安装及配置环境变量教程

Java - JDK8安装及配置环境变量教程 一、安装JDK教程 甲骨文官网下载JDK版本&#xff1a;windows64下载地址 下载完成后开始安装JDK&#xff1a;双击打开 点击下一步&#xff1a; 若不需要自定义路径&#xff0c;则安装到默认路径即可&#xff08;安装的路径需记住&#xff0…

JDK8安装和环境配置

JDK8的安装和环境配置 一、JDK8下载二、安装三、环境配置 一、JDK8下载 官网下载&#xff1a; https://www.oracle.com/java/technologies/downloads/#java8-windows 二、安装 打开安装&#xff0c;一直下一步即可&#xff0c;可以在安装过程中更改安装地址&#xff0c;我放…

Java JDK 8的安装与配置

文章目录 前言1. 安装JDK 8Step1&#xff1a;选择JDK的版本Step2&#xff1a;选择系统平台Step3&#xff1a;下载安装包Step4&#xff1a;开始安装 2. 配置JDK 8Step1&#xff1a;配置“环境变量path” 前言 本教程是在Windows 64位平台上安装JDK 8版本。 1. 安装JDK 8 官网…

JDK8安装与环境配置

前言&#xff1a;在网上看了下JDK的安装与环境配置&#xff0c;发现很多视频以及博客的讲的都很复杂&#xff0c;对初学者很不友好。很多小白看到这些配置步骤都一脸懵&#xff0c;即使一步一步看着操作还是配置失败。这主要是因为很多博主配置了不需要的配置环境&#xff0c;让…

Linux安装JDK8详细图文教程

第一步、获取JDK文件 JDK下载包&#xff1a;直接进入 如果跳转登录页面&#xff0c;注册一个账号登录即可 登录过后文件就下载完成 第二步、上传JDK到服务器 1、创建JDK目录 mkdir -p /developer/env/jdk进入目录 cd /developer/env/jdk2、安装lrzsz&#xff08;用于远程传…

JDK8安装教程(Windows、Ubuntu)

文章目录 下载JDK8在Windows上安装设置环境变量 在Ubuntu上安装 下载JDK8 Linux版&#xff0c;CSDN下载地址 推荐官网下载&#xff08;需要注册&#xff09;&#xff0c;能下到Windows和Ubuntu的最新版 在Windows上安装 双击打开下载好的安装包&#xff0c;点击下一步 点击左…

linux系统安装jdk8详细教程

文章目录 前言一、下载jdk8的安装包二、压缩包上传解压1.将下载好的压缩包使用ftp工具上传到服务器2.将压缩包解压到指定目录 三、配置jdk的环境变量四、测试是否安装成功 前言 虚拟机版本&#xff1a;centos7 jdk版本&#xff1a;1.8 一、下载jdk8的安装包 方法1&#xff1a…

jdk8安装教程及环境变量配置

目录 一、JDK下载 二、安装JDK 三、环境变量配置 四、测试环境变量 一、JDK下载 1、JDK下载地址&#xff1a;Java Downloads | Oraclehttps://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html 2、选这个位置 3、向下滑动鼠标&#xff0c…

WIN10javaJDK8安装教程

一、下载jdk8文件 下载地址: http://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html 下载步骤&#xff1a; 1、进入网站后往下拉&#xff0c;找到下载界面&#xff0c;选择自己合适的下载。 二、安装 双击运行刚下载好的exe文件…

JDK8安装教程-极其详细

下载安装JDK 前往oracle官网下载JDK 链接&#xff1a;https://www.oracle.com/java/technologies/downloads/#java8-windows下载完成后 打开下载的程序进行安装JDK 这里可以选择你要存放的位置 选择jre安装的路径 继续下一步&#xff0c;等待安装完成 配置环境变量 此电脑…

jdk8安装教程及环境变量部署

一、jdk下载 下载方式&#xff1a; 官网下载 地址&#xff1a;Java Downloads | Oracle网盘下载&#xff1a;链接&#xff1a;https://pan.baidu.com/s/1jjcOsdyaLfAy2N5zp64sew 提取码&#xff1a;tzsj 官网下载&#xff1a; 这是最新版本的可以根据自己的需要选择 选择W…

win10系统安装jdk8全过程

一 下载安装文件 jdk的安装与配置是Java学习的第一步&#xff0c;下面记录一下具体过程。首先根据自己系统下载对应版本。下载地址http://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html 二 安装 双击exe文件&#xff0c;按照默认设置一步一步…

JDK8下载安装及配置环境教程,一看就会

目录 一.下载安装包 二.安装步骤 三.配置环境 一.下载安装包 百度网盘链接:百度网盘 请输入提取码 提取码:QI20 也可以选择自己到ORACLE官网下载:www.oracle.com. 1.点击链接进入官网 可以看到上方有个"Products"&#xff0c;点击它 2.然后可以看到下滑的菜单…