java.util.concurrent.locks

ReentrantLock即可重入锁,实现了LockSerializable接口

ReentrantLocksynchronized都是可重入锁。

基本原理

ReentrantLock的基本原理:

    底层依赖AbstractQueuedSynchronizer。

    先通过CAS尝试获取锁。

    如果此时已经有线程占据了锁,那就加入CLH队列并且被挂起。

    当锁被释放之后,排在CLH队列队首的线程会被唤醒,然后CAS再次尝试获取锁。

非公平锁:
如果同时还有另一个线程进来尝试获取,那么有可能会让这个线程抢先获取 公平锁:
如果同时还有另一个线程进来尝试获取,就会排到队尾,由队首的线程获取到锁

可重入锁

同步方法调用同一对象另一个同步方法,可以重复获得此对象的锁,又叫递归锁:

  1. ReentrantLock和synchronized都是可重入的。
  2. synchronized因为可重入因此可以放在被递归执行的方法上,且不用担心线程最后能否正确释放锁;
  3. ReentrantLock需要手动加锁和解锁,且解锁的操作尽量要放在finally代码块中,保证线程正确释放锁。
  4. ReentrantLock在重入时要确保重复获取锁的次数必须和重复释放锁的次数一样,否则可能导致其他线程无法获得该锁。

显示锁

synchronized被称之为隐式锁,Lock被称之为显示锁

显示锁:

  1. 即需要显示通过lock()方法上锁,且必须显示通过unlock()方法进行释放锁。
  2. 通常为了保证锁的释放,将unlock()方法放在finally中。

公平锁

ReentrantLock构造函数中提供了两种锁:创建公平锁非公平锁(默认)

  1. 在公平的锁上,线程按照他们发出请求的顺序获取锁,如果有另一个线程持有锁或者有其他线程在等待队列中等待这个锁,那么新发出的请求的线程将被放入到队列中。
  2. 在非公平锁上,则允许插队(插队失败,则放入队尾排队)
  3. 非公平锁性能高于公平锁性能(JVM 会自动计算如何处理更快速来调度插队)
  4. 当持有锁的时间相对较长或者请求锁的平均时间间隔较长,应该使用公平锁。
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 的区别

  1. tryLock能获得锁就返回true,不能就立即返回false
  2. tryLock(long timeout,TimeUnit unit),可以增加时间限制,如果超过该时间段还没获得锁,返回false
  3. lock能获得锁就返回true,不能的话一直等待获得锁。
  4. locklockInterruptibly:如果两个线程分别执行这两个方法,但此时中断这两个线程(thread.interrupt()),lock优先考虑获取锁,待获取锁成功后,才响应中断,不会抛出异常。而lockInterruptibly会立即中断,抛出异常。(当一个线程获取了锁之后,是不会被interrupt()方法中断的)

Condition接口

Condition是一个多线程协调通信的工具类,可以让某些线程一起等待某个条件(condition),只有满足条件时,线程才会被唤醒。

