面试之ReentrantLock

这篇具有很好参考价值的文章主要介绍了面试之ReentrantLock。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一,ReentrantLock

1.ReentrantLock是什么?

ReentrantLock实现了Lock接口,是一个可重入且独占式的锁,和Synchronized关键字类似,不过ReentrantLock更灵活,更强大,增加了轮询、超时、中断、公平锁和非公平锁等高级功能。

面试之ReentrantLock,面试,java,开发语言,java-ee

ReentrantLock 里面有一个内部类 ,Sync 继承 AQS(AbstractQueuedSynchronizer),添加锁和释放锁的大部分操作实际上都是在 Sync中实现的。Sync 有公平锁 FairSync 和非公平锁 NonfairSync 两个子类。

2.公平锁与非公平锁

  • 公平锁:先到先得原则。如果锁被线程释放之后,先申请的线程先得到锁。公平锁性能较差一些,因为公平锁为了保证事件上的绝对顺序,上下文切换频繁
  • 非公平锁:相对公平锁来说,如果锁被线程释放之后,后续所有申请的线程都有可能获得到锁,不会按照某一个顺序来,是随机的。非公平锁性能更高,但是可能导致某些线程永远获取不到锁

3.ReentrantLock的使用

lock(): 加锁 , 如果获取不到锁就死等 .
trylock( 超时时间 ): 加锁 , 如果获取不到锁 , 等待一定的时间之后就放弃加锁 .
unlock(): 解锁

4.ReentrantLock与Synchronized区别

  • Sychronized依赖于JVM,而ReentrantLock依赖于API:Synchronized 关键字进行了很多优化,但是这些优化都是在虚拟机层面实现的,并没有直接暴露给我们。ReentrantLock 是 JDK 层面实现的(也就是 API 层面,需要 lock() 和 unlock() 方法配合 try/finally 语句块来完成),所以我们可以通过查看它的源代码,来看它是如何实现的
  • Sychronized使用时不需要手动释放锁,ReentrantLock使用时需要手动释放锁,使用起来更加的灵活,但是也容易遗漏unlock
  • Sychronized在申请锁失败时,会死等知道获取到锁;ReentrantLock可以通过trylock的方式等待一段时间之后就放弃等待
  • Synchronized是非公平锁;ReentrantLock默认是非公平锁,可以通过构造方法传入一个true开启公平锁模式
  • 更强大的唤醒机制 . synchronized 是通过 Object wait / notify 实现等待 - 唤醒 . 每次唤醒的是一个随机等待的线程. ReentrantLock 搭配 Condition 类实现等待 - 唤醒 , 可以更精确控制唤醒某个指定的线程.
  • ReentrantLock提供了一种能够中断等待锁的线程的机制,通过 lock.lockInterruptibly() 来实现这个机制。也就是说正在等待的线程可以选择放弃等待,改为处理其他事情。

5.可中断锁和不可中断锁有什么区别

  • 可中断锁:获取锁的过程中可以被中断,不需要一直等到获取锁之后 才能进行其他逻辑处理。ReentrantLock 就属于是可中断锁。
  • 不可中断锁:一旦线程申请了锁,就只能等到拿到锁以后才能进行其他的逻辑处理。 synchronized 就属于是不可中断锁

6.如何选择使用哪个锁?

  • 锁竞争不激烈的时候, 使用 synchronized, 效率更高, 自动释放更方便.
  • 锁竞争激烈的时候, 使用 ReentrantLock, 搭配 trylock 更灵活控制加锁的行为, 而不是死等.
  • 如果需要使用公平锁, 使用 ReentrantLock.

二,Atomic原子类

1.什么是原子类

Atomic 翻译成中文是原子的意思。在化学上,我们知道原子是构成一般物质的最小单位,在化学反应中是不可分割的。在我们这里 Atomic 是指一个操作是不可中断的。即使是在多个线程一起执行的时候,一个操作一旦开始,就不会被其他线程干扰。

