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

猜你喜欢

转载自blog.csdn.net/malipku/article/details/119971411