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();
}
}
}