java ReentrantLock可重入锁功能
1、可重入锁是可以中断的,如果发生了死锁,可以中断程序
//如下程序出现死锁,不去kill jvm无法解决死锁
public class Uninterruptible {
public static void main(String[] args) throws InterruptedException {
final Object o1 = new Object(); final Object o2 = new Object();
Thread t1 = new Thread() {
public void run() {
try {
synchronized(o1) {
Thread.sleep(1000);
synchronized(o2) {}
}
} catch (InterruptedException e) { System.out.println("t1 interrupted"); }
}
};
Thread t2 = new Thread() {
public void run() {
try {
synchronized(o2) {
Thread.sleep(1000);
synchronized(o1) {}
}
} catch (InterruptedException e) { System.out.println("t2 interrupted"); }
}
};
t1.start();
t2.start();
Thread.sleep(2000);
t1.interrupt();
t2.interrupt();
t1.join();
t2.join();
}
}
然后是解决方法:We can reimplement our threads using Reentrant-Lock instead of intrinsic locks, and we can use its lockInterruptibly() method
Thread A先取得lock,Thread B无法取得lock进入block状态,可以透过发出interrupt方式唤醒Thread B,这就是和lock的区别
- lock():若lock被thread A取得,thread B会进入block状态,直到取得lock。
- tryLock():若当下不能取得lock,thread就会放弃。
- lockInterruptibly():跟lock()情況一下,但是thread B可以透过interrupt被唤醒处理InterruptedException。
final ReentrantLock l1 = new ReentrantLock();
final ReentrantLock l2 = new ReentrantLock();
Thread t1 = new Thread() {
public void run() {
try {
➤ l1.lockInterruptibly();
Thread.sleep(1000);
➤ l2.lockInterruptibly();
} catch (InterruptedException e) { System.out.println("t1 interrupted"); }
}
};
Thread t2 = new Thread() {
public void run() {
try {
➤ l2.lockInterruptibly();
Thread.sleep(1000);
➤ l1.lockInterruptibly();
} catch (InterruptedException e) { System.out.println("t2 interrupted"); }
}
}; t1.start();
t2.start();
Thread.sleep(2000);
t1.interrupt(); //当执行到时,线程会捕获InterruptedException程序会结束
t2.interrupt();
t1.join();
t2.join();
}
2、设置锁超时时间
- tryLock():若线程一定时间不能取得lock,thread就会放弃。
public class LockTimeOut{
private ReentrantLock leftChopstick, rightChopstick;
public void method(){
Thread.sleep(random.nextInt(1000)); // Think for a while
leftChopstick.lock();
try {
➤ if (rightChopstick.tryLock(1000, TimeUnit.MILLISECONDS)) {
// Got the right chopstick
………
}
}
说明:Instead of using lock(), this code uses tryLock(), which times out if it fails to acquire the lock
3、条件变量,所谓的 Condition: await()/signal(),可以实现每个关注自己的信号量,每个人有每个人的信号量
When you use Condition: await()/signal() you can distinguish which object or group of objects/threads get a specific signal.
Here is a short example where some threads, the producers, will get the isEmpty signal while the consumers will get the isFull signal:
这里会有两个信号量,生产者和消费这关注的信号量是不一样的,每个人有每个人关注的信号量
private volatile boolean usedData = true;//mutex for data
private final Lock lock = new ReentrantLock();
private final Condition isEmpty = lock.newCondition();//生产者关心 isEmpty信号量
private final Condition isFull = lock.newCondition(); public void setData(int data) throws InterruptedException {
lock.lock();
try {
while(!usedData) {//wait for data to be used如果数据为没有消费,那么就会阻塞到 isEmpty信号量,生产者等待这个空信号量
isEmpty.await();
}
this.data = data;
isFull.signal();//broadcast that the data is now full. 释放 isFull满这个信号量
usedData = false;//tell others I created new data.
}finally {
lock.unlock();//interrupt or not, release lock
}
} public void getData() throws InterruptedException{
lock.lock();
try {
while(usedData) {//usedData is lingo for empty 这里都必须是循环,因为 从await()这个函数 退出阻塞说明条件可能满足,不是一定满足
isFull.await();
}
isEmpty.signal();//tell the producers to produce some more.
usedData = true;//tell others I have used the data.
}finally {//interrupted or not, always release lock
lock.unlock();
}
}
然后注意信号量和 条件是不一样的东西
if the condition is not true, it calls await(), which atomically unlocks the lock and blocks on the condition variable.
When another thread calls signal() or signalAll() to indicate that the condition might now be true, await() unblocks and automatically reacquires the lock. An important point is that when await() returns, it only indicates that the condition might be true. This is why await() is called within a loop—we need to go back,recheck whether the condition is true, and potentially block on await() again if necessary.
ReentrantLock lock = new ReentrantLock();
Condition condition = lock.newCondition();
lock.lock();
try {
while (!<<condition is true>>) //如果条件不满足
condition.await();
<<use shared resources>>
} finally { lock.unlock(); }
然后是具体实例
private void think() throws InterruptedException {
table.lock();
try {
eating = false;
left.condition.signal();//释放条件变量
right.condition.signal();//释放条件变量
} finally { table.unlock(); }
Thread.sleep(1000);
}
private void eat() throws InterruptedException {
table.lock();
try {
13 while (left.eating || right.eating)
14 condition.await();
eating = true;
} finally { table.unlock(); }
Thread.sleep(1000);
}
java ReentrantLock可重入锁功能的更多相关文章
- java ReentrantLock可重入锁的使用场景
摘要 从使用场景的角度出发来介绍对ReentrantLock的使用,相对来说容易理解一些. 场景1:如果发现该操作已经在执行中则不再执行(有状态执行) a.用在定时任务时,如果任务执行时间可能超过下次 ...
- JUC 一 ReentrantLock 可重入锁
java.util.concurrent.locks ReentrantLock即可重入锁,实现了Lock和Serializable接口 ReentrantLock和synchronized都是可重入 ...
- 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关键字所使用的锁也是可重入 ...
- Java中可重入锁ReentrantLock原理剖析
本文由码农网 – 吴极心原创,转载请看清文末的转载要求,欢迎参与我们的付费投稿计划! 一. 概述 本文首先介绍Lock接口.ReentrantLock的类层次结构以及锁功能模板类AbstractQue ...
- Java多线程——深入重入锁ReentrantLock
简述 ReentrantLock 是一个可重入的互斥(/独占)锁,又称为“独占锁”. ReentrantLock通过自定义队列同步器(AQS-AbstractQueuedSychronized,是实现 ...
- ReentrantLock——可重入锁的实现原理
一. 概述 本文首先介绍Lock接口.ReentrantLock的类层次结构以及锁功能模板类AbstractQueuedSynchronizer的简单原理,然后通过分析ReentrantLock的lo ...
- ReentrantLock可重入锁——源码详解
开始这篇博客之前,博主默认大家都是看过AQS源码的~什么居然没看过猛戳下方 全网最详细的AbstractQueuedSynchronizer(AQS)源码剖析(一)AQS基础 全网最详细的Abstra ...
随机推荐
- LayoutInflater(二)
每一个视图的绘制过程都必须经历三个最主要的阶段,即onMeasure().onLayout()和onDraw(),下面我们逐个对这三个阶段展开进行探讨. 一. onMeasure() measure是 ...
- Linux学习笔记(12)用户和用户组管理
越是对服务器安全性要求高的服务器,越需要建立合理的用户权限等级制度和服务器操作规范.在Linux中主要是通过用户配置文件来查看和修改用户信息. 1 用户信息文件 (1)用户信息文件/etc/passw ...
- JVM性能监控工具-Jvisualvm
用法:Jvisualvm是JDK自带的一款性能分析工具 使用方式: 1.配置好JDK环境变量 1.本地JVM监控略 2.远程JVM监控 用JMX对Resin内存状态进行监控 ,可以看到本地所有可监控的 ...
- Ubuntu中如何打开终端terminal
法一 先按住Alt,然后再按F2,出来一个运行框,在里面输入gnome-terminal即可 [编辑]法二 如果想从右键菜单中打开终端,需要安装一个软件: sudoapt-get install na ...
- 分数try catch
要求:编写一个程序,此程序在运行时要求用户输入一个 整数,代表某门课的考试成绩,程序接着给出“不及格”.“及格”.“中”.“良”.“优”的结论.要求程序必须具备足够的健壮性,不管用户输入什 么样的内容 ...
- ASP.NET MVC3 中整合 NHibernate3.3、Spring.NET2.0 使用AOP执行事务处理
方法1 <object id="ServiceOperation" type="Spring.Aop.Support.SdkRegularExpressionMet ...
- BestCoder Round #66 (div.2)
构造 1002 GTW likes gt 题意:中文题面 分析:照着题解做的,我们可以倒着做,记一下最大值,如果遇到了修改操作,就把最大值减1,然后判断一下这个人会不会被消灭掉,然后再更新一下最大值. ...
- 01_Swift2基础之Swift简介+创建
1.Swift 1> Swift初影响 Swift 是新一代的 iOS.OS X 和 watchOS 和 tvOS 的app开发编程语言. 中文名"雨燕",寓意为敏捷.灵巧而 ...
- Target:IG
https://www.zhihu.com/question/25525630 别人轻轻松松红名,我拼死挣扎才1700+分. 仔细想想,虽然我在这东西上花了太多的精力,可是我根本没有认真学.做题全靠抄 ...
- BZOJ3551 : [ONTAK2010]Peaks加强版
首先强制在线的话,肯定是不能再离线排序+平衡树启发式合并了. 这回要用的是线段树合并,每次把两棵线段树合并,总复杂度为$O(n\log n)$ 预处理: 把边按权值从小到大排序,依次加边, 对于边(x ...