2020 面试官会经常问到的三个并发工具类,你都知道吗?
cdl.countDown();//计数器值每次减去 1
System.out.println(Thread.currentThread().getName() + ",end...");
}
}).start();
cdl.await();// 減去为 0,恢复任务继续执行
System.out.println("两个 Thread 执行完毕....");
System.out.println("主线程继续执行.....");
for (int i = 0; i <10; i++) {
System.out.println(Thread.currentThread().getName()+",i:"+i);
}
}
}
从结果可以看出,主线程只有等两个子线程执行完成后才继续执行
## 1.2. **(**屏障)CyclicBarrier
CyclicBarrier 初始化时同样也规定一个数目,然后计算调用了 CyclicBarrier.await()进入等待的线程数。
当线程数达到了这个数目时,所有进入等待状态的线程被唤醒并继续。
CyclicBarrier 就象它名字的意思一样,可看成是个障碍, 所有的线程必须到齐后才能一起通过这个障碍。
CyclicBarrier 初始时还可带一个 Runnable 的参数, 此 Runnable 任务在 CyclicBarrier 的数目达到后,所有其它线程被唤醒前被执行。
package cn.enjoy.mt;
import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;
class Writer extends Thread {
private CyclicBarrier cbr;
public Writer(CyclicBarrier cbr) {
this.cbr = cbr;
}
@Override
public void run() {
try {
System.out.println("线程" + Thread.currentThread().getName() + ",正在写入数据");
Thread.sleep(3000);
System.out.println("线程" + Thread.currentThread().getName() + ",写入数据成功.....");
cbr.await();
System.out.println("所有线程执行完毕..........");
} catch (Exception e) {
e.printStackTrace();
}
}
}
public class CyclicBarrierTest {
public static void main(String[] args) {
CyclicBarrier cyclicBarrier = new CyclicBarrier(5);
for (int i = 0; i < 5; i++) {
Writer writer = new Writer(cyclicBarrier);
writer.start();
}
}
}
## 1.3. **(**计数信号量)Semaphore
Semaphore 是一种基于计数的信号量。
它可以设定一个阈值,基于此,多个线程竞争获取许可信号,做自己的申请后归还,超过阈值后,线程申请许可信号将会被阻塞。
Semaphore 可以用来构建一些对象池,资源池之类的,比如数据库连接池,我们也可以创建计数为 1 的 Semaphore,将其作为一种类似互斥锁的机制,这也叫二元信号量,表示两种互斥状态。它的用法如下:
availablePermits 函数用来获取当前可用的资源数量
wc.acquire(); //申请资源
wc.release();// 释放资源
需求: 一个按摩店只有 3 个技师,但是有 10 个老司机来按摩,那怎么办?
假设 10 的人的编号分别为 1-10,并且 1 号先到技师,10 号最后到技师。那么 1-3 号来的时候必然有可用的技师,顺利按摩,4 号来的时候需要看看前面 3 人是否有人按摩完成出来了,如果有人出来,进去按摩,否则等待。
同样的道理,4-10 号也需要等待正在上按摩的人出来后才能进去按摩,代码:
package cn.enjoy.mt;
import java.util.Random;
import **《一线大厂 Java 面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义》无偿开源 威信搜索公众号【编程进阶路】** java.util.concurrent.Semaphore;
class UserThrad extends Thread {
private String name;
private Semaphore jsShop;
public UserThrad(String name, Semaphore jsShop) {
this.name = name;
this.jsShop = jsShop;
}
@Override
public void run() {
// 剩下的技师
int availablePermits = jsShop.availablePermits();
if (availablePermits > 0) {
System.out.println(name + ",找到小姐姐按摩了....");
} else {
System.out.println(name + ",真可惜,还得等...");
}
try {
// 申请技师
评论