多线程之CountdownLatch应用

1 应用

1 多个线程准备完毕

模拟游戏在线,多个人同时加载完成.

AtomicInteger num = new AtomicInteger(0);
ExecutorService service = Executors.newFixedThreadPool(10, (r) -> {
     return new Thread(r, "t" + num.getAndIncrement());
});
CountDownLatch latch = new CountDownLatch(10);
String[] all = new String[10];
Random r = new Random();
for (int j = 0; j < 10; j++) {
     int x = j;
     service.submit(() -> {
         for (int i = 0; i <= 100; i++) {
             try {
                 Thread.sleep(r.nextInt(100));
             } catch (InterruptedException e) {
             }
             all[x] = Thread.currentThread().getName() + "(" + (i + "%") + ")";
             System.out.print("\r" + Arrays.toString(all));
         }
         latch.countDown();
     });
}
latch.await();
System.out.println("\n游戏开始...");
service.shutdown();

运行结果:

[t0(100%), t1(100%), t2(100%), t3(100%), t4(100%), t5(100%), t6(100%), t7(100%), t8(100%), 
t9(100%)] 
游戏开始... 

2 多个远程结果聚合

在平常的微服务项目中, 都是通过feign接口,互相调用不同业务的模块,涉及到多个远程调用.

模拟业务调用

@RestController
public class TestCountDownlatchController {
 @GetMapping("/order/{id}")
 public Map<String, Object> order(@PathVariable int id) {
     HashMap<String, Object> map = new HashMap<>();
     map.put("id", id);
     map.put("total", "2300.00");
     sleep(2000);
     return map;
 }
    
 @GetMapping("/product/{id}")
 public Map<String, Object> product(@PathVariable int id) {
 HashMap<String, Object> map = new HashMap<>();
     if (id == 1) {
         map.put("name", "小爱音箱");
         map.put("price", 300);
     } else if (id == 2) {
         map.put("name", "小米手机");
         map.put("price", 2000);
     }
     map.put("id", id);
     sleep(1000);
     return map;
 }
    
 @GetMapping("/logistics/{id}")
 public Map<String, Object> logistics(@PathVariable int id) {
     HashMap<String, Object> map = new HashMap<>();
     map.put("id", id);
     map.put("name", "中通快递");
     sleep(2500);
     return map;
 }
    
 private void sleep(int millis) {
     try {
         Thread.sleep(millis);
     } catch (InterruptedException e) {
         e.printStackTrace();
     }
 }
}

测试

RestTemplate restTemplate = new RestTemplate();
log.debug("begin");
ExecutorService service = Executors.newCachedThreadPool();
CountDownLatch latch = new CountDownLatch(4);
Future<Map<String,Object>> f1 = service.submit(() -> {
     Map<String, Object> r =
     restTemplate.getForObject("http://localhost:8080/order/{1}", Map.class, 1);
     return r;
});
Future<Map<String, Object>> f2 = service.submit(() -> {
     Map<String, Object> r =
     restTemplate.getForObject("http://localhost:8080/product/{1}", Map.class, 1);
     return r;
});
Future<Map<String, Object>> f3 = service.submit(() -> {
     Map<String, Object> r =
     restTemplate.getForObject("http://localhost:8080/product/{1}", Map.class, 2);
     return r;
})
    
Future<Map<String, Object>> f4 = service.submit(() -> {
     Map<String, Object> r =
     restTemplate.getForObject("http://localhost:8080/logistics/{1}", Map.class, 1);
     return r;
});
System.out.println(f1.get());
System.out.println(f2.get());
System.out.println(f3.get());
System.out.println(f4.get());
log.debug("执行完毕");
service.shutdown();

运行结果:

07:51:39.711 c.TestCountDownLatch [main] - begin 
{total=2300.00, id=1} 
{price=300, name=小爱音箱, id=1} 
{price=2000, name=小米手机, id=2} 
{name=中通快递, id=1} 
07:51:42.407 c.TestCountDownLatch [main] - 执行完毕

通过Future带有返回值同样可以起到一样的效果, 和CountDownLatch控制的效果一样.

2 CyclicBarrier

CyclicBarrier称为循环栅栏,用来进行线程协作,等待线程满足某个计数。构造时设置『计数个数』,每个线程执 行到某个需要“同步”的时刻调用 await() 方法进行等待,当等待的线程数满足『计数个数』时,继续执行.

CyclicBarrier cb = new CyclicBarrier(2); // 个数为2时才会继续执行
new Thread(()->{
     System.out.println("线程1开始.."+new Date());
     try {
         cb.await(); // 当个数不足时,等待
     } catch (InterruptedException | BrokenBarrierException e) {
         e.printStackTrace();
     }
     System.out.println("线程1继续向下运行..."+new Date());
}).start();
new Thread(()->{
     System.out.println("线程2开始.."+new Date());
     try { Thread.sleep(2000); } catch (InterruptedException e) { }
     try {
         cb.await(); // 2 秒后,线程个数够2,继续运行
     } catch (InterruptedException | BrokenBarrierException e) {
         e.printStackTrace();
     }
     System.out.println("线程2继续向下运行..."+new Date());
}).start();

从运行结果来看, 都是两秒后,线程才继续往下执行,即当CyclicBarrier线程等于设置值时,才能向下执行后续操作. 当CyclicBarrier归零后, 又恢复设置值, 即可重用机制. ( CountDownLatch没有可重用机制, 类似一次性工具,使用后即功能失效)

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值