Doug Lea是否在重复造轮子之ReentrantLock如何响应中断

synchronized在JDK1.6之后经过改进, 在性能方便已经不再被诟病, 但为什么大师Doug Lea还要开发一套新的并发包来改进java并发编程的开发呢?
原因在于: Doug Lea的并发包并非是为了重复造轮子, 弥补的也不是synchronized这把重量级锁, 在死锁场景中, synchronized代码块要依次获取两把锁, 当死锁发生时, 各自持有一把锁的两个线程, 都被阻塞住, 这个时候谁也不退让, 谁也争不到另一把锁, 于是, 程序阻塞住, 除去重启别无他法. 于是大师在ReentrantLock这把锁中, 实现了让阻塞的线程可以响应中断,这样, 我们就无需在不小心发生死锁时候重启了.
下面的程序, 依次复现了死锁时候synchronized方式和ReentrantLock方式如何响应中断的对比效果.

注释写的很详细了, 不再做过多文字说明.


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

/**
 * 功能说明:测试ReentrantLock比synchronized的优势
 * 对比使用synchronized与ReentrantLock两种方式, 当前线程被中断后, 是否会响应中断, 进而允许我们手动释放锁
 * 开发人员:@Author MaLi
 */
public class T12_ReentrantLock extends ReentrantLock {


    //测试synchronized是否能响应中断
    public void testSynchronized(T12_ReentrantLock other) {
        synchronized (this) {
            System.out.println(Thread.currentThread().getName() + " 获取到一把锁");
            //睡眠一段时间, 保证线程2启动并获取到第一把锁
            try {
                TimeUnit.SECONDS.sleep(2);
            } catch (InterruptedException e) {
                e.printStackTrace();
                // 和ReentrantLock的区别是: 使用synchronized方式不会自动释放锁
                // 进入死锁阻塞后, 甚至我们也没有办法捕获InterruptedException异常, 即使捕获异常也没有办法手动释放锁
            }
            synchronized (other) {
                System.out.println(Thread.currentThread().getName() + " 获取到两把锁");
            }
        }
    }

    //测试ReentrantLock是否能响应中断
    public void testReentrantLock(T12_ReentrantLock other) {
        try {
            // 获取第一把锁
            this.lockInterruptibly();
            System.out.println(Thread.currentThread().getName() + " 获取到一把锁");
            //睡眠一段时间, 保证线程2启动并获取到第一把锁
            try {
                TimeUnit.SECONDS.sleep(2);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            // 获取第二把锁
            other.lockInterruptibly();
            System.out.println(Thread.currentThread().getName() + " 获取到两把锁");
        } catch (InterruptedException e) {
            e.printStackTrace();
            // 和synchronized方式的区别, 如果进入死锁阻塞后, 我们可以在外界Interrupted当前线程
            // 在这里就可以捕获到异常, 手动释放锁
            // 锁被释放后, 另一个线程就可以获取到当前线程本来持有的锁
            this.unlock(); // 释放当前线程持有的锁
        }
    }

    public static void main(String[] args) {
        T12_ReentrantLock instance1 = new T12_ReentrantLock();
        T12_ReentrantLock instance2 = new T12_ReentrantLock();
        //测试synchronized
        /* 输出结果试验结果
            Thread_01 获取到一把锁
            Thread_02 获取到一把锁
            interrupt thread_01
         */
//        Thread thread_01 = new Thread(() -> instance1.testSynchronized(instance2), "Thread_01");
//        Thread thread_02 = new Thread(() -> instance2.testSynchronized(instance1), "Thread_02");

        //测试ReentrantLock
        /* 输出结果试验结果
            Thread_01 获取到一把锁
            Thread_02 获取到一把锁
            interrupt thread_01
            Thread_02 获取到两把锁    <--  这里在Thread_01释放掉持有的第一把锁后, Thread_02获取到了第二把锁.
            java.lang.InterruptedException
                at java.util.concurrent.locks.AbstractQueuedSynchronizer.doAcquireInterruptibly(AbstractQueuedSynchronizer.java:898)
                at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireInterruptibly(AbstractQueuedSynchronizer.java:1222)
                at java.util.concurrent.locks.ReentrantLock.lockInterruptibly(ReentrantLock.java:335)
                at mchapter04.T12_ReentrantLock.testReentrantLock(T12_ReentrantLock.java:43)
                at mchapter04.T12_ReentrantLock.lambda$main$0(T12_ReentrantLock.java:69)
                at java.lang.Thread.run(Thread.java:748)
         */
        Thread thread_01 = new Thread(() -> instance1.testReentrantLock(instance2), "Thread_01");
        Thread thread_02 = new Thread(() -> instance2.testReentrantLock(instance1), "Thread_02");

        thread_01.start();
        thread_02.start();
        try {
            TimeUnit.SECONDS.sleep(5);
            System.out.println("interrupt thread_01");
            thread_01.interrupt();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

Love Coding. Mark

  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

能力工场小马哥

如果对您有帮助, 请打赏支持~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值