所以,所谓原子类就是具有原子操作特征的类

根据操作的数据类型,可以将JUC包中的原子类分为四类:

  1. 基本类型
    1. AtomicInteger:整型原子类
    2. AtomicLong:长整型原子类
    3. AtomicBoolean:布尔型原子类
  2. 数组类型
    1. AtomicIntegerArray:整型数组原子类
    2. AtomicLongArray:长整型数组原子类
    3. AtomicReferenceArray:引用类型数组原子类
  3. 引用类型
    1. AtomicReference:引用类型原子类
    2. AtomicMarkableReference:原子更新带有标记的引用类型。该类将 boolean 标记与引用关联起来。
    3. AtomicStampedReference:原子更新带有版本号的引用类型。该类将整数值与引用关联起来,可用于解决原子的更新数据和数据的版本号,可以解决使用 CAS 进行原子更新时可能出现的 ABA 问题。
  4. 对象的属性修改类型
    1. AtomicIntegerFieldUpdater:原子更新整型字段的更新器
    2. AtomicLongFieldUpdater:原子更新长整型字段的更新器
    3. AtomicReferenceFieldUpdater:原子更新引用类型里的字段

2.基本类型原子类

以AtomicInteger为例:常用方法有

public final int get() //获取当前的值
public final int getAndSet(int newValue)//获取当前的值,并设置新的值
public final int getAndIncrement()//获取当前的值,并自增
public final int getAndDecrement() //获取当前的值,并自减
public final int getAndAdd(int delta) //获取当前的值,并加上预期的值
boolean compareAndSet(int expect, int update) //如果输入的数值等于预期值,则以原子方式将该值设置为输入值(update)
public final void lazySet(int newValue)//最终设置为newValue,使用 lazySet 设置之后可能导致其他线程在之后的一小段时间内还是可以读到旧的值。

3.基本数据类型优势

通过一个简单例子带大家看一下基本数据类型原子类的优势

1、多线程环境不使用原子类保证线程安全(基本数据类型)

class Test {
        private volatile int count = 0;
        //若要线程安全执行执行count++,需要加锁
        public synchronized void increment() {
                  count++;
        }

        public int getCount() {
                  return count;
        }
}

 2、多线程环境使用原子类保证线程安全(基本数据类型)

class Test2 {
        private AtomicInteger count = new AtomicInteger();

        public void increment() {
                  count.incrementAndGet();
        }
      //使用AtomicInteger之后,不需要加锁,也可以实现线程安全。
       public int getCount() {
                return count.get();
        }
}

AtomicInteger 类主要利用 CAS (compare and swap) + volatile 和 native 方法来保证原子操作,从而避免 synchronized 的高开销,执行效率大为提升。

三,CSA

1.什么是CAS?

CAS即compare and swap(比较与交换),它的主要思想很简单:就是用一个预期值和一个要更新变量的值进行比较,两值相等才会将新的值写入,否则不会操作成功。

CAS 是一个原子操作,底层依赖于一条 CPU 的原子指令实际上Java的CAS利用的是unsafe这个类提供的CAS操作;unsafe的CAS依赖于JVM针对于不同的操作系统实现Atomic::cmpxchg;Atomic::cmpxchg的实现使用了汇编语言的CAS操作,并使用CPU提供的lock机制保证其原子性。简而言之,有了硬件层面的支持,软件层面才可以做到

2.CAS操作

CAS涉及三个操作数:

  • V:要更新的变量
  • E:预期值
  • N:拟写入的新值

当且仅当V的值等于E的值时,CAS通过原子方式用新值N来更新V的值,如果不等,说明已经有其他线程更新了V,则当前线程放弃更新。

3.CAS的ABA问题

