当前位置: 首页 > news >正文

现在有t1,t2,t3三个线程,实现t1,t2线程同步执行,然后再执行t3线程,使用Java实现该程序

目录

1、利用CountDownLatch

2、利用Future


最近在面试的时候,经常遇到这个题目,首先从题目上看,就知道考察的是多线程方面知识,我第一次看到这个题目的时候,就想到了使用CountDownLatch这个计数器来实现,因为它的原理就是让一个线程或多个线程去等待另外线程执行完毕后再执行的,本篇文章,我打算用两种方式去实现这个题目,分别是使用CountDownLatch和Future来实现。

1、利用CountDownLatch

如果第一次听说CountDownLatch的话,也没用关系,我会用最通俗易懂的话去介绍。

CountDownLatch其实就是一个计数器,在new这个对象的时候,需要在CountDownLatch的参数中传递一个int类型的数字,并且这个int类型必须大于等于0。

而这个count值传递给Sync后,就会调用一个setState方法,这是AQS里面的内容,过多的我就不去讲述,如果有兴趣的话可以去网上搜一下AQS了解一下。这个state有三种,分别是0、1、大于1,首先0就代表着目前还没有线程去占用这个资源,可以抢占,1代表着目前已经有一个线程抢占着这个资源了,不允许其他线程再进来了,这时其他线程就会被放入到一个双向链表队列进行阻塞等待(可以去了解一下AQS的阻塞队列),大于1呢就是说这个资源被重入的次数。所以,利用该状态码值就可以实现阻塞线程。

接着介绍一下CountDownLatch的常用方法

方法名解释
await()让线程去阻塞等待
countDown()使计数器的值-1,直到减为0后,代表所有线程执行完毕
getCount()获取当前计数器值
boolean await(long timeout, TimeUnit unit)

此方法至多会等待指定的时间,超时后会自动唤醒,若 timeout 小于等于零,则不会等待。

若计数器变为零了,则返回 true;若指定的等待时间过去了,则返回 false

CountDownLatch(int count)count为计数器的初始值

示例代码

public class CountDownLatchTest {public static void main(String[] args) throws InterruptedException {CountDownLatch c = new CountDownLatch(2);//初始化一个对象,把计数器定为2new Thread(() -> {System.out.println("我是线程1");c.countDown();//计数器减一}, "t1").start();new Thread(() -> {System.out.println("我是线程2");c.countDown();//计数器减一}, "t2").start();c.await();//调用await方法,让线程3等待前两个线程执行完毕new Thread(() -> {System.out.println("我是线程3");}, "t3").start();}
}

2、利用Future

Future其实就是一个异步任务监视器,可以监视任务的执行,也可以取消任务的执行,同时也可以通过get()方法获取执行后的返回结果。

为了方便演示效果,我使用了有返回结果的Callable来实现。

Future的常用方法

方法解释
get()此方法就是获取线程池提交任务返回的结果,这个方法会一直阻塞其他线程,任务执行完毕后才会获取值。也可以传递超时时间,如果任务超过的超时时间,就会报错。
boolean isDone()判断任务是否运行完毕,如果运行完毕就返回true,否则返回false

boolean isCancelled()

判断任务是否已被取消,如果是返回true,否则false

boolean cancel(boolean mayInterruptIfRunning)

试图取消任务的执行。如果传入的参数是 true,执行任务的线程就会收到一个中断的信号,正在执行的任务可能会有一些处理中断的逻辑,进而停止。

如果传入的是 false 则就代表不中断正在运行的任务,也就是说,本次 cancel 不会有任何效果,同时 cancel 方法会返回 false。

public class FutureTest {public static void main(String[] args) throws ExecutionException, InterruptedException {//首先创建一个线程池ExecutorService executorService = Executors.newCachedThreadPool();//线程池的submit返回值类型就是FutureFuture<String> s1 = executorService.submit(new Callable<String>() {@Overridepublic String call() throws Exception {Thread.sleep(200);//模拟程序运行时间return "我是线程1";}});Future<String> s2 = executorService.submit(new Callable<String>() {@Overridepublic String call() throws Exception {Thread.sleep(200);//模拟程序运行时间return "我是线程2";}});try {System.out.println(s1.get(1000, TimeUnit.MILLISECONDS));} catch (TimeoutException e) {e.printStackTrace();}try {System.out.println(s2.get(1000, TimeUnit.MILLISECONDS));} catch (TimeoutException e) {e.printStackTrace();}if (s1.isDone() && s2.isDone()) {Thread.sleep(300);//模拟程序运行时间System.out.println("我是线程3");}executorService.shutdown();}
}

http://www.lryc.cn/news/61556.html

相关文章:

  • 83.qt qml-初步学习2D粒子影响器(二)
  • 4.17-4.18学习总结
  • Spring事务
  • Linux新的设备或分区挂载到系统中mount使用方法
  • 移动硬盘损坏如何恢复数据
  • Material Design:为你的 Android 应用提供精美的 UI 体验
  • springboot+vue学生毕业离校系统(源码+说明文档)
  • 【Android入门到项目实战-- 6.2】—— 如何访问其他应用程序的数据?
  • 【100个 Unity实用技能】 | InputField输入框组件实现输入限制,只能输入中文或特殊字符等
  • 倍数+路径之谜
  • 【Unity渲染】URP透明物体自身渲染穿插异常问题
  • c/c++:指针,指针定义和使用,指针大小4字节,野指针,空指针*p=NULL
  • CAS实现原⼦操作的三⼤问题,该如何解决?
  • Linux Shell 实现一键部署二进制go+caddy+filebrowser
  • 无人机巡检智能一体化解决方案
  • 2023-2029全球粘结剂喷射技术行业调研及趋势分析报告
  • Python每日一练(20230420)
  • 【社区图书馆】读《悲惨世界》有感
  • 随机蛙跳算法 (SFLA)简单实现(Matlab代码实现)
  • 【手把手做ROS2机器人系统开发二】熟悉ROS2基本命令
  • 如何训练个人的ChatGpt4
  • Rabbit与springboot整合-1
  • 项目沟通管理5大技巧 第4个很重要
  • vector(入门知识点)
  • 成人高考专业怎么选?看这三点
  • 设计模式:UML中的类图(6种关系)
  • 00后卷王的自述,我真有同事口中说的那么卷?
  • VS2022配置OpenGL+GLAD
  • javascript for循环的定义
  • 【安全与风险】安全研究的新课题