JUC高级多线程_06:多线程下得常用辅助类

Posted ABin-阿斌

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JUC高级多线程_06:多线程下得常用辅助类相关的知识,希望对你有一定的参考价值。

我是 ABin-阿斌:写一生代码,创一世佳话,筑一揽芳华。 如果小伙伴们觉得我的文章有点 feel ,那就点个赞再走哦。
在这里插入图片描述

1 . CountDownLatch(减少计数)

  • 做减法,计数器减为 0 才可以继续

1. 例子 - 未使用 CountDownLatch

  • 班长申请了教室组织自习,来了 6 个同学,这 6 个同学学完自行离开,但是班长必须最后一个离开,并锁上门
public class CountDownLatchTest {
    public static void main(String[] args) {
        for (int i = 0; i <= 5; i++) {
            new Thread(() -> {
                System.out.println(Thread.currentThread().getName() + "同学陆续离开教室");
            }, String.valueOf(i)).start();
        }
        System.out.println(Thread.currentThread().getName() + "班长最后离开教室");
    }
}

  • 结果: 会出现班长不是最后一个离开的情况,把其他同学锁在教室里了
  • 在这里插入图片描述

2. 使用 CountDownLatch 后

public class CountDownLatchTest {
    public static void main(String[] args) throws InterruptedException {
        CountDownLatch countDownLatch = new CountDownLatch(6);
        for (int i = 0; i <= 5; i++) {
            //总数是6,必须要执行任务的时候,再使用!
            new Thread(() -> {
                System.out.println(Thread.currentThread().getName() + "同学陆续离开教室");
                //每离开一个同学就减=1
                countDownLatch.countDown();
            }, String.valueOf(i)).start();
        }
        //等同学们都走完了,然后班长再离开
        countDownLatch.await();
        System.out.println(Thread.currentThread().getName() + "班长最后离开教室");
    }
}

3. 原理:

  • CountDownLatch 主要有两个方法,当一个或多个线程调用 await 方法时,这些线程会阻塞。
  • 其它线程调用 countDown 方法会将计数器减1(调用 countDown 方法的线程不会阻塞),
  • 当计数器的值变为0时,因 await 方法阻塞的线程会被唤醒,继续执行。

2 . CyclicBarrier(循环屏障)

  • 做加法: 计数器加到规定的数字才可以继续

1. 例子

  • 6 个人去会议室开会,只有人齐了才可以开始会议
// 一般使用的构造方法 : CyclicBarrier(int parties, Runnable barrierAction)
        CyclicBarrier cyclicBarrier = new CyclicBarrier(6,() -> {
            System.out.println("==== 会议开始 ====");});
        for (int i = 1; i <= 6; i++){
            // 加 final的作用:加了final后,这个变量会存在堆中的方法区里面。
           // 因为这样我们的一个子进程就可以共享进程的堆了,所以也就能读到这个 i 变量了。
           // 否则存在另一个线程的栈中,不同的线程是读取不到的。
            final int peo = i;
            new Thread(()->{
                System.out.println(Thread.currentThread().getName()+"已进入会议室的人数:"+peo+"个");
                try {
                    //等待...
                    cyclicBarrier.await();
                } catch (Exception e) {
                    e.printStackTrace();
                }
            },String.valueOf(i)).start();
        }

2. 原理:

  • CyclicBarrier 的字面意思是可循环(Cyclic)使用的屏障(Barrier)。
  • 它要做的事情是,是让一组线程到达一个屏障(也可以叫同步点)时被阻塞,直到最后一个线程到达屏障时,屏障才会开门,所有,被屏障拦截的线程才会继续干活。
  • 线程进入屏障通过 CyclicBarrierawait() 方法。

3 . Semaphore(信号灯)

  • 类似于抢车位,多辆车,抢多个车位,车位满就等,一空出来就继续抢

1. 例子

  • 6 辆车,抢 3 个车位
Semaphore semaphore = new Semaphore(3);
        for (int i = 1; i <= 6; i++){
            new Thread(()->{
                try {
                    //抢占资源,资源数自动减一
                    semaphore.acquire();
                    System.out.println(Thread.currentThread().getName()+"已占有该车位");
                    //持有资源一段时间
                    try {
                        TimeUnit.SECONDS.sleep(3);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName()+"已离开该车位");

                } catch (InterruptedException e) {
                    e.printStackTrace();
                }finally {
                    //释放持有的资源
                    semaphore.release();
                }
            },String.valueOf(i)).start();
        }

2. 原理:

  • 在信号量上我们定义两种操作:
    • acquire(获取) 当一个线程调用 acquire 操作时,它要么通过成功获取信号量(信号量减1),没有抢到资源的一直等下去,直到有线程释放信号量,或超时。
    • release(释放)实际上会将信号量的值加1,然后唤醒等待的线程。
  • 信号量主要用于两个目的,一个是用于多个共享资源的互斥使用,另一个用于并发线程数的控制。

3. 启发

  • 当把资源数变为 1 ,就等同于 synchronized 同步锁

以上是关于JUC高级多线程_06:多线程下得常用辅助类的主要内容,如果未能解决你的问题,请参考以下文章

Java——多线程高并发系列之JUC三大辅助类(CountDownLatchCyclicBarrierSemaphore)

JUC高级多线程_04:高并发下集合类的具体使用

Java多线程06——JUC并发包02

JUC学习--Java多线程辅助类

JUC高级多线程_02:线程间的通信

JUC高级多线程_03:关于多线程锁的八个常见问题