如果一个变量 V 初次读取的时候是 A 值,并且在准备赋值的时候检查到它仍然是 A 值,那我们就能说明它的值没有被其他线程修改过了吗?很明显是不能的,因为在这段时间它的值可能被改为其他值,然后又改回 A,那 CAS 操作就会误认为它从来没有被修改过。这个问题被称为 CAS 操作的 "ABA"问题。

面试之ReentrantLock,面试,java,开发语言,java-ee

对于这个问题,我们解决的方法是:给要修改的值,引入版本号,在CAS比较当前值和预期值相同的同时也需要比较版本号是否符合预期

  • CAS 操作在读取旧值的同时, 也要读取版本号.
  • 真正修改的时候,
    • 如果当前版本号和读到的版本号相同, 则修改数据, 并把版本号 + 1.
    • 如果当前版本号高于读到的版本号. 就操作失败(认为数据已经被修改过了)

四,AQS

1.AQS是什么?

AQS 的全称为 AbstractQueuedSynchronizer ,翻译过来的意思就是抽象队列同步器。这个类在 java.util.concurrent.locks 包下面。AQS就是一个抽象类,主要用来构建锁和同步器。AQS 为构建锁和同步器提供了一些通用功能的实现,因此,使用 AQS 能简单且高效地构造出应用广泛的大量的同步器,比如我们提到的ReentrantLock,Semaphore等

2.AQS的原理是什么

AQS的核心思想就是:如果被请求的共享资源空闲,则将当前请求资源的线程设置为有效的工作线程,并将共享资源设置为锁定状态;如果被请求的共享资源被占用,那么就需要一套线程阻塞等待以及被唤醒时锁分配的机制,这个机制AQS使用CLH锁队列实现的,即将暂时获取不到锁的线程加载到队列当中。

CLH队列是一个虚拟的双向队列(虚拟的双向队列即不存在队列实例,仅存在节点之间的关联关系)。AQS是将每一个请求共享资源的线程封装成一个CLH锁队列的一个结点来实现锁的分配。在CLH同步队列中,一个结点表示一个线程,它保存着线程的引用,当前结点在队列中的状态,前驱结点,后继结点。

面试之ReentrantLock,面试,java,开发语言,java-ee

AQS的核心原理图:面试之ReentrantLock,面试,java,开发语言,java-ee 

AQS使用int成员变量state表示同步状态 ,通过内置的线程等待队列来完成获取资源线程的排队工作。state变量由volatile修饰,用于展示当前临界资源获锁情况。

// 共享变量,使用volatile修饰保证线程可见性
private volatile int state;

以ReentrantLock为例,state初始值为0,表示资源未锁定状态 。此时线程A调用lock()时,底层会调用tryAcquire()方法独占该锁并将state + 1,此后,其他线程想要再次tryAcquire()时都会失败,直到A线程unlock()之后,state = 0,其他线程才有机会获取该锁。ReentrantLock是一个可重入锁,加锁多少次就需要解锁多少次,直到state = 0,此时证明资源无锁状态,其他线程均可获取。

CountDownLatch 以例,任务分为 N 个子线程去执行,state 也初始化为 N(注意 N 要与线程个数一致)。这 N 个子线程是并行执行的,每个子线程执行完后countDown() 一次,state 会 CAS(Compare and Swap) 减 1。等到所有子线程都执行完后(即 state=0 ),会 unpark() 主调用线程,然后主调用线程就会从 await() 函数返回,继续后余动作

3.Semaphore 有什么用?

synchronized 和 ReentrantLock 都是一次只允许一个线程访问某个资源,而Semaphore(信号量)可以用来控制同时访问特定资源的线程数量

Semaphore简单使用:在某时刻同时存在N(N > n)个线程来获取Semaphore中的共享资源,但是Semaphore在创建实例对象时传入参数来设置同一时刻只能有n个对象访问,其他线程都会阻塞等待,直到有线程释放资源之后,其他线程才可以获取,但是只要同时访问线程数量为n,其他线程就会阻塞

 例:

