JUC - ReentrantLock 的基本用法 以及 lock()、tryLock()、lockInterruptibly()的区别
ReentrantLock 与 synchronized对比
最近有在阅读Java并发编程实战这本书,又看到了ReentrantLock和synchronized的对比,发现自己以前对于RenntrantLock的理解很片面,特此做一番总结,如果有总结不到位的,欢迎指出
java.util.concurrent.locks
接口 Lock
为什么需要Lock?
java.util.concurrent.locks
接口 Lock
Lock提供了一种如条件的、可轮询的、定时的以及可以终端的获取锁的操作,所有的加锁方式和解锁方式都是显式的。
public class Lock {
private boolean locked = false;
public Lock() {
}
public final synchronized void lock() throws InterruptedException {
while(this.locked) {
this.wait();
}
this.locked = true;
}
public final synchronized void unlock() {
this.locked = false;
this.notifyAll();
}
}
Lock是JAVA5.0出现的,它的出现并不是为了替代synchronized,而是在synchronized不适用的时候使用。
那么synchronized有什么局限性呢?
- 无法中断一个正在获取锁的线程
当一个线程想获取已经被其他线程持有的锁时,就会发生堵塞,假设已经持有锁的线程一直不释放锁,那么线程就会一直等待下去。
- 无法指定获得锁的等待时间
比如,想要A线程执行某个操作,想在指定时间内A线程没有获取到锁就返回。synchronized是做不到的。
相同点:
- 独占锁: 一次只允许一个线程访问
- 可重入锁: 一个线程可重复获得自己已获得锁,不会发生死锁。简单来说,递归的时候不会发生死锁
不同点:
- Lock不是java内置的,synchronized是JVM内置的,因此是内置特性。
- 释放锁的方式:
- Lock 必须要在finally中手动释放锁
- synchronized 会根据锁区域代码自动执行完毕,或者发生异常,JVM会自动释放锁
- 公平:
- Lock是可公平可不公平锁
- synchronized是不公平锁
ReentrantLock的使用:
基本使用
Lock lock = new ReentrantLock();
lock.lock();
try {
//....
} finally {
lock.unlock();
}
- lock: 调用后一直阻塞直到获得锁。
package com.amber; import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; public class TestReentrantLock {
Lock lock = new ReentrantLock(); public static void main(String[] args) {
TestReentrantLock testReentrantLock = new TestReentrantLock();
new Thread(() -> {
try {
testReentrantLock.testConcurrency(Thread.currentThread());
} catch (InterruptedException e) {
e.printStackTrace();
}
}, "线程1").start(); new Thread(() -> {
try {
testReentrantLock.testConcurrency(Thread.currentThread());
} catch (InterruptedException e) {
e.printStackTrace();
}
}, "线程2").start();
} private void testConcurrency(Thread thread) throws InterruptedException {
//获取锁成功返回true,如果获取失败,等待2S,规定时间内还是没有获得锁,那么就返回false
if (lock.tryLock( 2000, TimeUnit.MICROSECONDS)) {
try {
System.out.println(thread.getName() + " : " + "获取锁");
Thread.sleep(3000);
} finally {
System.out.println(thread.getName() + "释放锁");
lock.unlock(); //一定记得要释放锁
}
} else {
System.out.println(Thread.currentThread().getName() + " : " + "等待了,没有获取锁");
}
}
}
- tryLock:拿到锁返回true,否则false;带有时间限制的tryLock(long time, TimeUnit timeUnit),拿不到锁,就等待一段时间,超时返回false
- lockInterruptibly :调用后如果没有获取到锁会一直阻塞,阻塞过程中会接受中断信号。
lockInterruptibly有点难以理解,假设A线程想去获取锁,但是锁被B线程持有,那么A就会发生堵塞。
A堵塞的时候,可以有以下两种方法发生状态改变:
- A获取锁资源
- A被其他线程中断:
- 这里只得被其他线程中断的意思是,C线程调用A线程的interrupt()。那么此时A线程就会被唤醒,处理中断信号。
lockInterruptibly是被中断,就由阻塞状态被唤醒去处理中断信号。
在JAVA并发编程实战这本书中还提到了ReentrantLock的一个重要用法,那就是轮询锁。下面是书中的源代码:
public boolean transferMoney(Account fromAcct, Account toAcct, DollarAmount amount, long timeout, TimeUnit unit) throws InsufficientFundsException, InterruptedException {
long fixedDelay = 1;
long randMod = 2;
long stopTime = System.nanoTime() + unit.toNanos(timeout);
while (true) {
if (fromAcct.lock.tryLock()) {
try {
if (toAcct.lock.tryLock()) { //如果不能同事获得两个锁,那么线程就会释放已经获得的锁。这样可以很有效的解决死锁问题。
try {
if (fromAcct.getBalance().compareTo(amount) < 0)
throw new InsufficientFundsException();
else{
fromAcct.debit(amount);
toAcct.credit(amount);
returntrue;
}
} finally {
toAcct.lock.unlock();
}
}
} finally {
fromAcct.lock.unlock();
}
}
if (System.nanoTime() < stopTime)
returnfalse;
NANOSECONDS.sleep(fixedDelay + rnd.nextLong() % randMod);
}
}
程序发生死锁的时候,往往只能通过重新启动程序解决。而有时候因为获取锁的时序不一致,很容易发生死锁。根据上述代码第6行和第8行,假设此时我们使用的synochronized内置锁,A线程从cc账号转账到dd账号,B线程从dd账号转账到cc账号,就很容易发生死锁。但是使用tryLock()却可以避免锁顺序造成死锁的问题,
如果线程A、B不能同时获取cc和dd对象的锁,那么就会放弃自己已经获得的锁。
JUC - ReentrantLock 的基本用法 以及 lock()、tryLock()、lockInterruptibly()的区别的更多相关文章
- ReentrantLock可重入锁lock,tryLock的区别
void lock(); Acquires the lock. Acquires the lock if it is not held by another thread and returns im ...
- java并发-ReentrantLock的lock和lockInterruptibly的区别
ReentrantLock的加锁方法Lock()提供了无条件地轮询获取锁的方式,lockInterruptibly()提供了可中断的锁获取方式.这两个方法的区别在哪里呢?通过分析源码可以知道lock方 ...
- 【JDK1.8】JUC——ReentrantLock
一.前言 在之前的几篇中,我们回顾了锁框架中比较重要的几个类,他们为实现同步提供了基础支持,从现在开始到后面,就开始利用之前的几个类来进行各种锁的具体实现.今天来一起看下ReentrantLock,首 ...
- java高并发系列 - 第12天JUC:ReentrantLock重入锁
java高并发系列 - 第12天JUC:ReentrantLock重入锁 本篇文章开始将juc中常用的一些类,估计会有十来篇. synchronized的局限性 synchronized是java内置 ...
- Java同步锁——lock与synchronized 的区别【转】
在网上看来很多关于同步锁的博文,记录下来方便以后阅读 一.Lock和synchronized有以下几点不同: 1)Lock是一个接口,而synchronized是Java中的关键字,synchroni ...
- Lock和synchronized的区别和使用
Java并发编程:Lock 今天看了并发实践这本书的ReentantLock这章,感觉对ReentantLock还是不够熟悉,有许多疑问,所有在网上找了很多文章看了一下,总体说的不够详细,重点和焦点问 ...
- java并发之Lock以及和synchronized区别
从Java5之后,在Java.util.concurrent.locks包下提供了另外一种方式来实现同步访问,那就是Lock. 1.Lock 首先要说明的就是Lock,通过查看Lock的源码可知,Lo ...
- 002 Lock和synchronized的区别和使用
转自 https://www.cnblogs.com/baizhanshi/p/6419268.html 今天看了并发实践这本书的ReentantLock这章,感觉对ReentantLock还是不够熟 ...
- Lock和synchronized的区别和使用(转发)
今天看了并发实践这本书的ReentantLock这章,感觉对ReentantLock还是不够熟悉,有许多疑问,所有在网上找了很多文章看了一下,总体说的不够详细,重点和焦点问题没有谈到,但这篇文章相当不 ...
随机推荐
- 【linux】【ELK】搭建Elasticsearch+Logstash+Kibana+Filebeat日志收集系统
前言 ELK是Elasticsearch.Logstash.Kibana的简称,这三者是核心套件,但并非全部. Elasticsearch是实时全文搜索和分析引擎,提供搜集.分析.存储数据三大功能:是 ...
- 品Spring:实现bean定义时采用的“先进生产力”
前景回顾 当我们把写好的业务代码交给Spring之后,Spring都会做些什么呢? 仔细想象一下,再稍微抽象一下,Spring所做的几乎全部都是: “bean的实例化,bean的依赖装配,bean的初 ...
- SPN扫描
0x01介绍 Kerberos是一种支持票证身份验证的安全协议.如果客户端计算机身份验证请求包含有效的用户凭据和服务主体名称 (SPN),则 Kerberos 身份验证服务器将授予一个票证以响应该请求 ...
- mybatis 插件的原理-责任链和动态代理的体现
目录 1 拦截哪些方法 2 如何代理 3 代理对象 4 责任链设计模式 @ 如果没有自定义过拦截器, 可以看我前面的文章.如果不知道 JDK 动态代理怎么使用的, 可以看我这文章. 责任链设计模式理解 ...
- Spring 梳理 - AOP那些学术概念—通知、增强处理连接点(JoinPoint)切面(Aspect)
Spring AOP那些学术概念—通知.增强处理连接点(JoinPoint)切面(Aspect) 1.我所知道的AOP 初看起来,上来就是一大堆的术语,而且还有个拉风的名字,面向切面编程,都说是 ...
- ajax跨域问题以及解决方案
转:https://blog.csdn.net/csdn_ds/article/category/6937392/3 在工作中,大家应该都遇到过ajax跨域问题,浏览器的错误如下: XMLHttpRe ...
- Git tag相关命令
常见命令如下: // 查看标签,可加上参数-l(列表形式列出) -n(附加说明) git tag [-l -n] // 查看符合检索条件的标签 git tag -l .*.* // 查看对应标签状态 ...
- EL十一大内置对象
这是一个内置对象可以直接拿来使用,不需要再去声明. 1.读取页面上下文: (1)pageContext对象: 获取URL和URI: <body> URI:${pageContext.req ...
- java跬步积累
1.eclipse自动生成get/set方法快捷键 alt+shift+s +r 2.eclipse自动生成等号左边快捷键 将光标移到:号右边,然后按Ctrl+1 3.补全代码快捷键 Alt+/ 4. ...
- 最近太多人问Protobuf的问题了,把这个重新搬出来!
pb杀手 我先让pbkiller做个自我介绍 pbkiller:我是一位专业的争对 protobuf 问题训练有素的杀手,我可以为您轻松搞定 protobuf 在 Cocos Creaotr 开发中的 ...