AQS与Synchronized异曲同工的加锁流程

news2025/1/4 20:26:42

在并发多线程的情况下,为了保证数据安全性,一般我们会对数据进行加锁,通常使用Synchronized或者ReentrantLock同步锁。Synchronized是基于JVM实现,而ReentrantLock是基于Java代码层面实现的,底层是继承的AQS。

AQS全称AbstractQueuedSynchronizer,即抽象队列同步器,是一种用来构建锁和同步器的框架。它维护了一个共享资源state(volatile修饰) + 一个双向链表结构的同步队列 + 一个单链表结构的条件队列,底层利用了CAS机制来保证操作的原子性

AQS通过控制state变量(volatile的int类型)的值来判断锁的状态,对于不可重入锁state不是0则阻塞;对于可重入锁如果state=0则执行抢占,非0则判断当前线程是否是占有锁的线程,是就把state加1,比如重入5次,那么state加1执行5次,可重入锁在释放锁的时候,同样需要释放5次直到state=0其他线程才有资格获得锁。

我们常见的并发锁ReentrantLock、CountDownLatch、Semaphore、CyclicBarrier都是基于AQS实现的。

仔细研究AQS底层的加锁原理,其实跟Synchronized的加锁原理有惊人的相似。

Synchronized的加锁流程

这里直接讨论重量级锁,忽略锁升级的过程(对锁升级过程感兴趣可移步Java对synchronized锁的实现与优化),重量级锁底层使用操作系统的互斥锁来实现。

考虑Synchronized的加锁流程,Synchronized的对象锁要能满足以下几个场景:

  1. 多个线程执行到Synchronized代码块,只有一个线程获取锁,然后执行同步代码块(需要记录哪个线程获取了对象锁)。
  2. 其他线程被阻塞(被阻塞的线程,是不是可以设计一个阻塞队列)
  3. 持有锁的线程调用wait方法释放锁,等待被唤醒(等待的线程,是不是可以设计一个等待队列)
  4. 被阻塞的线程开始竞争锁
  5. 调用notify方法,唤醒等待的线程,被唤醒的线程进入阻塞队列,一块竞争锁

上面描述了Synchronized的加锁流程,Synchronized的对象锁存储结构如下图所示。

对象锁的基本工作机制:

  1. 当多个线程同时访问一段同步代码时,首先会进入 _EntryList队列中阻塞
  2. 当某个线程获取到对象的对象锁后进入临界区域,并把对象锁中的_owner变量设置为当前线程,即获得对象锁
  3. 若持有对象锁的线程调用wait() 方法,将释放当前持有的对象锁,_owner变量恢复为null,同时该线程进入_WaitSet 集合中等待被唤醒
  4. 在WaitSet集合中的线程被唤醒,会被再次放到EntryList队列中,重新竞争获取锁
  5. 若当前线程执行完毕也将释放对象锁并复位变量的值,以便其他线程进入获取

可以发现,Synchronized对象锁存储结构完美符合最初设想时它需要满足的场景。

AQS的加锁流程

照例,先分析使用AQS的加锁需求:

  1. 多个线程执行到acquire方法的时候,只有一个线程获取锁然后执行同步代码块(需要记录哪个线程获取了对象锁)
  2. 其他线程被阻塞(被阻塞的线程,是不是可以设计一个阻塞队列)
  3. 持有锁的线程调用await方法,释放锁,等待被唤醒(等待的线程,是不是可以用设计个等待队列)
  4. 被阻塞的线程开始竞争锁
  5. 调用signal方法,唤醒等待的线程,被唤醒的线程进入阻塞队列,一块竞争锁

AQS的加锁需求跟Synchronized是一样的。AQS实际的加锁机制是怎么设计的呢?如下图所示。

