线程间通信(线程间的通信方式三种)
线程间通信(线程间的通信方式三种)
前言
开发中不免会遇到需要所有子线程执行完毕通知主线程处理某些逻辑的场景。
或者是线程 A 在执行到某个条件通知线程 B 执行某个操作。
可以通过以下几种方式实现:等待通知机制
等待通知模式是 Java 中比较经典的线程通信方式。
两个线程通过对同一对象调用等待 wait() 和通知 notify() 方法来进行通讯。
如两个线程交替打印奇偶数:publicclassTwoThreadWaitNotify{privateintstart=1;privatebooleanflag=false;publicstaticvoidmain(String[]args){ TwoThreadWaitNotifytwoThread=newTwoThreadWaitNotify(); Threadt1=newThread(newOuNum(twoThread)); t1.setName("A"); Threadt2=newThread(newJiNum(twoThread)); t2.setName("B"); t1.start(); t2.start(); }/** *偶数线程 */ publicstaticclassOuNumimplementsRunnable{privateTwoThreadWaitNotifynumber;publicOuNum(TwoThreadWaitNotifynumber){this.number=number; }@Override publicvoidrun(){while(number.start<=100){synchronized(TwoThreadWaitNotify.class){ System.out.println("偶数线程抢到锁了");if(number.flag){ System.out.println(Thread.currentThread().getName()+"+-+偶数"+number.start); number.start++; number.flag=false; TwoThreadWaitNotify.class.notify(); }else{try{ TwoThreadWaitNotify.class.wait(); }catch(InterruptedExceptione){ e.printStackTrace(); } } } } } }/** *奇数线程 */ publicstaticclassJiNumimplementsRunnable{privateTwoThreadWaitNotifynumber;publicJiNum(TwoThreadWaitNotifynumber){this.number=number; }@Override publicvoidrun(){while(number.start<=100){synchronized(TwoThreadWaitNotify.class){ System.out.println("奇数线程抢到锁了");if(!number.flag){ System.out.println(Thread.currentThread().getName()+"+-+奇数"+number.start); number.start++; number.flag=true; TwoThreadWaitNotify.class.notify(); }else{try{ TwoThreadWaitNotify.class.wait(); }catch(InterruptedExceptione){ e.printStackTrace(); } } } } } } }
输出结果:t2+-+奇数93 t1+-+偶数94 t2+-+奇数95 t1+-+偶数96 t2+-+奇数97 t1+-+偶数98 t2+-+奇数99 t1+-+偶数100
这里的线程 A 和线程 B 都对同一个对象 TwoThreadWaitNotify.class 获取锁,A 线程调用了同步对象的 wait() 方法释放了锁并进入 WAITING 状态。
B 线程调用了 notify() 方法,这样 A 线程收到通知之后就可以从 wait() 方法中返回。
这里利用了 TwoThreadWaitNotify.class 对象完成了通信。
有一些需要注意:
wait() 、notify()、notifyAll() 调用的前提都是获得了对象的锁(也可称为对象监视器)。
调用 wait() 方法后线程会释放锁,进入 WAITING 状态,该线程也会被移动到等待队列中。
调用 notify() 方法会将等待队列中的线程移动到同步队列中,线程状态也会更新为 BLOCKED
从 wait() 方法返回的前提是调用 notify() 方法的线程释放锁,wait() 方法的线程获得锁。
等待通知有着一个经典范式:
线程 A 作为消费者:
获取对象的锁。
进入 while(判断条件),并调用 wait() 方法。
当条件满足跳出循环执行具体处理逻辑。
线程 B 作为生产者:
获取对象锁。
更改与线程 A 共用的判断条件。
调用 notify() 方法。
伪代码如下://ThreadAsynchronized(Object){while(条件){Object.wait(); }//dosomething}//ThreadBsynchronized(Object){ 条件=false;//改变条件 Object.notify(); }join() 方法privatestaticvoidjoin()throwsInterruptedException{ Threadt1=newThread(newRunnable(){@Override publicvoidrun(){ LOGGER.info("running");try{ Thread.sleep(3000); }catch(InterruptedExceptione){ e.printStackTrace(); } } }); Threadt2=newThread(newRunnable(){@Override publicvoidrun(){ LOGGER.info("running2");try{ Thread.sleep(4000); }catch(InterruptedExceptione){ e.printStackTrace(); } } }); t1.start(); t2.start();//等待线程1终止 t1.join();//等待线程2终止 t2.join(); LOGGER.info("mainover"); }
输出结果:2018-03-1620:21:30.967[Thread-1]INFOc.c.actual.ThreadCommunication-running22018-03-1620:21:30.967[Thread-0]INFOc.c.actual.ThreadCommunication-running2018-03-1620:21:34.972[main]INFOc.c.actual.ThreadCommunication-mainover
在 t1.join() 时会一直阻塞到 t1 执行完毕,所以最终主线程会等待 t1 和 t2 线程执行完毕。
其实从源码可以看出,join() 也是利用的等待通知机制:
核心逻辑:while(isAlive()){wait(0); }
在 join 线程完成后会调用 notifyAll() 方法,是在 JVM 实现中调用,所以这里看不出来。volatile 共享内存
因为 Java 是采用共享内存的方式进行线程通信的,所以可以采用以下方式用主线程关闭 A 线程:publicclassVolatileimplementsRunnable{privatestaticvolatilebooleanflag=true;@Override publicvoidrun(){while(flag){ System.out.println(Thread.currentThread().getName()+"正在运行…"); } System.out.println(Thread.currentThread().getName()+"执行完毕"); }publicstaticvoidmain(String[]args)throwsInterruptedException{ VolatileaVolatile=newVolatile();newThread(aVolatile,"threadA").start(); System.out.println("main线程正在运行"); TimeUnit.MILLISECONDS.sleep(100); aVolatile.stopThread(); }privatevoidstopThread(){ flag=false; } }
输出结果:threadA正在运行…threadA正在运行…threadA正在运行…threadA正在运行…threadA执行完毕
这里的 flag 存放于主内存中,所以主线程和线程 A 都可以看到。
flag 采用 volatile 修饰主要是为了内存可见性,更多内容可以查看这里。CountDownLatch 并发工具
CountDownLatch 可以实现 join 相同的功能,但是更加的灵活。privatestaticvoidcountDownLatch()throwsException{intthread=3;longstart=System.currentTimeMillis();finalCountDownLatchcountDown=newCountDownLatch(thread);for(inti=0;i<thread;i++){newThread(newRunnable(){@Override publicvoidrun(){ LOGGER.info("threadrun");try{ Thread.sleep(2000); countDown.countDown(); LOGGER.info("threadend"); }catch(InterruptedExceptione){ e.printStackTrace(); } } }).start(); } countDown.await();longstop=System.currentTimeMillis(); LOGGER.info("mainovertotaltime={}",stop-start); }
输出结果:2018-03-1620:19:44.126[Thread-0]INFOc.c.actual.ThreadCommunication-threadrun2018-03-1620:19:44.126[Thread-2]INFOc.c.actual.ThreadCommunication-threadrun2018-03-1620:19:44.126[Thread-1]INFOc.c.actual.ThreadCommunication-threadrun2018-03-1620:19:46.136[Thread-2]INFOc.c.actual.ThreadCommunication-threadend2018-03-1620:19:46.136[Thread-1]INFOc.c.actual.ThreadCommunication-threadend2018-03-1620:19:46.136[Thread-0]INFOc.c.actual.ThreadCommunication-threadend2018-03-1620:19:46.136[main]INFOc.c.actual.ThreadCommunication-mainovertotaltime=2012
CountDownLatch 也是基于 AQS(AbstractQueuedSynchronizer) 实现的,更多实现参考 ReentrantLock 实现原理
初始化一个 CountDownLatch 时告诉并发的线程,然后在每个线程处理完毕之后调用 countDown() 方法。
该方法会将 AQS 内置的一个 state 状态 -1 。
最终在主线程调用 await() 方法,它会阻塞直到 state == 0 的时候返回。CyclicBarrier 并发工具privatestaticvoidcyclicBarrier()throwsException{ CyclicBarriercyclicBarrier=newCyclicBarrier(3);newThread(newRunnable(){@Override publicvoidrun(){ LOGGER.info("threadrun");try{ cyclicBarrier.await(); }catch(Exceptione){ e.printStackTrace(); } LOGGER.info("threadenddosomething"); } }).start();newThread(newRunnable(){@Override publicvoidrun(){ LOGGER.info("threadrun");try{ cyclicBarrier.await(); }catch(Exceptione){ e.printStackTrace(); } LOGGER.info("threadenddosomething"); } }).start();newThread(newRunnable(){@Override publicvoidrun(){ LOGGER.info("threadrun");try{ Thread.sleep(5000); cyclicBarrier.await(); }catch(Exceptione){ e.printStackTrace(); } LOGGER.info("threadenddosomething"); } }).start(); LOGGER.info("mainthread"); }
CyclicBarrier 中文名叫做屏障或者是栅栏,也可以用于线程间通信。
它可以等待 N 个线程都达到某个状态后继续运行的效果。
首先初始化线程参与者。
调用 await() 将会在所有参与者线程都调用之前等待。
直到所有参与者都调用了 await() 后,所有线程从 await() 返回继续后续逻辑。
运行结果:2018-03-1822:40:00.731[Thread-0]INFOc.c.actual.ThreadCommunication-threadrun2018-03-1822:40:00.731[Thread-1]INFOc.c.actual.ThreadCommunication-threadrun2018-03-1822:40:00.731[Thread-2]INFOc.c.actual.ThreadCommunication-threadrun2018-03-1822:40:00.731[main]INFOc.c.actual.ThreadCommunication-mainthread2018-03-1822:40:05.741[Thread-0]INFOc.c.actual.ThreadCommunication-threadenddosomething2018-03-1822:40:05.741[Thread-1]INFOc.c.actual.ThreadCommunication-threadenddosomething2018-03-1822:40:05.741[Thread-2]INFOc.c.actual.ThreadCommunication-threadenddosomething
可以看出由于其中一个线程休眠了五秒,所有其余所有的线程都得等待这个线程调用 await() 。
该工具可以实现 CountDownLatch 同样的功能,但是要更加灵活。甚至可以调用 reset() 方法重置 CyclicBarrier (需要自行捕获 BrokenBarrierException 处理) 然后重新执行。线程响应中断publicclassStopThreadimplementsRunnable{@Override publicvoidrun(){while(!Thread.currentThread().isInterrupted()){//线程执行具体逻辑 System.out.println(Thread.currentThread().getName()+"运行中…"); } System.out.println(Thread.currentThread().getName()+"退出…"); }publicstaticvoidmain(String[]args)throwsInterruptedException{ Threadthread=newThread(newStopThread(),"threadA"); thread.start(); System.out.println("main线程正在运行"); TimeUnit.MILLISECONDS.sleep(10); thread.interrupt(); } }
输出结果:threadA运行中…threadA运行中…threadA退出…
可以采用中断线程的方式来通信,调用了 thread.interrupt() 方法其实就是将 thread 中的一个标志属性置为了 true。
并不是说调用了该方法就可以中断线程,如果不对这个标志进行响应其实是没有什么作用(这里对这个标志进行了判断)。
但是如果抛出了 InterruptedException 异常,该标志就会被 JVM 重置为 false。线程池 awaitTermination() 方法
如果是用线程池来管理线程,可以使用以下方式来让主线程等待线程池中所有任务执行完毕:privatestaticvoidexecutorService()throwsException{ BlockingQueue<Runnable>queue=newLinkedBlockingQueue<>(10); ThreadPoolExecutorpoolExecutor=newThreadPoolExecutor(5,5,1,TimeUnit.MILLISECONDS,queue); poolExecutor.execute(newRunnable(){@Override publicvoidrun(){ LOGGER.info("running");try{ Thread.sleep(3000); }catch(InterruptedExceptione){ e.printStackTrace(); } } }); poolExecutor.execute(newRunnable(){@Override publicvoidrun(){ LOGGER.info("running2");try{ Thread.sleep(2000); }catch(InterruptedExceptione){ e.printStackTrace(); } } }); poolExecutor.shutdown();while(!poolExecutor.awaitTermination(1,TimeUnit.SECONDS)){ LOGGER.info("线程还在执行…"); } LOGGER.info("mainover"); }
克己复礼什么意思(克己复礼的传统解释是正确的吗?)克己复礼什么意思(克己复礼的传统解释是正确的吗?)克己复礼是孔子学说的一个重要概念,出自论语颜渊一章颜渊问仁。子曰克己复礼为仁。一日克己复礼,天下归仁焉。为仁由己,而由人乎哉?颜渊
数码知识oppoa9怎么开启人像模式人像模式怎么打开如今使用IT数码设备的小伙伴们是越来越多了,那么IT数码设备当中是有很多小技巧的,这些技巧很多小伙伴一般都是不知道如何来实用的,就好比最近就有很多小伙伴们想要知道oppoa9怎么开
黄绮珊年龄(离不开你黄绮珊)黄绮珊年龄(离不开你黄绮珊)原创会火群像20210517170527近日,知名女歌手黄绮珊在社交平台中发布了一个在家做饭的视频,双手抓着一条大鱼看起来相当接地气,看起来跟普通的做饭
手机怎么查定位自己去过哪里?(怎么定位手机位置)手机怎么查定位自己去过哪里?(怎么定位手机位置)现在的社会,手机现已成为了万众随身的一种设备,它能够上网和联络朋友,能够作为人们的掌上电脑。现在的手机大多数都是智能化的,能够和电脑
四大守财奴(中外四大守财奴)四大守财奴(中外四大守财奴)原创东观校书郎20210320072000文艺作品中被梁冀追杀的清河王刘蒜的形象东汉中后期有一个很有权势的外戚,名叫梁冀,曾经拥立过三位皇帝毒死过一位皇
电位器的作用(电位器怎么用)电位器的作用(电位器怎么用)电位器是具有三个引出端阻值可按某种变化规律调节的电阻元件。电位器通常由电阻体和可移动的电刷组成。当电刷沿电阻体移动时,在输出端即获得与位移量成一定关系的
数码知识红米5Plus拍照怎么样红米5Plus摄像头参数如今使用IT数码设备的小伙伴们是越来越多了,那么IT数码设备当中是有很多小技巧的,这些技巧很多小伙伴一般都是不知道如何来实用的,就好比最近就有很多小伙伴们想要知道红米5Plus拍照
三星拍照手机(拍照什么手机好看)三星拍照手机(拍照什么手机好看)S21Ultra正面三星s21Ultra可以说是今年的安卓机皇是准没错的。搭载高通最强处理器骁龙888,性能强的没话说,但是其发热也被大家吐槽的可以
手机拍照(2021年十大最佳拍照手机排名)手机拍照(2021年十大最佳拍照手机排名)今天分享有关手机摄影的三个基础的操作技巧,你可以不懂大光圈功能,也可以不会慢门操作,甚至不熟悉专业模式都行,但这三个操作技巧你必须学会并且
扯淡碑(史上最扯淡的石碑)扯淡碑(史上最扯淡的石碑)今天咱们来讲一讲那些疑似穿越的文物。穿越这个话题已经火了好些年了,穿越爱好者也发挥出了强大的行动能力,在历史中翻找穿越者的痕迹,你别说他们找到的东西还真的
数码知识vivox27怎么开启虚拟定位如何开启虚拟定位如今使用IT数码设备的小伙伴们是越来越多了,那么IT数码设备当中是有很多小技巧的,这些技巧很多小伙伴一般都是不知道如何来实用的,就好比最近就有很多小伙伴们想要知道vivox27怎么
乒乓球事件(乒乓球十大热点事件)乒乓球事件(乒乓球十大热点事件)本人鄙陋,见识浅薄,斗胆对2015年乒乓球十大热点事件盘点一番,管窥之见,难免挂一漏万,不当之处,敬请指正。一马龙大四喜加身2015年是马龙大丰收之
男乒世界排名(国际乒联最新排名)男乒世界排名(国际乒联最新排名)国际乒联近日公布了2021年第8期世界排名,国乒队员占据男单女单男双和混双四个单项的头把交椅。国际乒联最新男单排名。图片来源国际乒联官网全新改版的2
快递停运(中国所有快递公司排名)快递停运(中国所有快递公司排名)5月28日024时,31个省(自治区直辖市)和新疆生产建设兵团报告新增确诊病例16例,其中境外输入病例14例(上海7例,四川2例,陕西2例,浙江1例
40岁的女人(过了四十岁的女人,对性还有需求吗?)40岁的女人(过了四十岁的女人,对性还有需求吗?)我是雪落无尘,你有故事,就来找我。点击上面关注,你的心事有我愿意听。都说女人四十到了如狼似虎的年龄,这个形容词实际上所说的,就是女
怀化市哪里(怀化市有多大)中方县,隶属于湖南省怀化市,地处湖南省西南部,怀化市中部,东接溆浦县南邻洪江市西界芷江县北依辰溪县,西北环抱鹤城区。距离怀化市区最短距离6公里。中方1998年4月建县,曾被誉为共和
爱买卖网(鞋子一手货源)爱买卖网(鞋子一手货源)本期再给大家解决一个大痛点,无论是自由职业者还是实体行业的朋友,做生意都需要找货源,可能绝大多数人只知道阿里巴巴批发网,但其实现在阿里巴巴已经开始一键批发,
数码知识拼多多怎么一件代发货源一件代发货源教程如今使用IT数码设备的小伙伴们是越来越多了,那么IT数码设备当中是有很多小技巧的,这些技巧很多小伙伴一般都是不知道如何来实用的,就好比最近就有很多小伙伴们想要知道拼多多怎么一件代发
怎么定位对方手机位置不被发现(悄悄定位老公手机位置)怎么定位对方手机位置不被发现(悄悄定位老公手机位置)这篇文章咱们聊一下如何通过手机定位你的男朋友或者是女朋友的位置。其实这个不仅仅适用于情侣之间。也可以通过这个办法寻找你的家人。在
进口音响品牌!(世界十大音响品牌你知道几个?)进口音响品牌!(世界十大音响品牌你知道几个?)音响什么牌子音质好,就是厂商推出的整体性的音响套装机,其功能尽可能齐全,使用方便,外观华丽。组合音响的所有的组成部分,如音箱功放卡座C
教你看穿一个人(教你看穿人心,嘴笨老实人再也不吃亏!)教你看穿一个人(教你看穿人心,嘴笨老实人再也不吃亏!)在现在这样一个相对比较混乱的社会中,可能每一个人都有自己不能说的话,可能他们都戴着面具生活,所以我们在一定程度上判断对方究竟是
看穿一个人(心理学怎么看透一个人)看穿一个人(心理学怎么看透一个人)三国时期的诸葛亮,是运筹帷幄的绝代谋神,智慧的象征,古人中的头脑担当。他担任丞相,协助刘备刘禅父子,将蜀国治理得井井有条。无论是行军打仗,还是处理