`
m635674608
  • 浏览: 5061279 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

Java中Lock,tryLock,lockInterruptibly有什么区别?

    博客分类:
  • java
 
阅读更多

ReentrantLock 锁有好几种,除了常用的lock ,tryLock ,其中有个lockInterruptibly 。
先把API粘贴上来

lock
public void lock()
获取锁。
如果该锁没有被另一个线程保持,则获取该锁并立即返回,将锁的保持计数设置为 1。
如果当前线程已经保持该锁,则将保持计数加 1,并且该方法立即返回。
如果该锁被另一个线程持有,则该线程不可被调度(disabled for thread scheduling purposes)(即阻塞状态,CPU不会给该线程分配时间片)直到该线程获取到该锁,并且在获取到锁后,将保持计数设置为1

指定者:
接口 Lock 中的 lock

 

lockInterruptibly
public void lockInterruptibly() throws InterruptedException
1)如果当前线程未被中断,则获取锁。 

2)如果该锁没有被另一个线程保持,则获取该锁并立即返回,将锁的保持计数设置为 1。 

3)如果当前线程已经保持此锁,则将保持计数加 1,并且该方法立即返回。 

如果该锁被另一个线程持有,则该线程不可被调度(disabled for thread scheduling purposes)(即阻塞状态,CPU不会给该线程分配时间片)直到该线程获取到该锁,并且在获取到锁后,将保持计数设置为1

-----> getHoldCount

5)如果当前线程获得该锁,则将锁保持计数设置为 1。 
   如果当前线程: 
       1)在进入此方法时已经设置了该线程的中断状态;或者 

       2)在等待获取锁的同时被中断。 

   则抛出 InterruptedException,并且清除当前线程的已中断状态。 


6)在此实现中,因为此方法是一个显式中断点,所以要优先考虑响应中断,而不是响应锁的普通获取或
重入获取。 

指定者: 接口 Lock 中的 lockInterruptibly
抛出:   InterruptedException   如果当前线程已中断。

 

tryLock    public boolean tryLock()

仅在调用时锁未被另一个线程保持的情况下,才获取该锁。 

1)如果该锁没有被另一个线程保持,并且立即返回 true 值,则将锁的保持计数设置为 1。
即使已将此锁设置为使用公平排序策略,但是调用 tryLock() 仍将 立即获取锁(如果有可用的),
而不管其他线程当前是否正在等待该锁。在某些情况下,此“闯入”行为可能很有用,即使它会打破公
平性也如此。如果希望遵守此锁的公平设置,则使用 tryLock(0, TimeUnit.SECONDS) 
,它几乎是等效的(也检测中断)。 

2)如果当前线程已经保持此锁,则将保持计数加 1,该方法将返回 true。 

3)如果锁被另一个线程保持,则此方法将立即返回 false 值。 

指定者:
   接口 Lock 中的  tryLock
返回: 
   如果锁是自由的并且被当前线程获取,或者当前线程已经保持该锁,则返回 true;否则返回 
false



关于中断又是一段很长的叙述,先不谈。

1)lock(), 拿不到lock就不罢休,不然线程就一直block。 比较无赖的做法。
2)tryLock(),马上返回,拿到lock就返回true,不然返回false。 比较潇洒的做法。
带时间限制的tryLock(),拿不到lock,就等一段时间,超时返回false。比较聪明的做法。

 

3)lockInterruptibly()就稍微难理解一些。

 

先说说线程的打扰机制,每个线程都有一个 打扰 标志。这里分两种情况,
1. 线程在sleep或wait,join, 此时如果别的进程调用此进程的 interrupt()方法,此线程会被唤醒并被要求处理InterruptedException;(thread在做IO操作时也可能有类似行为,见java thread api)

 


2. 此线程在运行中, 则不会收到提醒。但是 此线程的 “打扰标志”会被设置, 可以通过isInterrupted()查看并 作出处理。

 

lockInterruptibly()和上面的第一种情况是一样的, 线程在请求lock并被阻塞时,如果被interrupt,则“此线程会被唤醒并被要求处理InterruptedException”。并且如果线程已经被interrupt,再使用lockInterruptibly的时候,此线程也会被要求处理interruptedException




先看lock()方法

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * @author 作者 E-mail:
 * @version 创建时间:2015-10-23 下午01:47:03 类说明
 */
public class TestLock
{
    // @Test
    public void test() throws Exception
    {
        final Lock lock = new ReentrantLock();
        lock.lock();

        
        Thread t1 = new Thread(new Runnable()
        {
            @Override
            public void run()
            {
                lock.lock();
                System.out.println(Thread.currentThread().getName() + " interrupted.");
            }
        },"child thread -1");
        
        t1.start();
        Thread.sleep(1000);
        
        t1.interrupt();
        
        Thread.sleep(1000000);
    }
    
    public static void main(String[] args) throws Exception
    {
        new TestLock().test();
    }
}


用eclipse对这个程序进行debug发现,即使子线程已经被打断,但是子线程仍然在run,可见lock()方法并不关心线程是否被打断,甚至说主线程已经运行完毕,子线程仍然在block().


而使用LockInterupptibly,则会响应中断

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * @author 作者 E-mail:
 * @version 创建时间:2015-10-23 下午01:53:10 类说明
 */
public class TestLockInterruptibly
{

    // @Test
    public void test3() throws Exception
    {
        final Lock lock = new ReentrantLock();
        lock.lock();

        Thread t1 = new Thread(new Runnable()
        {
            @Override
            public void run()
            {
                try
                {
                    lock.lockInterruptibly();
                }
                catch(InterruptedException e)
                {
                    System.out.println(Thread.currentThread().getName() + " interrupted.");
                }
            }
        }, "child thread -1");

        t1.start();
        Thread.sleep(1000);

        t1.interrupt();

        Thread.sleep(1000000);
    }

    public static void main(String[] args) throws Exception
    {
        new TestLockInterruptibly().test3();
    }
}


try{
      Thread.sleep(2000);
      lock.lockInterruptibly();
   }catch(InterruptedException e){
       System.out.println(Thread.currentThread().getName()+" interrupted.");
}

 

  t1.start();
  t1.interrupt();
  Thread.sleep(1000000);

 

如果将代码改成这样,那么将会在在阻塞之前已经中断,此时再lockInterruptibly()也是会相应中断异常的

 

 

 

 

转自:https://www.zhihu.com/question/36771163

分享到:
评论

相关推荐

    Java的Lock小结

    - **尝试获取锁**:`Lock`还提供了`tryLock()`方法,允许开发者尝试获取锁而不阻塞线程,这对于某些不需要长时间等待的场景非常有用。 #### 四、Lock的优势与局限 - **性能优势**:在Java 5引入`Lock`时,它的实现...

    java的Lock锁原理详解.docx

    Lock接口提供了比synchronized更丰富的功能,如尝试获取锁(tryLock)、可中断的锁获取(lockInterruptibly)、超时获取锁(tryLock(long time, TimeUnit unit))以及显式解锁(unlock)。Lock接口的实现类,例如...

    java的lock和synchronized的区别.docx

    Java 中的 Lock 和 Synchronized 的区别 Java 语言中有很多相似关键字或相似意义的字,但 lock 和 synchronized 是两个最容易混淆的关键字。它们都是锁的意思,都是为了线程安全性、应用合理性和运行效率的。下面...

    Lock总结.doc

    例如,在转账操作中,如果使用`tryLock()`,可以避免因无法获取锁而无限等待,从而可能导致的死锁问题。通过检查`tryLock()`的返回值,我们可以决定是否继续转账操作。 总的来说,`Lock`提供了更精细的控制和更高的...

    java中Locks的使用详解

    * boolean tryLock(long timeout, TimeUnit timeUnit) – 和 tryLock() 很像,只是多了一个尝试获取锁的时间。 * void unlock() – unlock 实例。 * Condition newCondition() - 生成一个和当前 Lock 实例绑定的 ...

    java 锁 Lock接口详解.docx

    `Lock`的`tryLock(long time, TimeUnit unit)`方法允许设置等待时间,超时未获得锁则返回。 3. **读写分离**:`synchronized`无法区分读操作和写操作,可能导致读操作的并行性降低。`ReentrantReadWriteLock`提供...

    JavaLock与Condition的理解Reentran

    1. 使用tryLock()避免死锁:在高并发环境中,可以先尝试获取锁,失败后再进行其他操作,减少死锁的风险。 2. 使用LockInterruptibly()处理中断:配合`Thread.interrupt()`,可以优雅地处理线程中断。 3. 避免锁的...

    java中的Lock类和Condition类.docx

    - `tryLock()`: 这个方法尝试获取锁,如果立即可以获得锁则成功返回,否则立即返回false。还有对应的带超时参数的版本,可以在指定时间后自动放弃获取锁。 - `lockInterruptibly()`: 这个方法会在获取锁时响应中断...

    Java并发编程(20)并发新特性-Lock锁和条件变量(

    3. tryLock():尝试获取锁,如果成功则立即返回true,否则返回false,不会阻塞。 4. unlock():释放锁,必须在获取锁后调用,否则可能导致死锁。 在Java中,ReentrantLock(可重入锁)是Lock接口的主要实现,它支持...

    Java多线程编程之Lock用法实例

    4. `boolean tryLock(long timeout, TimeUnit unit) throws InterruptedException`: 在指定的时间内尝试获取锁,如果超时还未获取到锁,则返回`false`。同样,如果线程在等待期间被中断,会抛出`...

    Java并发编程实践-电子书-07章.pdf

    4. **`boolean tryLock(long time, TimeUnit unit)` throws `InterruptedException`**:在给定时间内尝试获取锁。如果锁在指定时间内变为可用状态,并且当前线程未被中断,则获取锁并返回`true`;如果锁未能在指定...

    java_汇总知识点1

    - Lock、tryLock和lockInterruptibly:Lock接口提供了比synchronized更细粒度的锁控制,tryLock尝试获取锁,lockInterruptibly允许被中断。 5. **并发与JUC**: - JUC(Java Util Concurrency)包包含各种并发...

    Java concurrency之互斥锁_动力节点Java学院

    9. `lock()`, `lockInterruptibly()`, `tryLock()`, 和 `tryLock(long timeout, TimeUnit unit)`: 不同的锁获取方式,`lockInterruptibly()`允许线程在等待锁时响应中断,`tryLock()`尝试获取锁,存在超时版本。...

    Java锁机制详解.pdf

    Java锁机制是Java多线程编程中的核心概念之一,其主要目的是确保在多线程环境下,多个线程能够安全地访问共享资源,避免数据不一致的问题。Java锁机制的发展历经了多个版本的改进,尤其是Java 5.0引入的显示锁...

    java经典面试 .doc

    此外,Lock的`tryLock()`方法允许非阻塞尝试获取锁,`lockInterruptibly()`则允许线程在等待锁时响应中断。 这些是Java面试中常见的基础和并发编程知识点,理解并熟练运用它们对于成为一名合格的Java开发者至关重要...

    Java concurrency之互斥锁_动力节点Java学院整理

    这与`synchronized`块类似,但`ReentrantLock`提供了更细粒度的控制,如尝试非阻塞获取锁(`tryLock()`)、可中断的锁获取(`lockInterruptibly()`)以及对锁状态的查询方法。 总结来说,`ReentrantLock`是Java并发编程...

    Java并发编程实践-电子书-07章

    深入地探讨Java并发编程实践中的显示锁概念,我们聚焦于《Java并发编程实践-电子书-07章》所提供的丰富内容。本章着重于展示如何使用显示锁(也称为高级锁),并详细介绍了Lock和ReentrantLock接口以及它们在并发...

    Java多线程程序设计

    3. Lock接口与ReentrantLock类:提供比synchronized更细粒度的锁控制,如公平锁、非公平锁、读写锁等,以及tryLock()尝试获取锁、lockInterruptibly()响应中断等高级功能。 四、线程通信 Java提供了wait(), notify...

    java 读写锁代码

    Java 读写锁是Java并发编程中的一种重要机制,它为多线程环境下的数据访问提供了更为精细的控制。在Java的`java.util.concurrent.locks`包中,`ReentrantReadWriteLock`类实现了读写锁的功能。这个锁允许多个读取者...

Global site tag (gtag.js) - Google Analytics