`
tenyears
  • 浏览: 206714 次
  • 性别: Icon_minigender_1
  • 来自: 湖北武汉
文章分类
社区版块
存档分类
最新评论

再谈重入锁--ReentrantLock

阅读更多

入锁(ReentrantLock)是一种递归无阻塞的同步机制。以前一直认为它是synchronized的简单替代,而且实现机制也不相差太远。不过最近实践过程中发现它们之间还是有着天壤之别。

以下是官方说明:一个可重入的互斥锁定 Lock,它具有与使用 synchronized 方法和语句所访问的隐式监视器锁定相同的一些基本行为和语义,但功能更强大。ReentrantLock 将由最近成功获得锁定,并且还没有释放该锁定的线程所拥有。当锁定没有被另一个线程所拥有时,调用 lock 的线程将成功获取该锁定并返回。如果当前线程已经拥有该锁定,此方法将立即返回。可以使用 isHeldByCurrentThread() 和 getHoldCount() 方法来检查此情况是否发生。

它提供了lock()方法:
如果该锁定没有被另一个线程保持,则获取该锁定并立即返回,将锁定的保持计数设置为 1。
如果当前线程已经保持该锁定,则将保持计数加 1,并且该方法立即返回。
如果该锁定被另一个线程保持,则出于线程调度的目的,禁用当前线程,并且在获得锁定之前,该线程将一直处于休眠状态,此时锁定保持计数被设置为 1。

最近在研究Java concurrent中关于任务调度的实现时,读了延迟队列DelayQueue的一些代码,比如take()。该方法的主要功能是从优先队列(PriorityQueue)取出一个最应该执行的任务(最优值),如果该任务的预订执行时间未到,则需要wait这段时间差。反之,如果时间到了,则返回该任务。而offer()方法是将一个任务添加到该队列中。

后来产生了一个疑问:如果最应该执行的任务是一个小时后执行的,而此时需要提交一个10秒后执行的任务,会出现什么状况?还是先看看take()的源代码:

<!---->
public E take() throws InterruptedException {
    final ReentrantLock lock = this.lock;
    lock.lockInterruptibly();
    try {
        for (;;) {
            E first = q.peek();
            if (first == null) {
                available.await();
            } else {
                long delay =  first.getDelay(TimeUnit.NANOSECONDS);
                if (delay > 0) {
                    long tl = available.awaitNanos(delay);
                } else {
                    E x = q.poll();
                    assert x != null;
                    if (q.size() != 0)
                        available.signalAll(); // wake up other takers
                    return x;
                }
            }
        }
    } finally {
        lock.unlock();
    }
}

而以下是offer()的源代码:

public boolean offer(E e) {
    final ReentrantLock lock = this.lock;
    lock.lock();
    try {
        E first = q.peek();
        q.offer(e);
        if (first == null || e.compareTo(first) < 0)
            available.signalAll();
        return true;
    } finally {
        lock.unlock();
    }
}

 

如代码所示,take()和offer()都是lock了重入锁。如果按照synchronized的思维(使用诸如synchronized(obj)的方法),这两个方法是互斥的。回到刚才的疑问,take()方法需要等待1个小时才能返回,而offer()需要马上提交一个10秒后运行的任务,会不会一直等待take()返回后才能提交呢?答案是否定的,通过编写验证代码也说明了这一点。这让我对重入锁有了更大的兴趣,它确实是一个无阻塞的锁。

下面的代码也许能说明问题:运行了4个线程,每一次运行前打印lock的当前状态。运行后都要等待5秒钟。

public static void main(String[] args) throws InterruptedException {
  final ExecutorService exec = Executors.newFixedThreadPool(4);
  final ReentrantLock lock = new ReentrantLock();
  final Condition con = lock.newCondition();
  final int time = 5;
  final Runnable add = new Runnable() {
    public void run() {
      System.out.println("Pre " + lock);
      lock.lock();
      try {
        con.await(time, TimeUnit.SECONDS);
      } catch (InterruptedException e) {
        e.printStackTrace();
      } finally {
        System.out.println("Post " + lock.toString());
        lock.unlock();
      }
    }
  };
  for(int index = 0; index < 4; index++)
    exec.submit(add);
  exec.shutdown();
}

 

这是它的输出:
Pre ReentrantLock@a59698[Unlocked]
Pre ReentrantLock@a59698[Unlocked]
Pre ReentrantLock@a59698[Unlocked]
Pre ReentrantLock@a59698[Unlocked]
Post ReentrantLock@a59698[Locked by thread pool-1-thread-1]
Post ReentrantLock@a59698[Locked by thread pool-1-thread-2]
Post ReentrantLock@a59698[Locked by thread pool-1-thread-3]
Post ReentrantLock@a59698[Locked by thread pool-1-thread-4]

每一个线程的锁状态都是“Unlocked”,所以都可以运行。但在把con.await改成Thread.sleep(5000)时,输出就变成了:
Pre ReentrantLock@a59698[Unlocked]
Pre ReentrantLock@a59698[Locked by thread pool-1-thread-1]
Pre ReentrantLock@a59698[Locked by thread pool-1-thread-1]
Pre ReentrantLock@a59698[Locked by thread pool-1-thread-1]
Post ReentrantLock@a59698[Locked by thread pool-1-thread-1]
Post ReentrantLock@a59698[Locked by thread pool-1-thread-2]
Post ReentrantLock@a59698[Locked by thread pool-1-thread-3]
Post ReentrantLock@a59698[Locked by thread pool-1-thread-4]

以上的对比说明线程在等待时(con.await),已经不在拥有(keep)该锁了,所以其他线程就可以获得重入锁了。

有必要会过头再看看Java官方的解释:“如果该锁定被另一个线程保持,则出于线程调度的目的,禁用当前线程,并且在获得锁定之前,该线程将一直处于休眠状态”。我对这里的“保持”的理解是指非wait状态外的所有状态,比如线程Sleep、for循环等一切有CPU参与的活动。一旦线程进入wait状态后,它就不再keep这个锁了,其他线程就可以获得该锁;当该线程被唤醒(触发信号或者timeout)后,就接着执行,会重新“保持”锁,当然前提依然是其他线程已经不再“保持”了该重入锁。

总结一句话:对于重入锁而言,"lock"和"keep"是两个不同的概念。lock了锁,不一定keep锁,但keep了锁一定已经lock了锁。

分享到:
评论
13 楼 xlshl43 2017-05-27  
感觉有点误认子弟!!
“如果按照synchronized的思维(使用诸如synchronized(obj)的方法),这两个方法是互斥的。”这句跟哪个锁没关系,主要看你是用sleep还是用await/wait
12 楼 xlshl43 2017-05-27  
sulpha 写道
"它确实是一个无阻塞的锁"这种说法有所不妥。

其实核心在于Condition.await()。JDK里面的文档:
The lock associated with this Condition is atomically released and the current thread becomes disabled for thread scheduling purposes...

也就是与available绑定的lock在调用available.await()时,lock会被释放。在await()返回后(准确点是返回之前),lock会重新获得。JDK里面的原话:
In all cases, before this method can return the current thread must re-acquire the lock associated with this condition. When the thread returns it is guaranteed to hold this lock.

10楼说的对!!!
11 楼 生活小丑 2014-04-10  
其实重点就是await的问题
10 楼 sulpha 2013-11-28  
"它确实是一个无阻塞的锁"这种说法有所不妥。

其实核心在于Condition.await()。JDK里面的文档:
The lock associated with this Condition is atomically released and the current thread becomes disabled for thread scheduling purposes...

也就是与available绑定的lock在调用available.await()时,lock会被释放。在await()返回后(准确点是返回之前),lock会重新获得。JDK里面的原话:
In all cases, before this method can return the current thread must re-acquire the lock associated with this condition. When the thread returns it is guaranteed to hold this lock.
9 楼 淘气天空lc 2013-08-12  
如果该锁定被另一个线程保持,则出于线程调度的目的,禁用当前线程,并且在获得锁定之前,该线程将一直处于休眠状态,此时锁定保持计数被设置为 1。  这个意思是线程锁不会释放吧?????????????
8 楼 fenshen6046 2012-03-29  
java官方文档Condition的await描述
await

void await()
           throws InterruptedException

    造成当前线程在接到信号或被中断之前一直处于等待状态。

    与此 Condition 相关的锁以原子方式释放,并且出于线程调度的目的,将禁用当前线程,且在发生以下四种情况之一 以前,当前线程将一直处于休眠状态:

        其他某个线程调用此 Condition 的 signal() 方法,并且碰巧将当前线程选为被唤醒的线程;或者
        其他某个线程调用此 Condition 的 signalAll() 方法;或者
        其他某个线程中断当前线程,且支持中断线程的挂起;或者
        发生“虚假唤醒”
7 楼 zephyrum 2012-02-09  
amos_tl 写道
原子操作 底层也是用的 synchronized

Atomic类底层用的是sun.misc.unsafe里的系统级别cas操作,没有synchronize。
6 楼 amos_tl 2011-08-31  
原子操作 底层也是用的 synchronized
5 楼 amos_tl 2011-08-31  
引用

推荐一下原子操作,更简便,实用!java.concurrent.atomic下的几种常用类型也很全了,性能比lock好!
前提是不会出现race condition!
当然LOCK比synchronized好多了!lock()与unlock()结合在try.finally块中,使用也比较方便!


原子操作底层也是使用的synchronized.
4 楼 lordhong 2007-01-21  
多谢,受益非浅
3 楼 YuLimin 2007-01-20  
推荐一篇我在IBM DW上面看到的不错的文章

《Java 理论与实践: JDK 5.0 中更灵活、更具可伸缩性的锁定机制》

http://www.ibm.com/developerworks/cn/java/j-jtp10264/index.html
2 楼 歆渊 2007-01-20  
DelayQueue 实现里面只用到了一个 Lock 和 它的一个 Condition 的组合, 这个没有超出原有 synchronized + Object.wait() + Object.notify() 的范围, 新的同步机制的好处, 比如 公平竞争, 单资源多条件 等等都没有派上用场.

在对一个对象的 synchronized 块中 Object.wait(); 过程中, 同样是不会阻塞其它线程进入该对象的 synchronized 块的. 所以它目前的实现是完全可以用原有机制代替的

现在的代码执行速度方面不知道是不是一定有提升, 但从内存占用角度看, 还不如原有机制效率高.
1 楼 galaxystar 2007-01-20  
推荐一下原子操作,更简便,实用!java.concurrent.atomic下的几种常用类型也很全了,性能比lock好!
前提是不会出现race condition!
当然LOCK比synchronized好多了!lock()与unlock()结合在try.finally块中,使用也比较方便!

相关推荐

    Java分布式应用学习笔记06浅谈并发加锁机制分析

    `ReentrantLock`是一种可重入的互斥锁,支持公平和非公平两种模式。它内部有一个`Sync`类,继承自`AbstractQueuedSynchronizer`,并且有两个子类分别实现了公平和非公平锁的逻辑。 - **非公平锁** (`NonfairSync`):...

    浅谈多线程中的锁的几种用法总结(必看)

    ReentrantLock 的特点是可以重入,即一个线程可以多次获得锁,每次获得锁时,锁的计数器都会增加,直到锁被释放时,计数器才会减少。 在上面的代码中,我们使用 ReentrantLock 来实现线程同步。在 UseReentrantLock...

    浅谈Java并发编程之Lock锁和条件变量

    Lock接口有三种实现类:ReentrantLock、ReetrantReadWriteLock.ReadLock和ReetrantReadWriteLock.WriteLock,即重入锁、读锁和写锁。Lock锁必须被显式地创建、锁定和释放,以确保锁最终一定会被释放。 在使用Lock锁...

    浅谈JAVA中多线程的实现.zip

    ReentrantLock是Lock接口的一个实现,具有可重入性,即一个线程可以进入已经由该线程持有的锁。 7. ThreadLocal:为每个线程提供独立的变量副本,避免了线程之间的数据冲突。每个线程都拥有自己的ThreadLocal变量,...

    多线程面试59题(含答案)_.zip

    而Lock接口提供更细粒度的锁控制,如ReentrantLock可重入锁。 此外,死锁、活锁和饥饿现象也是面试热点。死锁是指两个或多个线程相互等待对方释放资源,导致都无法继续执行;活锁则是线程不断尝试获取资源但都失败...

    浅谈Java中ABA问题及避免

    例如,在并发栈的例子中,可以使用ReentrantLock 锁机制来保证线程安全,避免ABA问题的出现。 ABA问题是Java并发编程中的一种常见问题,需要开发者对其进行认真对待和处理,以避免程序的不正确执行和数据的不一致。...

    sesvc.exe 阿萨德

    再来看看 get 函数: public V get(Object key) { if (key == null) return getForNullKey(); Entry,V&gt; entry = getEntry(key); return null == entry ? null : entry.getValue(); } final Entry,V&gt; getEntry...

    浅谈Java的两种多线程实现方式

    3. **Lock接口和实现**:如`ReentrantLock`,提供了更细粒度的锁控制和更多的同步机制。 在实际开发中,根据需求选择合适的多线程实现方式,并合理使用同步机制,可以有效地提高程序的并发性能和安全性。

    java 并发编程的艺术

    第5章深入介绍了Java并发包中与锁相关的API和组件,如ReentrantLock、ReadWriteLock等,以及它们的使用方式和实现细节。同时,书中也探讨了锁优化技术,如锁粗化、锁消除等,这些都是提高并发程序性能的关键技术点。...

    javaSE代码实例

    17.3.2 ReentrantLock锁的具体使用 387 17.3.3 ReadWriteLock接口与ReentrantReadWriteLock类简介 390 17.3.4 ReentrantReadWriteLock读/写锁的具体使用 391 17.4 信号量的使用 393 17.4.1 Semaphore类简介...

Global site tag (gtag.js) - Google Analytics