lockInterruptibly 和 lock

本文通过一个生动的例子介绍了ReentrantLock中的lockInterruptibly方法的特点。该方法允许在未获取到锁的情况下响应中断请求,这对于实现可中断的等待逻辑非常有用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一.序言

    ReentrantLock 锁有好几种,除了常用的lock tryLock 等意外,其中有个lockInterruptibly 用的比较少,看了下JDK 说明,官方的东西总是那么不容易理解,这里用稍微通俗的语言来描述一下。

 

二.栗子

     1.我们用张三和李四 吃水果来描述:

        如果张三吃苹果->休息5秒->吃香蕉

        李四:发现张三在吃,就等待,直到李四吃完

import java.util.concurrent.locks.ReentrantLock;

/**
 * Created by qqr on 15/5/17.
 */
public class Lock {


    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(new TaskThread());
        t1.setName("张三 ");
        Thread t2  = new Thread(new TaskThread());
        t2.setName("李四 ");
        t1.start();
        Thread.sleep(100);// 保证先后顺序

        t2.start();
        Thread.sleep(100);
        // 中断
        t2.interrupt();


    }


    static class TaskThread implements Runnable{
        // 所有线程都用同一把锁
        static ReentrantLock lock = new ReentrantLock();

        @Override
        public void run() {

            try {
                // 
                lock.lockInterruptibly();
                System.out.println(Thread.currentThread().getName() + " 吃苹果");

                Thread.sleep(5 * 1000);

                System.out.println(Thread.currentThread().getName() + " 吃香蕉");
            } catch (InterruptedException e) {
                System.out.println(Thread.currentThread().getName() + " 被中断了,没吃了");
            }finally {
                lock.unlock();
            }

        }

    }
}

   返回结果是:

   

张三  吃苹果
张三  吃香蕉
李四  吃苹果
李四  被中断了,没吃了

 

   第一次如果用 lock() ,那么就是正常的逻辑:

   张三持有锁的时候,李四没有获得锁,打断效果也会阻塞着,等张三吃完,李四吃苹果,然后被打断,就是上面结果了。

 

     如果换成lockInterruptibly,那么结果是:

    

张三  吃苹果
李四  被中断了,没吃了
Exception in thread "李四 " java.lang.IllegalMonitorStateException
张三  吃香蕉

   

     我们会发现,即使李四没有获得锁,也是可以被打断的, 这个区别是很明显的。

 

    在linkedBlockingQuene 的put/take 方法中,我们可以看到类似这样的代码: 

    

public void put(E e) throws InterruptedException {
        // ....
        final ReentrantLock putLock = this.putLock;
        final AtomicInteger count = this.count;
        putLock.lockInterruptibly();
        try {
           while (count.get() == capacity) {
                notFull.await();
            }
            // ....
       }
}
  

 

   最为加锁的阻塞队列, 这里不允许多线程同时put,多线程对进入等待,但是用了lockInterruptibly,那么就意味着可以中断那些等待着的线程

 

 四.小结

      1.这里仅仅简单理解下他效果,可能并不全

      2.遇到其他的场景 或者 解释有问题,请分享 和指出,谢谢。

 

   

### synchronized Lock 的区别及使用场景 #### 1. 基本概念 `synchronized` 是 Java 中用于实现线程同步的关键字,它在 JVM 层面实现锁机制。`Lock` 是一个接口,提供了比 `synchronized` 更加灵活的锁机制[^4]。 #### 2. 功能对比 - **锁的获取与释放** `synchronized` 在方法或代码块执行完毕后自动释放锁,而 `Lock` 需要显式调用 `lock()` 方法获取锁,并在 `finally` 块中调用 `unlock()` 方法释放锁。 - **可中断性** `synchronized` 不支持锁的中断,而 `Lock` 提供了 `tryLock()` `lockInterruptibly()` 等方法,允许线程在等待锁时被中断[^4]。 - **超时功能** `synchronized` 不支持超时机制,而 `Lock` 提供了 `tryLock(long timeout, TimeUnit unit)` 方法,允许线程在指定时间内尝试获取锁。 - **性能表现** 在简单的同步场景下,`synchronized` 的性能通常优于 `Lock`,因为其由 JVM 优化。然而,在需要复杂锁控制的场景下,`Lock` 的灵活性使其成为更好的选择[^1]。 #### 3. 使用示例 ##### (1) `synchronized` 示例 ```java public class Counter { private int count; public synchronized void increment() { count++; } public synchronized int getCount() { return count; } } ``` ##### (2) `Lock` 示例 ```java import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; public class Counter { private int count; private final Lock lock = new ReentrantLock(); public void increment() { lock.lock(); // 显式获取锁 try { count++; } finally { lock.unlock(); // 显式释放锁 } } public int getCount() { lock.lock(); try { return count; } finally { lock.unlock(); } } } ``` #### 4. 使用场景 - **简单同步需求** 如果只需要简单的线程同步,且不需要复杂的锁控制(如中断、超时等),可以优先使用 `synchronized`,因为它更简洁且性能较好[^1]。 - **复杂同步需求** 如果需要更高级的锁控制(如公平锁、读写锁、锁中断等),则应选择 `Lock` 接口[^4]。 - **性能敏感场景** 在高并发场景下,如果需要对锁进行优化(如减少锁竞争),可以考虑使用 `Lock` 提供的多种锁实现(如 `ReentrantLock` 或 `ReadWriteLock`)[^1]。 #### 5. 注意事项 - 使用 `Lock` 时必须确保在 `finally` 块中释放锁,否则可能导致死锁。 - `synchronized` 是不可重入的锁,而 `ReentrantLock` 支持重入,即同一个线程可以多次获取同一把锁[^3]。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值