参考上图,可以发现AQS的加锁流程并不复杂,只要理解了同步队列和条件队列,以及它们之间的数据流转,就算彻底理解了AQS。

  1. 当多个线程竞争AQS锁时,如果有个线程获取到锁,就把ower线程设置为自己
  2. 没有竞争到锁的线程,在同步队列中阻塞(同步队列采用双向链表,CAS尾插)
  3. 持有锁的线程调用await方法,释放锁,追加到条件队列的末尾(条件队列采用单链表,尾插)
  4. 持有锁的线程调用signal方法,唤醒条件队列的头节点,并转移到同步队列的末尾
  5. 同步队列的头节点优先获取到锁

可以看到AQS和Synchronized的加锁流程几乎是一模一样的,AQS中同步队列就是Synchronized中EntryList,AQS中条件队列就是Synchronized中的waitSet,两个队列之间的数据转移流程也是一样的,阻塞状态的线程被放到同步队列中,等待状态的线程被放到条件队列中,从条件队列唤醒的线程又被转移到同步队列末尾,一块竞争锁。

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

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

相关文章

c++函数对象(仿函数)、谓词、内建函数对象

1、函数对象 1.1 概念 重载函数调用操作符的类,这个类的对象就是函数对象,在使用这个函数对象对应使用重载的()符号时,行为类似于函数调用,因此这个函数也叫仿函数。 注意:函数对象&#xff0…

多个任务并行的时候,你是否总是会手忙脚乱?

很多重要事情之所以变得迫在眉睫,需要立刻处理、应付,是因为被延误或没有进行足够的预防和准备,筹划。 面对多个任务并行的时候,你是否总是会手忙脚乱? 在项目工作中,管理者每天要面对各种工作&#xff…

移动WEB开发二、流式布局

零、文章目录 文章地址 个人博客-CSDN地址:https://blog.csdn.net/liyou123456789个人博客-GiteePages:https://bluecusliyou.gitee.io/techlearn 代码仓库地址 Gitee:https://gitee.com/bluecusliyou/TechLearnGithub:https:…

【Linux】线程函数和线程同步详细整理(金针菇般细)

目录 一,线程函数 1.获取当前线程ID 2.创建线程 3.退出线程 4.阻塞线程 5.分离线程 6.取消线程 7.线程比较 8.测试代码(线程函数总结) 二,线程同步 1.互斥锁 2.读写锁 3.条件变量 4.信号量 一,线程函数 …

【阿旭机器学习实战】【29】产品广告投放实战案例---线性回归

【阿旭机器学习实战】系列文章主要介绍机器学习的各种算法模型及其实战案例,欢迎点赞,关注共同学习交流。 目录问题描述数据处理过程及源码通过数据可视化分析数据训练线性回归模型可视化训练好的线性回归模型结果预测问题描述 你所在的公司在电视上做产…

mybatis狂神(附自学过程中疑问解决)

首先先附上mybatis的官方文本链接mybatis – MyBatis 3 | 简介一、Mybatis介绍MyBatis 是一款优秀的持久层框架,它支持自定义 SQL、存储过程以及高级映射。MyBatis 免除了几乎所有的 JDBC 代码以及设置参数和获取结果集的工作。MyBatis 可以通过简单的 XML 或注解来…

Comparator和Comparable的区别以及Collections.sort排序原理

一、概述 Comparable和Comparator都是两个接口,接口都可以用来实现集合中元素的比较、排序,Comparator位于包java.util下,而Comparable位于包java.lang下,Comparable接口将比较代码嵌入自身类中,而Comparator既可以嵌…

非标题党:前端Vue React 项目编程规范化配置(大厂规范)

前端项目编程规范化配置 下述例子主要是从 代码规范化 以及 git 提交规范化 两方面进行配置。内容很多,请做好心理准备 一、代码检测工具 ESLint 在我们通过 vue create “项目名” 时,我们可以通过手动配置的方式,来配置 ESLint 来对代码进…

QDateTime的11种显示方式

QDateTime datetime QDateTime::currentDateTime(); datetime.toString(“hh:mm:ss\nyyyy/MM/dd”); datetime.toString(“hh:mm:ss ap\nyyyy/MM/dd”); datetime.toString(“hh:mm:ss\nyyyy-MM-dd”); datetime.toString(“hh:mm:ss ap\nyyyy-MM-dd”); datetime.to…

