java面试题——CountDownLatch/CyclicBarrier/Semaphore

CountDownLatch

定义:让一些线程阻塞直到另一些线程完成一些列操作之后才被唤醒。
即使门闩的意思,给定一个数,当这个数被减到0时才能开始运行另一个线程。

主要有两个方法一个是await方法等待,一个是countDown方法让门闩减一。

实例代码:

public static void closeDoor() throws Exception{

        CountDownLatch countDownLatch = new CountDownLatch(6);//初始化的时候必须给定一个初始门闩值

        for(int i = 1; i <= 6; i++){
            new Thread(()->{
                System.out.println(Thread.currentThread().getName()+"\t 上完自习,离开教室");
                countDownLatch.countDown();//让门闩减一
            },String.valueOf(i)).start();
        }
        countDownLatch.await();//被阻塞,只有在门闩值减到0的时候才可以运行
        System.out.println(Thread.currentThread().getName()+"\t 班长锁门走人");
    }

CyclicBarrier

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

和CountDownLatch相反,这个是加法,而countdownLatch是减法

实例代码:

public class CyclicBarrierDemo {

    public static void main(String[] args) {
        CyclicBarrier cyclicBarrier= new CyclicBarrier(7,() -> {System.out.println("召唤神龙");});


        for(int i = 1; i <= 7; i++){
            final int tempInt = i;
            new Thread(()->{
                System.out.println(Thread.currentThread().getName()+"\t 收集到第"+tempInt+"龙珠");
                try {
                    cyclicBarrier.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } catch (BrokenBarrierException e) {
                    e.printStackTrace();
                }
            },String.valueOf(i)).start();
        }
    }
}

Semaphore

信号量主要用于两个目的,一个是用于多个共享资源的互斥使用,另一个用于并发线程数的控制。

简单的说就是抢车位,30个人来抢20个车位。

很重要的一个,经常会用到

/**
 * @author shihangqi
 * @date 2019/10/25 - 9:46
 */
public class SemaphoreDemo {


    public static void main(String[] args) {
        Semaphore semaphore = new Semaphore(3);//默认非公平锁

        for(int i = 1; i <= 6; i++){
            new Thread(()->{
                try {
                    semaphore.acquire();
                    System.out.println(Thread.currentThread().getName()+"\t 抢到车位");
                    try { TimeUnit.SECONDS.sleep(3); } catch (InterruptedException e){ e.printStackTrace(); }
                    System.out.println(Thread.currentThread().getName()+"\t 停车3秒后离开车位");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }finally {
                    semaphore.release();
                }
            },String.valueOf(i)).start();
        }
    }

}

上一篇:Java 多线程之Semaphore (限流Java 版)


下一篇:在Java中释放Semaphore对象的正确方法是什么?