java高并发系列第11篇文章。
本文主要探讨一下中断线程的几种方式。
代码:
package com.itsoku.chat05; import java.util.concurrent.timeunit; /** * 微信公众号:路人甲java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注! */ public class demo1 { public volatile static boolean exit = false; public static class t extends thread { @override public void run() { while (true) { //循环处理业务 if (exit) { break; } } } } public static void setexit() { exit = true; } public static void main(string[] args) throws interruptedexception { t t = new t(); t.start(); timeunit.seconds.sleep(3); setexit(); } }
代码中启动了一个线程,线程的run方法中有个死循环,内部通过exit变量的值来控制是否退出。timeunit.seconds.sleep(3);
让主线程休眠3秒,此处为什么使用timeunit?timeunit使用更方便一些,能够很清晰的控制休眠时间,底层还是转换为thread.sleep实现的。程序有个重点:volatile关键字,exit变量必须通过这个修饰,如果把这个去掉,程序无法正常退出。volatile控制了变量在多线程中的可见性,关于volatile前面的文章中有介绍,此处就不再说了。
示例代码:
package com.itsoku.chat05; import java.util.concurrent.timeunit; /** * 微信公众号:路人甲java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注! */ public class demo2 { public static class t extends thread { @override public void run() { while (true) { //循环处理业务 if (this.isinterrupted()) { break; } } } } public static void main(string[] args) throws interruptedexception { t t = new t(); t.start(); timeunit.seconds.sleep(3); t.interrupt(); } }
运行上面的程序,程序可以正常结束。线程内部有个中断标志,当调用线程的interrupt()实例方法之后,线程的中断标志会被置为true,可以通过线程的实例方法isinterrupted()获取线程的中断标志。
示例代码:
package com.itsoku.chat05; import java.util.concurrent.timeunit; /** * 微信公众号:路人甲java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注! */ public class demo3 { public static class t extends thread { @override public void run() { while (true) { //循环处理业务 //下面模拟阻塞代码 try { timeunit.seconds.sleep(1000); } catch (interruptedexception e) { e.printstacktrace(); } } } } public static void main(string[] args) throws interruptedexception { t t = new t(); t.start(); } }
运行上面代码,发现程序无法结束。
在此先补充几点知识:
那么上面代码可以调用线程的interrupt()方法来引发interruptedexception异常,来中断sleep方法导致的阻塞,调整一下代码,如下:
package com.itsoku.chat05; import java.util.concurrent.timeunit; /** * 微信公众号:路人甲java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注! */ public class demo3 { public static class t extends thread { @override public void run() { while (true) { //循环处理业务 //下面模拟阻塞代码 try { timeunit.seconds.sleep(1000); } catch (interruptedexception e) { e.printstacktrace(); this.interrupt(); } if (this.isinterrupted()) { break; } } } } public static void main(string[] args) throws interruptedexception { t t = new t(); t.start(); timeunit.seconds.sleep(3); t.interrupt(); } }
运行结果:
java.lang.interruptedexception: sleep interrupted at java.lang.thread.sleep(native method) at java.lang.thread.sleep(thread.java:340) at java.util.concurrent.timeunit.sleep(timeunit.java:386) at com.itsoku.chat05.demo3$t.run(demo3.java:17)
程序可以正常结束了,分析一下上面代码,注意几点:
thread.interrupt()
方式中断该线程,注意此时将会抛出一个interruptedexception的异常,同时中断状态将会被复位(由中断状态改为非中断状态)java高并发系列连载中,总计估计会有四五十篇文章,可以关注公众号:javacode2018,获取最新文章。
如对本文有疑问, 点击进行留言回复!!
首席架构师推荐:金融保险领域数字化转型实践--如何优雅地修改业务中台中分层应用Maven多模块的版本号?(命令导入式)
[JVM学习之路]一、初识JVM,了解其结构、模型及生命周期
【JAVA并发编程】LinkedBlockingQueue原理
网友评论