JUC 一 ReentrantLock 可重入锁
java.util.concurrent.locks
ReentrantLock
即可重入锁,实现了Lock
和Serializable
接口
ReentrantLock
和synchronized
都是可重入锁。
基本原理
ReentrantLock的基本原理:
底层依赖AbstractQueuedSynchronizer。
先通过CAS尝试获取锁。
如果此时已经有线程占据了锁,那就加入CLH队列并且被挂起。
当锁被释放之后,排在CLH队列队首的线程会被唤醒,然后CAS再次尝试获取锁。
非公平锁:
如果同时还有另一个线程进来尝试获取,那么有可能会让这个线程抢先获取
公平锁:
如果同时还有另一个线程进来尝试获取,就会排到队尾,由队首的线程获取到锁
可重入锁
同步方法调用同一对象另一个同步方法,可以重复获得此对象的锁,又叫递归锁:
- ReentrantLock和synchronized都是可重入的。
- synchronized因为可重入因此可以放在被递归执行的方法上,且不用担心线程最后能否正确释放锁;
- ReentrantLock需要手动加锁和解锁,且解锁的操作尽量要放在finally代码块中,保证线程正确释放锁。
- ReentrantLock在重入时要确保
重复获取锁的次数
必须和重复释放锁的次数
一样,否则可能导致其他线程无法获得该锁。
显示锁
synchronized被称之为隐式锁,Lock被称之为显示锁
显示锁:
- 即需要显示通过lock()方法上锁,且必须显示通过unlock()方法进行释放锁。
- 通常为了保证锁的释放,将unlock()方法放在finally中。
公平锁
ReentrantLock
构造函数中提供了两种锁:创建公平锁
和非公平锁(默认)
- 在公平的锁上,线程按照他们发出请求的顺序获取锁,如果有另一个线程持有锁或者有其他线程在等待队列中等待这个锁,那么新发出的请求的线程将被放入到队列中。
- 在非公平锁上,则允许插队(插队失败,则放入队尾排队)
- 非公平锁性能高于公平锁性能(JVM 会自动计算如何处理更快速来调度插队)
- 当持有锁的时间相对较长或者请求锁的平均时间间隔较长,应该使用公平锁。
public ReentrantLock() {
sync = new NonfairSync();
}
public ReentrantLock(boolean fair) {
sync = fair ? new FairSync() : new NonfairSync();
}
Lock接口
Modifier and Type | Method | Description |
---|---|---|
void | lock() | 获取锁 |
void | lockInterruptibly() | 当前线程没有被中断时获取锁(中断抛出中断异常) |
Condition | newCondition() | 返回绑定到此Lock实例的新Condition实例 |
boolean | tryLock() | 只有在调用时它是空闲的才能获取锁 |
boolean | tryLock(long time, TimeUnit unit) | 如果在给定的等待时间内空闲并且当前线程未被中断,则获取锁 |
void | unlock() | 释放锁 |
tryLock、lock 和 lockInterruptibly 的区别
tryLock
能获得锁就返回true
,不能就立即返回false
。tryLock(long timeout,TimeUnit unit)
,可以增加时间限制,如果超过该时间段还没获得锁,返回false
。lock
能获得锁就返回true
,不能的话一直等待
获得锁。lock
和lockInterruptibly
:如果两个线程分别执行这两个方法,但此时中断这两个线程(thread.interrupt()),lock
优先考虑获取锁,待获取锁成功后,才响应中断,不会抛出异常。而lockInterruptibly
会立即中断,抛出异常。(当一个线程获取了锁之后,是不会被interrupt()方法中断的)
Condition接口
Condition
是一个多线程协调通信的工具类,可以让某些线程一起等待某个条件(condition)
,只有满足条件时,线程才会被唤醒。
synchronized
结合Object
上的wait
和notify
方法可以实现线程间的等待通知机制。
ReentrantLock
结合Condition
接口同样可以实现这个功能。
Condition
的强大之处在于它可以为多个线程间建立不同的Condition
, 使用synchronized
wait()
只有一个阻塞队列,notifyAll
会唤起所有阻塞队列下的线程,而使用lock
condition
,可以实现多个阻塞队列,signalAll
只会唤起某个阻塞队列下的阻塞线程。
Condition由ReentrantLock对象创建,并且可以同时创建多个
static Condition notEmpty = lock.newCondition();
static Condition notFull = lock.newCondition();
Condition接口在使用前必须先调用ReentrantLock的lock()方法获得锁。
之后调用Condition接口的await()将释放锁,并且在该Condition上等待,
直到有其他线程调用Condition的signal()方法唤醒线程。
使用方式和wait,notify类似。
Condition的signal()方法,将会唤醒在等待队列中等待时间最长的节点(首节点)
JUC 一 ReentrantLock 可重入锁的更多相关文章
- ReenTrantLock可重入锁(和synchronized的区别)总结
ReenTrantLock可重入锁(和synchronized的区别)总结 可重入性: 从名字上理解,ReenTrantLock的字面意思就是再进入的锁,其实synchronized关键字所使用的锁也 ...
- ReentrantLock可重入锁的理解和源码简单分析
import java.util.concurrent.TimeUnit; import java.util.concurrent.locks.ReentrantLock; /** * @author ...
- ReenTrantLock可重入锁和synchronized的区别
ReenTrantLock可重入锁和synchronized的区别 可重入性: 从名字上理解,ReenTrantLock的字面意思就是再进入的锁,其实synchronized关键字所使用的锁也是可重入 ...
- ReentrantLock可重入锁——源码详解
开始这篇博客之前,博主默认大家都是看过AQS源码的~什么居然没看过猛戳下方 全网最详细的AbstractQueuedSynchronizer(AQS)源码剖析(一)AQS基础 全网最详细的Abstra ...
- java ReentrantLock可重入锁功能
1.可重入锁是可以中断的,如果发生了死锁,可以中断程序 //如下程序出现死锁,不去kill jvm无法解决死锁 public class Uninterruptible { public static ...
- ReentrantLock——可重入锁的实现原理
一. 概述 本文首先介绍Lock接口.ReentrantLock的类层次结构以及锁功能模板类AbstractQueuedSynchronizer的简单原理,然后通过分析ReentrantLock的lo ...
- 40 多线程(十二)——ReentrantLock 可重入锁
我们使用的synchronized加的锁是可以延续使用的,如下: public void test() { //第一次获得锁 synchronized(this) { while(true) { // ...
- ReentrantLock可重入锁、公平锁非公平锁区别与实现原理
ReentrantLock是lock接口的一个实现类,里面实现了可重入锁和公平锁非公平锁 ReentrantLock公平锁和不公平锁实现原理 公平锁会获取锁时会判断阻塞队列里是否有线程再等待,若有获取 ...
- ReentrantLock可重入锁的使用场景(转)
摘要 从使用场景的角度出发来介绍对ReentrantLock的使用,相对来说容易理解一些. 场景1:如果发现该操作已经在执行中则不再执行(有状态执行) a.用在定时任务时,如果任务执行时间可能超过下次 ...
随机推荐
- 基于React Native的跨三端应用架构实践
作者|陈子涵 编辑|覃云 “一次编写, 到处运行”(Write once, run anywhere ) 是很多前端团队孜孜以求的目标.实现这个目标,不但能以最快的速度,将应用推广到各个渠道,而且还能 ...
- SQL关于:警告: 聚合或其他 SET 操作消除了空值。
方法一: create table tb ( id int, num int ) insert into tb select 1,10 insert into tb select 1,20 inser ...
- centos7-关闭 rpcbind 服务
1.关闭 rpcbind 服务 sudo systemctl disable rpcbind 2.关闭开机自启动 sudo systemctl disable rpcbind 3.立即执行关闭 sy ...
- JavaSE---枚举
1.概述 1.1 某些情况下,一个类的对象是 有限且固定的,eg:四季... 1.2 手动实现枚举类: 1.1.1 私有化构造器 1.1.2 将类的属性用private final修饰: 将类的实例 ...
- 【原理】RabbitMQ概要图
使用流程 发布者(推送消息的一端): 创建一个tcp长连接connection,连接rabbitmq的监听端口5672: 在TCP长连接下创建一个信道channel,信道可以理解为connection ...
- 管理mysql
要管理MySQL,可以使用可视化图形界面MySQL Workbench. MySQL Workbench可以用可视化的方式查询.创建和修改数据库表,但是,归根到底,MySQL Workbench是一个 ...
- js和php语法区别
参考 : https://www.wangjingxian.cn/php/51.html
- AutoCAD2016简体中文破解版32位64位下载
AutoCAD2016序列号:666-69696969 667-98989898 400-45454545 066-66666666(任意一个) AutoCAD2016产品密钥:001H1 AutoC ...
- JAVA调用R脚本 windwos路径下
RConnection c = new RConnection();// REXP x = c.eval("source('D:\\\\jiaoben\\\\RJava_test.R',en ...
- Java-框架-Dubbo:Dubbo
ylbtech-Java-框架-Dubbo:Dubbo Dubbo是阿里巴巴公司开源的一个高性能优秀的服务框架,使得应用可通过高性能的 RPC 实现服务的输出和输入功能,可以和Spring框架无缝集成 ...