显式锁

2017-02-15  本文已影响0人  captain_fu

在JAVA5.0之前,在协调对共享对象的访问时,有synchronized和volatile两种机制。JAVA5.0增加了ReentrantLock机制。ReentrantLock不是用来替代前两者,而是在内置加锁机制不适用时使用的高级功能。

Lock

接口的方法。

void lock();

获取锁。如果获取锁失败,当前线程休眠,直到获取锁成功。

void lockInterruptibly() throws InterruptedException;

获取锁,和上一方法不同之处在于,有两种方式可以终止当前线程的休眠:1.获取锁成功。2.其他线程中断当前线程。

boolean tryLock();

获取锁,立刻返回结果,获取成功返回true,失败返回false。

boolean tryLock(long time, TimeUnit unit) throws InterruptedException;

在指定时间内获取锁。获取到锁之前;时间结束之前;被其他线程终端之前;当前线程会休眠等待获取锁成功。

void unlock();

释放锁。

Condition newCondition();

返回一个绑定到当前锁的Condition实例,等待条件之前,锁必须被当前线程持有,调用Condition的await()方法会自动释放锁,知道重新获取锁。

ReentrantLock

与synchronized相比,ReentrantLock为处理锁的不可用性问题提供了更高的灵活性。

1. ReentrantLock可以中断一个正在等待获取锁的线程。

2. ReentrantLock可以在请求获取一个锁时无限地等待下去。

3. ReentrantLock可以实现非阻塞结构的加锁规则。

轮询锁和定时锁

1. 轮询锁:如果不能获得所有需要的锁,那么可以使用可定时的或可轮询的锁获取方式。这种方式会释放已经获得的锁,然后重新尝试获取所有锁。

2. 定时锁:在实现具有时间限制的操作时,调用了一个阻塞方法是,它可以根据剩余时间来提供一个时限。如果操作不能在指定的时间内给出结果,会使程序提前结束。

可中断的锁获取操作

lockInterruptibly()方法能够在获得锁的同时保持对中断的相应。

tryLock()方法同样可以响应中断,因此当需要实现一个定时的和可中断的锁获取操作时,可以使用tryLock()。

公平性

ReentrantLock支持非公平锁,以支持更多的场景。

synchronized VS ReentrantLock

1. synchronized大家更熟悉。

2. ReentrantLock必须要在finally中unlock。

3. 线程dump中能给出哪些调用帧获得了哪些锁,并能够检测和识别发生死锁的线程。JVM并不知道那线程持有ReentrantLock。

concurrent包支持读写锁

在ReadWriteLock中定义。

一些可选实现:

1. 释放优先。

2. 读线程插队。

3. 重入性。

4. 降级。持有写锁,是否可以不释放锁直接获取读锁。

5. 升级。持有读锁线程,是否可以优先获取写锁。

以上内容部分摘自《java并发编程实战》。

上一篇 下一篇

猜你喜欢

热点阅读