Java并发之ReentrantLock基础(一)

ReentrantLock是Java中java.util.concurrent.locks.Lock的一个实现类,顾名思义它是Java中锁的一种实现,具体一点来说它是Java中一种可重入的独占锁。它与synchronized相比在功能上与之基本相同,但ReentrantLock提供了更为丰富灵活的使用方式,也更加适合复杂的并发场景。但是synchronized是基于JVM层面实现的,而Lock是基于JDK层面实现的。ReentrantLock内部通过内部类实现了AQS框架(AbstractQueuedSynchronizer)的API来实现独占锁的功能。

核心不同对比:

 

1. 类声明

ReentrantLock类

 ReentrantLock的UML图

说明:ReentrantLock 类内部总共存在Sync、NonfairSync、FairSync三个类,NonfairSync与 FairSync类继承自 Sync类,Sync类继承自 AbstractQueuedSynchronizer抽象类。

2. 构造声明

 

ReentrantLock()构造,默认创建非公平策略(false)的ReentrantLock对象,即new ReentrantLock(false)。

ReentrantLock(boolean fair) 构造,fair参数可指定创建公平策略(true)/非公平策略(false)的ReentrantLock对象。

3. 使用方式

a)    ReentrantLock的基本使用

  • 在希望保证线程同步的代码之前显示调用 lock.lock() 尝试获取锁,若被其他线程占用则阻塞;

  • 在希望保证线程同步的代码执行完毕之后,需要手动释放锁lock.unlock(),否则会造成死锁(一般来讲把释放锁的逻辑放在需要线程同步的代码包装外的finally块中,以确保锁的释放);

  • 有N个lock.lock()就要有M(M=N)个lock.unlock(),若出现N>M则会造成死锁,若出现N<M< span="">则会抛出IllegalMonitorStateException异常;

b)   加锁和解锁

  • 加锁:

  • 通过ReentrantLock的加锁方法Lock进行加锁操作;

  • 会调用到内部类Sync的Lock方法,由于Sync#lock是抽象方法,根据ReentrantLock初始化选择的公平锁和非公平锁,执行相关内部类的Lock方法,本质上都会执行AQS的Acquire方法;

  • AQS的Acquire方法会执行tryAcquire方法,但是由于tryAcquire需要自定义同步器实现,因此执行了ReentrantLock中的tryAcquire方法,由于ReentrantLock是通过公平锁和非公平锁内部类实现的tryAcquire方法,因此会根据锁类型不同,执行不同的tryAcquire;

  • tryAcquire是获取锁逻辑,获取失败后,会执行框架 AQS的后续逻辑,跟ReentrantLock自定义同步器无关。

  • 解锁:

  • 通过 ReentrantLock的解锁方法 Unlock进行解锁;

  • Unlock会调用内部类 Sync的 Release方法,该方法继承于AQS;

  • Release中会调用 tryRelease方法,tryRelease需要自定义同步器实现,tryRelease只在ReentrantLock中的Sync实现,因此可以看出,释放锁的过程,并不区分是否为公平锁;

  • 释放成功后,所有处理由AQS框架完成,与自定义同步器无关。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值