JAVA线程协作:ReentrantLock重入锁

一、ReentrantLock

我们都知道锁是为了保护临界区资源被多线程并发访问时的安全性。 而 ReentrantLock是JAVA提供的完全可以替代Synchronized的方案。在JDK6.0以前Synchronized的执行效率远远比ReentrantLock的执行效率差。而在JDK6.0以后对Synchronized的优化后,ReentrantLock与Synchronized的执行效率差距不大。

二、ReentrantLock的lock与unlock

  • ReentrantLock的lock与unlock是一套灵活的保护临界区资源的方法。我们需要注意如果在使用lock的时候,我们必须要执行unlock操作。如果我们只调用了lock而没有调用unlock的释放锁的情况下,会造成其他线程获取不到锁,而导致阻塞。
public class ReentrantLockDemo implements Runnable{

    public ReentrantLock lock = new ReentrantLock();
    public static int count = 0;

    @Override
    public void run() {
        for (int i = 0; i < 10000; i++) {
            try {
                //通过重入锁,防止并发增加Count的值
                lock.lock();
                count++;
            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                //使用重入锁,一定要解锁,如果没有解锁会导致其他线程不可以访问。
                lock.unlock();
            }
        }
    }
    public static void main(String[] args) throws InterruptedException {
        ReentrantLockDemo reentrantLockDemo = new ReentrantLockDemo();
        Thread t1 = new Thread(reentrantLockDemo);
        Thread t2 = new Thread(reentrantLockDemo);
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("Over Count="+count);
    }
}
  • ReentrantLock有一种特性就是ReentrantLock的锁可以反复的进入,而这种反复的进入仅限于统一线程下,也就是说我们可以将代码改成下面这种形式。但是我们要记住一点,重复进入锁调时,一定要释放锁.
public class ReentrantLockDemo implements Runnable{

    public ReentrantLock lock = new ReentrantLock();
    public static int count = 0;

    @Override
    public void run() {
        for (int i = 0; i < 10000; i++) {
            try {
                //通过重入锁,防止并发增加Count的值
                lock.lock();
                //重复进入锁
                lock.lock();
                count++;
            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                //使用重入锁,一定要解锁,如果没有解锁会导致其他线程不可以访问。
                lock.unlock();
                lock.unlock();
            }
        }
    }
    
    public static void main(String[] args) throws InterruptedException {
        ReentrantLockDemo reentrantLockDemo = new ReentrantLockDemo();
        Thread t1 = new Thread(reentrantLockDemo);
        Thread t2 = new Thread(reentrantLockDemo);
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("Over Count="+count);
    }
}

二、ReentrantLock的lockInterruptibly

我们知道使用Synchronized的时候,如果锁被其他线程获取。而当前线程只能等在其他线程释放锁,并且获得锁才会执行,没有获得锁时处于阻塞状态。而ReentrantLock提供了lockInterruptibly来优化这种情况。就是通过中断锁。例如我们在死锁的情况下可以通过中断其中一个线程的锁来解决这种死锁的状态。

public class IntReentrantLock implements Runnable {

    public ReentrantLock   r1 = new ReentrantLock();
    public ReentrantLock   r2 = new ReentrantLock();
    int lock = 0;
    //控制死锁
    public IntReentrantLock(int lock) {
        this.lock = lock;
    }

    @Override
    public void run() {
        try {
            if(lock == 1){
                //使用lockInterruptibly表明当前的锁可以被终止.
                r1.lockInterruptibly();
                try {
                    Thread.sleep(5000);
                } catch (InterruptedException e) {

                }
                r2.lockInterruptibly();
            }else{
                r2.lockInterruptibly();
                try {
                    //线程在睡眠的时候被中断,会抛出InterruptedException异常
                    Thread.sleep(5000);
                } catch (InterruptedException e) {
                    
                }
                r1.lockInterruptibly();
            }
        }catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            if(r1.isHeldByCurrentThread()){
                r1.unlock();
            }
            if(r2.isHeldByCurrentThread()){
                r2.unlock();
            }
            System.out.println("Thread :"+Thread.currentThread().getId()+"-"+Thread.currentThread().getName()+":当前线程退出");
        }
    }
    
    public static void main(String[] args) throws InterruptedException {
        IntReentrantLock testLock1 = new IntReentrantLock(1);
        IntReentrantLock testLock2 = new IntReentrantLock(2);
        Thread t1 = new Thread(testLock1,"线程1");
        Thread t2 = new Thread(testLock2,"线程2");
        t1.start();
        t2.start();
        Thread.sleep(1000);
        //中断线程2
        t2.interrupt();
    }
}
输出结果:
Thread :11-线程2:当前线程退出
Thread :10-线程1:当前线程退出