// 初始共享资源数量
final Semaphore semaphore = new Semaphore(5);
// 获取1个许可
semaphore.acquire();
// 释放1个许可
semaphore.release();

此时只允许同时5个线程访问。用我们生活中的例子来讲:一个加油站同时只能有5辆车加油,其他车辆来到只能等待,只有其他车辆完成加油之后,这个位置空闲,后续车辆才能加油!

当初始资源数为1的时候,Semaphore为排他锁!

Semaphore有两种模式:

  • 公平模式:调用acquire()方法的顺序就是获取许可证的顺序,遵循FIFO,
  • 非公平模式:抢占式执行

两种模式对应两个构造方法: 两个构造方法必须提供许可证数量,第二个方法可以指定是否为公平模式,默认是非公平模式

public Semaphore(int permits) {
  	sync = new NonfairSync(permits);
}

public Semaphore(int permits, boolean fair) {
  	sync = fair ? new FairSync(permits) : new NonfairSync(permits);
}

4.Semaphore的原理是什么

Semaphore是共享锁的一种实现,它默认构造AQS的state为permits,可以将permits的值看作许可证的数量,只有拿到许可证的线程才能执行。

调用Semaphore.acquire(),线程尝试获取许可证,如果state >= 0的话,则表示获取成功,如果获取成功,则进程CAS操作去修改state的值 state = state - 1;如果state  < 0 时,则表示许可证数量不足,此时将线程封装成一个结点加入到阻塞队列当中,挂起线程。

调用Semaphore.release(); ,线程尝试释放许可证,并使用 CAS 操作去修改 state 的值 state=state+1。释放许可证成功之后,同时会唤醒同步队列中的一个线程。被唤醒的线程会重新尝试去修改 state 的值 state=state-1 ,如果 state>=0 则获取令牌成功,否则重新进入阻塞队列,挂起线程

5. CountDownLatch有什么用?

CountDownLatch允许count个线程阻塞在一个地方,直至所有线程的任务执行完毕才结束

CountDownLatch是一次性的,计数器的值只能在构造方法中初始化一次,之后没有任何机制再对其设置值,当CountDownLatch使用完毕之后,不能再被使用。

6.CountDownLatch 的原理是什么?

CountDownLatch是一种共享锁的实现,它默认构造AQS的state值作为count,当线程使用countDown()方法时,其实使用了tryReleaseShared以CAS的操作来减少state的值直至为0,当调用await()方法时,如果state不等于0,证明还有线程没有执行完毕任务,await()方法一值会阻塞,也就是说await()方法之后的语句不会被执行直到count的个数位0,也就是state = 0,await()方法之后的语句才执行。文章来源地址https://www.toymoban.com/news/detail-653962.html

