美文网首页
可重入锁🔒的一点小事儿

可重入锁🔒的一点小事儿

作者: happyleijun | 来源:发表于2018-05-21 11:05 被阅读0次

对于java的可重入锁借助下面这个例子进行一点说明

问题:启动两个线程,线程1依次打印1,2,3;线程2再依次打印4,5,6;然后线程1接着打印7,8,9

package cn.focus.adv.hero.util;

import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * Copyright (C) 1998 - 2016 SOHU Inc., All Rights Reserved.
 * <p>
 *
 * @Author: hanleijun (leijunhan@sohu-inc.com)
 * @Date: 2018/5/21
 */
public class TestLocl {

    public static void main(String[] args) {
        final AtomicInteger value = new AtomicInteger(1);
        Lock lock = new ReentrantLock();
        Condition cThree = lock.newCondition();
        Condition cSix = lock.newCondition();

        Thread thread1 = new Thread(() -> {
            lock.lock();
            System.out.println("A is coming");
            while(value.get() <= 3){
                System.out.println(Thread.currentThread().getName() + " is printing: " + value.get());
                value.incrementAndGet();
            }
            cThree.signal();
            lock.unlock();

            lock.lock();
            try {
                System.out.println("A is coming again");
                cSix.await();
                while(value.get() <=9){
                    System.out.println(Thread.currentThread().getName() + " is printing: " + value.get());
                    value.incrementAndGet();
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }finally {
                lock.unlock();
            }
        });

        Thread thread2 = new Thread(() -> {
            lock.lock();
            System.out.println("2---1");
            try {
//                while(value.get() <= 3){                        // **
                    cThree.await();
//                }
                System.out.println("B is coming");
                while(value.get() <= 6){
                    System.out.println(Thread.currentThread().getName() + " is printing: " + value.get());
                    value.incrementAndGet();
                }
                cSix.signal();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }finally {
                lock.unlock();
            }
        });

        thread1.start();
        thread2.start();
    }
}

· 在线程执行体内的外部变量需要为final的,所以不可以用int或integer,这里用了AtomicInteger这一封装类
· 代码中注释掉掉部分,如果不打开,则程序可能卡死,原因是,如果线程2先于线程1启动,则先await,之后按照正常逻辑,不会有问题,如果是线程1先于线程2启动,则打印1,2,3之后signal了第一个信号量,然后线程2await这个信号量则永远不能得到(先signal后await为非法流程),所以线程2在await之前需要加入条件判断
· 之所以用while而不是if,原因是:为了防止虚假唤醒,当然如果改成if也是可以正常运行的,虚假唤醒指的是await的线程,没有收到针对于自己的notify就被唤醒了,一般处理方式就是在await前加入条件判断,使用while()

相关文章

  • ReentrantLock 源码分析

    锁的基本概念 可重入锁 Reentrant 就是可重入的意思,如果锁具备可重入性,则称作为可重入锁。像synchr...

  • java可重入锁

    可重入概念: java的可重入锁: 可重入锁的一种实现方式: 可重入锁的两种使用例子: 例子1: 例子2: 例子1...

  • Java中的各种锁

    一个线程中的多个流程能不能获取同一把锁:可重入锁和非可重入锁 可重入锁 可重入性:表明了锁的分配机制,是基于线程的...

  • 可重入锁和非可重入锁

    1 可重入锁 (ReentrantLock和synchronized)可重入锁指的是可重复可递归调用的锁,在外层使...

  • Java 可重入锁 公平锁 读写锁

    1.可重入锁 如果锁具备可重入性,则称作为可重入锁。 像synchronized和ReentrantLock都是可...

  • J.U.C-AQS-ReentrantLock

    ReentrantLock(可重入锁)和synchronized区别 可重入性 锁的实现(ReentrantLoc...

  • 可重入锁🔒的一点小事儿

    对于java的可重入锁借助下面这个例子进行一点说明 问题:启动两个线程,线程1依次打印1,2,3;线程2再依次打印...

  • Java温故而知新之Lock与Synchronized

    两者比较: 相关概念解析 可重入锁如果锁具备可重入性,则称作为可重入锁。假设方法A需要获取锁,方法B也需要获取锁,...

  • 各种锁的概念

    锁的概念 可重入不可重入公平锁非公平锁锁中断通过一个故事理解可重入锁的机制 - 小勇DW3 - 博客园[https...

  • 可重入锁-面试题:synchronized是可重入锁吗?

    前言 面试题:synchronized是可重入锁吗? 答案:synchronized是可重入锁。Reentrant...

网友评论

      本文标题:可重入锁🔒的一点小事儿

      本文链接:https://www.haomeiwen.com/subject/bxffjftx.html