【分享】订阅用友YonSuite集简云连接器同步销售出库数据至用友YonSuite

方案场景 在企业中因多种系统孤立导致数据割裂,是现企业中现阶段面临的最大问题,而钉钉作为常用的OA审批系统,用友YonSuite作为ERP系统,原方式钉钉内完成审批再由人工将数据同步到用友YonSuite系统,数据同步过程中不仅…

将HTTP接口配置成HTTPS

一、使用Java的keytool.exe程序生成本机的TLS许可找到Java的jdk目录进入bin默认安装路径C:\Program Files\Java\jdk1.8.0_91\bin 进入命令面板,在bin的路径栏中输入cmd敲击回车即可使用keytoolkeytool -genkeypair -alias tomcat_https -keypass 123456 -keyalg RSA…

linux线程的基本知识

这里用的是Linux的pthread线程库,需要加pthread线程库。 线程的创建 第一个参数是线程id的地址。第二个参数是线程属性,一般为NULL。第三个是要执行的函数。第四个是函数的参数,一般也为NULL 线程的等待,第一个参数是线程的id,第…

VBA提高篇_27 OptionBOX_CheckBox_Frame_Image_VBA附加控件

文章目录1.单选按钮OptionBOX:2.复选框CheckBox:3.框架Frame:4.图像Image: (loadPictrue)5. VBA附加控件:6. 适用于很多控件的重要属性:1.单选按钮OptionBOX: 默认时,同一窗体的所有单选按钮均属于同一组,只能选中一个 可通过Frame控件进行分组解决. 2.复选框CheckBox: 一次可以…

备考软考系统分析师-1

系统分析师教程网盘资源:链接: https://pan.baidu.com/s/1ekHuCJJ3o5RrW1xeMkxhdA 提取码: 6666 信息系统战略规划 信息系统开发方法: 结构化法 瀑布模型 原型法 自顶向下 用于需求阶段较多 面向对象 自底向上 面向服务的方法 系统建模 政府信息…

在Excel接入 chatgtp (图文教学)

效果图 话不多说,开始教学 首先点击插入,然后点击获取加载项 office Excel 加载加载项时出错 解决办法_long_songs的博客-CSDN博客今天在添加维基百科的时候,怎么都添加不了,网上的办法都是关闭,重启,或者…

[架构之路-114]-《软考-系统架构设计师》-软件架构设计-7-软件架构评估

前言第7节 软件架构评估7.1 什么是架构评估/为什么要软件架构评估在软硬件系统总体架构设计完成之后,为保证架构设计的合理性、完整性和针对性,从根本上保证系统质量,降低成本及投资风险,需要对总体架构进行评估。7.2 软件架构评估…

word高效技巧:几个快速填充表格的操作方法

我们办公人员在面对大量表格数据的时候,都希望以最简便、快捷的方式完成对数据的填充、美化等整理工作。比如,日常工作中几种常用的Word表格填充类型:1. 填充序号;2. 填充文本;3. 填充颜色。因此,接下来给大…

你了解互联网APP推荐的背后逻辑么(下)?

上篇重点介绍了互联网APP在搜索交互场景下的通用逻辑,让大众对每天离不开的搜索进行了一个普遍介绍。这一篇,我们来聊聊抖音、头条等APP划一划这个动作背后,是怎么做推荐的。推荐的背后,离不开每个用户的数据,而且这个…

谈谈Linux内核的噪声

Linux内核是广被使用的操作系统,从嵌入式家用设备,航空航天设备到超级计算机,到处都有Linux内核的身影,这归功于Linux内核丰富的配置带来的巨大灵活性。 网络虚拟化和软件定义网络的发展,也从另外一个方面证实了在网络…

Java线程的6中状态

Java 线程的状态 Java线程有六种状态: 初始(NEW)、运行(RUNNABLE)、阻塞(BLOCKED)、 等待(WAITING)、超时等待(TIMED_WAITING)、终止&#xff08…