到了这里,关于面试之ReentrantLock的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Java EE企业级应用开发(SSM)第9章

    1.MyBatis框架的特点 1-1.简单易学 1-2.灵活 1-3.提供映射标签 2.MyBatis核心类 2-1.Configuration 2-2.SqlSessionFactory 2-3.SqlSession 2-4.Exector 2-5.MappedStatement 3.MyBatis工作流程 4.Mybatis入门程序 4-0:导入jar包资源 4-1:编写实体类 4-2:编写mapper接口以及映射文件 4-3:编写mybatis-config配置文件 4-4:编

    2024年02月04日
    浏览(50)
  • Java EE企业级应用开发(SSM)第10章

    1.第九章的细节处理 1-1.mappers标签中的配置 1-2.jdbc属性文件的配置 1-3.包的别名配置 2.Mybatis核心配置文件 2-1:settings标签(P145-146中的表10-1) 2-2.类型别名 3.Mybatis映射文件 3-1:insert、update、delete元素属性 3-2:select元素 3-3:ResultMap详解 ResultMap详解 4.动态SQL 4-1:if 注意模糊查询

    2024年02月03日
    浏览(49)
  • Java EE开发系列教程 - 使用EJB组件与JPA层交互

    因为之前JPA原生 property 属性在Wildfly实现中并不总是生效,所以这里换成了Wildfly默认的JPA实现,即Hibernate。把属性改成了Hibernate专有的。 hibernate.hbm2ddl.auto 定义是否自动生成表, create-and-drop 意为如果表存在,则删除后再创建。 hibernate.dialect 用来指定数据库厂商,以根据不同

    2024年04月13日
    浏览(35)
  • 【多线程面试题十六】、谈谈ReentrantLock的实现原理

    文章底部有个人公众号: 热爱技术的小郑 。主要分享开发知识、学习资料、毕业设计指导等。有兴趣的可以关注一下。为何分享? 踩过的坑没必要让别人在再踩,自己复盘也能加深记忆。利己利人、所谓双赢。 面试官:谈谈ReentrantLock的实现原理 参考答案: ReentrantLock是基

    2024年02月06日
    浏览(60)
  • 面试10000次依然会问的【ReentrantLock】,你还不会?

    引言 在并发编程的世界中,ReentrantLock扮演着至关重要的角色。它是一个实现了重入特性的互斥锁,提供了比synchronized更加灵活的锁定机制。ReentrantLock属于java.util.concurrent.locks包,是Java并发API的一部分。 与传统的synchronized方法或代码块相比,ReentrantLock提供了更丰富的

    2024年02月06日
    浏览(40)
  • 《吊打面试官》从根上剖析ReentrantLock的来龙去脉

    🔥如果感觉博主的文章还不错的话,请👍三连支持👍一下博主哦 🍂 一个人可能走得更快,但是一群人会走得更远! 📝联系方式:wx,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬👀 并发编程 在互联网技术存储方面使用如此广泛,几乎所有的后端技术面试官

    2023年04月21日
    浏览(36)
  • SSM框架的学习与应用(Spring + Spring MVC + MyBatis)-Java EE企业级应用开发学习记录(第五天)MyBatis的注解开发

    ​ 昨天我们深入学习了 MyBatis多表之间的关联映射,了解掌握了一对一关联映射,一对多关联映射,嵌套查询方式以及嵌套结果方式,掌握了缓存机制的一级缓存,二级缓存等概念,也使用了代码进行复现理解 。但是都是基于XML配置文件的方式来实现的,现在我们要学习一下

    2024年02月11日
    浏览(64)
  • SSM框架的学习与应用(Spring + Spring MVC + MyBatis)-Java EE企业级应用开发学习记录(第三天)动态SQL

    昨天我们深入学习了 Mybatis的核心对象SqlSessionFactoryBuilder , 掌握MyBatis核心配置文件以及元素的使用 ,也掌握MyBatis映射文件及其元素的使用。那么今天我们需要掌握的是更加复杂的查询操作。 学会编写MyBatis中动态SQL 学会MyBatis的条件查询操作 学会MyBatis的更新操作 学会MyBati

    2024年02月11日
    浏览(54)
  • Java中的ReentrantLock实现原理

    在并发编程中,线程安全问题一直是非常重要的问题。Java中提供了多种解决线程安全问题的机制,其中一个比较常用的就是ReentrantLock。本文将介绍ReentrantLock的实现原理,从原子性、可见性等方面解释并结合源码分析,以便更好地理解在多线程环境下实现线程安全的过程。

    2024年02月01日
    浏览(44)
  • (第六天)初识Spring框架-SSM框架的学习与应用(Spring + Spring MVC + MyBatis)-Java EE企业级应用开发学习记录

    ​ 昨天我们已经把Mybatis框架的基本知识全部学完,内容有Mybatis是一个半自动化的持久层ORM框架,深入学习编写动态SQL,Mybatis的关联映射,一对一、一对多、多对多、Mybatis的缓存机制,一二级缓存的开启和设置,缓存命中率、如何使用idea链接数据库自动生成pojo类等。我们学

    2024年02月10日
    浏览(67)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包