synchronized结合Object上的waitnotify方法可以实现线程间的等待通知机制。

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 可重入锁的更多相关文章

  1. ReenTrantLock可重入锁(和synchronized的区别)总结

    ReenTrantLock可重入锁(和synchronized的区别)总结 可重入性: 从名字上理解,ReenTrantLock的字面意思就是再进入的锁,其实synchronized关键字所使用的锁也 ...

  2. ReentrantLock可重入锁的理解和源码简单分析

    import java.util.concurrent.TimeUnit; import java.util.concurrent.locks.ReentrantLock; /** * @author ...

  3. ReenTrantLock可重入锁和synchronized的区别

    ReenTrantLock可重入锁和synchronized的区别 可重入性: 从名字上理解,ReenTrantLock的字面意思就是再进入的锁,其实synchronized关键字所使用的锁也是可重入 ...

  4. ReentrantLock可重入锁——源码详解

    开始这篇博客之前,博主默认大家都是看过AQS源码的~什么居然没看过猛戳下方 全网最详细的AbstractQueuedSynchronizer(AQS)源码剖析(一)AQS基础 全网最详细的Abstra ...

  5. java ReentrantLock可重入锁功能

    1.可重入锁是可以中断的,如果发生了死锁,可以中断程序 //如下程序出现死锁,不去kill jvm无法解决死锁 public class Uninterruptible { public static ...

  6. ReentrantLock——可重入锁的实现原理

    一. 概述 本文首先介绍Lock接口.ReentrantLock的类层次结构以及锁功能模板类AbstractQueuedSynchronizer的简单原理,然后通过分析ReentrantLock的lo ...

  7. 40 多线程(十二)——ReentrantLock 可重入锁

    我们使用的synchronized加的锁是可以延续使用的,如下: public void test() { //第一次获得锁 synchronized(this) { while(true) { // ...

  8. ReentrantLock可重入锁、公平锁非公平锁区别与实现原理

    ReentrantLock是lock接口的一个实现类,里面实现了可重入锁和公平锁非公平锁 ReentrantLock公平锁和不公平锁实现原理 公平锁会获取锁时会判断阻塞队列里是否有线程再等待,若有获取 ...

  9. ReentrantLock可重入锁的使用场景(转)

    摘要 从使用场景的角度出发来介绍对ReentrantLock的使用,相对来说容易理解一些. 场景1:如果发现该操作已经在执行中则不再执行(有状态执行) a.用在定时任务时,如果任务执行时间可能超过下次 ...

随机推荐

  1. groupby 技术

    分组键可以有很多形式,且类型不必相同: 1.列表或数组,其长度与待分组的轴一样 2.表示DataFrame某个列名的值 3.字典或Series,给出待分组轴上的值与分组名之间的对应关系 4.函数,用于 ...

  2. JSON对象排序并生成URL参数

    1、for in function jsontourl(param) { let params = {}, data=[]; let arr = Object.keys(param).sort(); ...

  3. 【LeetCode】Stack

    [503] Next Greater Element II [Medium] 给一个循环数组,找到离当前元素最近的比它大的元素. Input: [1,2,1] Output: [2,-1,2] Exp ...

  4. 【Luogu】【关卡2-1】简单的模拟(2017年10月)

    任务说明:开始普及组的训练!所谓模拟,就是直接根据题意编写,思维难度简单. 铺地毯 进制转换 多项式输出 机器翻译 排座椅 笨小猴 都是简单模拟题  

  5. python_django_admin

    admin 是Django提供的基于web的管理工具,是系统管理员用于数据的输入,删除和查询的管理工具. 超级管理员在settings中的集成:INSTALLED_APPS=['django.cont ...

  6. Jmeter实现百分比业务比例

    Jmeter实现百分比业务比例   相较于LoadRunner,jmeter在复杂场景方式貌似略有欠缺.前一段时间,想实现一个功能,如有两个采样器a与b,a采样器与b采样器被执行的概率分别为1/4与3 ...

  7. linux中CentOS、Ubuntu、Debian三个版本系统 差别

    Linux有非常多的发行版本,从性质上划分,大体分为由商业公司维护的商业版本与由开源社区维护的免费发行版本. 商业版本以Redhat为代表,开源社区版本则以debian为代表.这些版本各有不同的特点, ...

  8. 【Java架构:基础技术】一篇文章搞掂:Eclipse

    Eclipse中使用SVN 1.打开资源库视图 https://www.cnblogs.com/liangguangqiong/p/7965770.html 一.编辑器方面 格式化取消自动换行:打开E ...

  9. (转)ubuntu下如何安装使用SSH?

    转:http://os.51cto.com/art/201109/291634.htm ubuntu默认并没有安装ssh服务,如果通过ssh链接ubuntu(比如使用securecrt客户端来访问ub ...

  10. 源码编译安装PHP Version 5.6.30------踩了无数坑,重装了十几次服务器才会的,不容易啊!

    1 环境准备 yum install gcc bison bison-devel zlib-devel libmcrypt-devel mcrypt mhash-devel openssl-devel ...