简述
countDownLatch:类似减法计数器,给计数器一个初始值,当运行一个线程给他计数器值减一( countDownLatch.countDown();
),当计数器归零,唤醒 await()线程。
- 适用场景:当需要其他线程执行一定数量时,在执行的任务(可以是多个);
CyclicBarrier:类似加法计数器,给计数器一个初始值,当(cyclicBarrier.await();
)等待线程数量够初始值数量时,开启线程;
- 适用场景:需要多个线程同时执行时
Semaphone:类似与抢车位,同一时间只能有指定数量得到线程,设置一个资源数量,然后线程进行资源抢占(semaphore.acquire();
),没抢到的阻塞等待抢到资源的线程释放资源然后在进行资源抢占,抢到资源的执行任务结束后释放资源(semaphore.release();
)。
- 适用场景:当资源有限,只允许一定数量的线程执行。(限流)
使用方法:
countDownLatch
public class CountDownLatchDemo {
public static void main(String[] args) {
//创建类 设置倒计时次数
CountDownLatch countDownLatch = new CountDownLatch(5);
// countDownLatch.countDown(); 用countDown进行减数
for (int i = 0; i < 5; i++) {
new Thread(()->{
countDownLatch.countDown();
System.out.println("运行线程"+Thread.currentThread().getName()+"结束");
}).start();
}
//当计数器归0,线程继续执行
try {
countDownLatch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("线程继续执行");
}
}
运行结果
CyclicBarrier
public class CyclicBarrierDemo {
public static void main(String[] args) {
//给一个初始值,当线程数到达初始值时,执行任务
CyclicBarrier cyclicBarrier=new CyclicBarrier(5,()->{
System.out.println("数量够了");
});
for (int i = 0; i < 5; i++) {
final int a = i;
new Thread(() -> {
System.out.println(Thread.currentThread().getName()+"运行");
try {
cyclicBarrier.await();
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"结束");
},a+"").start();
}
}
}
运行结果
Semaphone
public class SemaphoneDemo {
public static void main(String[] args) {
//设置资源数量
Semaphore semaphore = new Semaphore(3);
for (int i = 0; i < 6; i++) {
new Thread(()->{
try {
semaphore.acquire();
System.out.println(Thread.currentThread().getName()+"抢到资源");
//模拟任务时间
TimeUnit.SECONDS.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
}finally {
System.out.println(Thread.currentThread().getName()+"释放资源");
semaphore.release();
}
}).start();
}
}
}
运行结束