如何优雅地停下线程?除了这两种,你还能说出一种算我输
前言
今天主要来聊一聊如何优雅地停下线程。
在开始之前,我们可以思考一下,如何能够让线程停下?
通过查阅JDK,我们不难发现Thread为我们提供了一个stop方法,只要使用stop方法,就立即停止线程,但是发现stop()方法被标注为废弃的方法,因为这个方法会强行把执行到一半的线程终止,可能会引发一些数据不一致或者我们没发预估的问题。
除了stop()方法,我能想到的方案还有两个,
方案一:使用volatile标记位,利用其可见性
方案二:调用Thread的方法interrupted方案实现方案一:使用volatile标记位,利用其可见性
通过代码我们来看下方案一,这是一个很经典的生产者和消费者模式。
生产者Demo生产者classProducerimplementsRunnable{publicvolatilebooleancancfalse;privateProductproduct;Producer(Productproduct){this。productproduct;}Overridepublicvoidrun(){try{while(!canc){try{Thread。sleep(1000);product。put(iphone6s);System。out。println(put:Thread。currentThread()。getName());}catch(InterruptedExceptione){e。printStackTrace();}}}catch(Exceptionex){。。。}finally{System。out。println(结束);}}}
消费者Demo消费者classConsumerimplementsRunnable{privateProductproduct;Consumer(Productproduct){this。productproduct;}Overridepublicvoidrun(){while(Math。random()0。9){try{Thread。sleep(1000);product。take(iPhone6s);System。out。println(take:Thread。currentThread()。getName());}catch(InterruptedExceptione){e。printStackTrace();}}}}
调用生产者和消费者publicstaticvoidmain(String〔〕args){ArrayBlockingQueuequeuenewArrayBlockingQueue(1);ProductproductnewProduct(queue);ProducerproducernewProducer(product);ConsumerconsumernewConsumer(product);Threadc1newThread(consumer);Threadp1newThread(producer);p1。start();try{Thread。sleep(500);}catch(InterruptedExceptione){e。printStackTrace();}c1。start();System。out。println(消费者不需要更多数据了。);producer。canctrue;System。out。println(producer。canc);System。out。println(p1。getState());}
场景一:我们把消费者和生产者的线程都开起来,生产者生产一个产品,消费者都会消费一个产品,这个时候volatile的值,在下一次的轮询中值已经变成了true,就跳出while循环,线程就停止,这个场景下volatile就适用了。
场景二:我们将消费者线程不启动,只生产不消费。理论上我们期待的结果应该也是值变成true,跳出while循环,线程停止。
结果打印:Putaiphone6sput:Thread2消费者不需要更多数据了。valatile的值:true线程状态:WAITING
根据打印的结果我们会观察到他没有输出结束的语句,
我们看到了生产者生产了产品,valatile也修改了值,但是线程却没有结束,
这主要的原因是因为,生产者执行了product。put(iphone6s),没有被消费,造成了阻塞,在它唤醒之前,
无法进入下一次的轮询判断。造成了值修改了,却没有做出相应处理。
我们发现在消费的时候,take方法内部会触发唤醒,当检测到线程已经停止,则抛出InterruptedException异常。开源码说话,可以看到dequeue,唤醒了线程。publicEtake()throwsInterruptedException{finalReentrantLocklockthis。lock;lock。lockInterruptibly();try{while(count0)notEmpty。await();returndequeue();}finally{lock。unlock();}}publicfinalvoidacquireInterruptibly(intarg)throwsInterruptedException{if(Thread。interrupted())thrownewInterruptedException();}privateEdequeue(){。。。释放notFull。signal();returnx;}二、方案二:调用Thread的方法interruptedstaticclassCreateRunableimplementsRunnable{publicCreateRunable(inti){this。ii;}privateinti;publicintgetI(){returni;}publicvoidsetI(inti){this。ii;}Overridepublicvoidrun(){synchronized(this){while(!Thread。currentThread()。isInterrupted()){System。out。println(Runable接口,实现线程i);}}}}ThreadcreateThreadnewThread(newCreateRunable(0));createThread。start();Thread。sleep(5);createThread。interrupt();
休眠5毫秒后,该线程检查到了中断信号,就会停止线程。
那如果任务正在休眠状态,线程会如何处理呢Overridepublicvoidrun(){synchronized(this){while(!Thread。currentThread()。isInterrupted()){try{Thread。sleep(10000);}catch(InterruptedExceptione){e。printStackTrace();}System。out。println(Runable接口,实现线程i);}}}
抛出异常,同时清除中断状态,线程会继续执行Runable接口,实现线程0java。lang。InterruptedException:sleepinterruptedatjava。lang。Thread。sleep(NativeMethod)atmain。Thread。threadStartThreeWaysCreateRunable。run(threadStartThreeWays。java:48)atjava。lang。Thread。run(Thread。java:748)Runable接口,实现线程1Runable接口,实现线程2Runable接口,实现线程3总结
我们在这里就不说Stop()方法,因为他太暴力了,不够优雅。这里的优雅指的是可以让线程有时间做好收尾工作,避免数据的错乱。优雅停下线程的方式主要有两种方案一:使用volatile标记位。方案二:调用Thread的方法interrupted。
通过上面的demo案例,我们可以看到使用方案一的volatile,在某一些特殊的场景下,会发生不能关闭线程的情况。
所以volatile是不够全面的。方案二则是一种更优的选择。
新概念英语适合哪种英语水平的人?新概念英语共分四册,下面分别谈一下各册的难度水平第一册最简单,基本上是从零基础开始的,对没有任何英语基础的人来说,他们需要从第一册开始学起。第二册的难度大概属于小学初中的水平,除了
洛阳,南阳信阳,哪个地方更适合旅游?应该说,河南的这三个城市在河南的18个地市里来说,算是比较不错的了。南阳信阳生态会更好一些吧。这三个城市自己都曾去过两次以上,洛阳曾去过多次。说到旅游,就目前的情况来看,个人感觉还
在化工厂上班是什么感受?你好,很高兴回答你的问题!我的感受是非常的无奈,累,危险,工作环境差,我实在是忍不住就走了!我2012年在我们当地的一个化工厂做了两个月就出来了!化工厂可能大家只是听说过,而且还有
对农村白事攀比愈演愈烈花费惊人,你怎么看?坐标鲁西南农村,在我们这里白事的消费大都是一成不变的,不存在什么攀比的情况,死不起这种情况更是子虚乌有的事情,白事上面过度消费是会被严格禁止的,村里面有红白理事会监管,每个家族的长
说说上世纪六七十年代农村过春节,你那里都吃些什么?我就是七零后的农村娃,我最有发言权。六七十年代的中国农村,你想想吧,还没有实行家庭联产承包,大家都是在大集体参加生产劳作。那时候还是公社制度,参加集体劳作的农民吃饭都是在生产大队的
河南北部农村一天三顿饭都吃什么,你最爱吃什么?我在河南洛阳,也不算河南很靠北,但是我觉得饮食都没有什么大的差异。农村的,民风比较淳朴,所以每天吃的东西也比较简单。可能我们在城市中一天能吃三样,在农村,一般一天吃两样东西。早上和
33岁被外贸公司辞退,想转行做会计,好找工作吗?目前在考初级,来得及吗?会计真的是专业性的。可能你也曾听过身边非专业的人做会计吧!刚好我现在的工作是这样的,我做的是一家商贸公司,他们的财务就是非专业的都可以做。我开始也是特别郁闷,因为我用专业来看他们做
唐山有可能成为二线城市吗?真的不需要对几线城市感兴趣,没有明确标准,也没有可比性。城市的发展一定要找好自己的定位,要根据自己的客观条件制定符合自己情况的发展战略。唐山有自身的优势,只是这几年产业结构调整的步
只有女儿没有儿子的父母算不算绝后?按生物学说法,生儿生女都是将血统传递下去,但按古代传统观念确实是绝后了只要有儿子,哪怕这个儿子和自己没有血缘关系,只要跟自己姓,也不算绝后(所以戴绿帽子不算绝后)比如后周郭威,传位
找工作中,面试官问你最大的缺点是什么?该如何回答?我最大的缺点,就是不能及时的发现自己的缺点和错误,并加以改正。这个问题的本质,是面试官在考察的自我认知。自我认知是一个心理学的概念,它的英文叫做selfcognition,讲的是一
喜欢与校长对着干的老师,是什么心理?我曾经就和校长对着干过,因为他一直欺负老实人,老实人忍不住,终于爆发了!我是出身农民,老公也是普通老百姓,没有任何背景和关系。工作全凭自己的兢兢业业,认真负责。有些人当校长,安排工