上面的代码是通过模拟死锁的状态。testLock1调用的时候需要用到testLock2 的锁,testLock2 调用的时候需要用到testLock1的锁,而双方有拿着各自的锁。导致死锁。我们通过对线程t2进行终端从而解决这种死锁的状态。

三、ReentrantLock的trylock

除了中断锁通知的这种操作处理死锁的这种情况,我们也可以通过设置获取锁的等待时间来进行处理。

public class TryLock implements Runnable {

    public ReentrantLock lock = new ReentrantLock();

    @Override
    public void run() {
        try {
            //等待获取锁
            if (lock.tryLock(3, TimeUnit.SECONDS)) {
                System.out.println("get Lock Success");
                Thread.sleep(5000);
            } else {
                System.out.println("get Lock Faild");
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            if (lock.isHeldByCurrentThread()) {
                System.out.println("releas Lock Success");
                lock.unlock();
            }
        }
    }

    public static void main(String[] args) throws InterruptedException {
        TryLock tryLock = new TryLock();
        Thread t1 = new Thread(tryLock);
        Thread t2 = new Thread(tryLock);
        t1.start();
        t2.start();
        t1.join();
        t2.join();
    }

}

输出结果:
get Lock Success
get Lock Faild
releas Lock Success

四、ReentrantLock的condition

我们在使用Synchronized的时候,当某些时候需要在双线程的协同处理的时候会使用到了wait()和notify()进行处理这种协同处理。而ReentrantLock也提供了这种协同处理就是await()和signal()。

public class ReentrantLockCondition implements Runnable{
    
    public static ReentrantLock lock = new ReentrantLock(); 

    public static Condition condition = lock.newCondition();
     
    @Override
    public void run() {
        try {
            lock.lock();
            condition.await();
            System.out.println("out lock wait");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
    }

    public static void main(String[] args) throws InterruptedException {
        ReentrantLockCondition reentrantLockCondition= new ReentrantLockCondition();
        Thread t1 = new Thread(reentrantLockCondition);
        t1.start();
        Thread.sleep(2000);
        lock.lock();
        //唤醒t1线程的等待
        condition.signal();
        System.out.println("signal Thread t1");
        //释放当前的主线程的锁,如果没有释放则唤醒的t1是不可能获取到锁,而t1唤醒后获取不到锁会处理阻塞状态
        lock.unlock();
    }
}

输出结果:
signal Thread t1
out lock wait
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,313评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,369评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,916评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,333评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,425评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,481评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,491评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,268评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,719评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,004评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,179评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,832评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,510评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,153评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,402评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,045评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,071评论 2 352

推荐阅读更多精彩内容

  • 作者: 一字马胡 转载标志 【2017-11-03】 更新日志 前言 在java中,锁是实现并发的关键组件,多个...
    一字马胡阅读 44,149评论 1 32
  • 摘要: 我们已经知道,synchronized 是Java的关键字,是Java的内置特性,在JVM层面实现了对临界...
    kingZXY2009阅读 1,830评论 0 20
  •  既然java内置了synchronized,为什么还要出现lock呢? 由于synchronized的并发是阻塞...
    扈扈哈嘿阅读 898评论 0 1
  • 愿我快乐,祝自己平安。 愿我高瞻远瞩,看到眼前。 愿我生命长长久久,朝发夕至。 愿我命中知遇好人,指导良善。 愿我...
    大爱无痕阅读 233评论 0 0
  • 什么时候我们开始找不到发自内心的笑, 把虚伪变成外衣, 把假面戴在脸上, 压箱底的校服上, 那充满豪情壮志的签名,...
    狂野番茄阅读 89评论 0 1