首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

多线程开发必看:守护线程优雅地停止用户线程

欲善编程,多看、多敲、多讨论;动眼、动手、动大脑。

1 守护线程的意义

在java多线程开发中,有两类线程,分别是User Thread(用户线程)和Daemon Thread(守护线程) 。

用户线程很好理解,我们日常开发中编写的业务逻辑代码,运行起来都是一个个用户线程。而守护线程相对来说则要特别理解一下。

守护线程,类似于操作系统里面是守护进程。由于Java语言机制是构建在JVM的基础之上,这一机制意味着Java平台是把操作系统的进程给屏蔽了。所以需要在JVM里面构造出对自己有利的机制,于是守护线程应运而生。

所谓的守护线程,指的是程序运行时在后台提供的一种通用服务的线程。比如垃圾回收线程就是一个很称职的守护者,并且这种线程并不属于程序中不可或缺的部分。因此,当所有的非守护线程结束时,程序也就终止了,同时会杀死进程中的所有守护线程。反过来说,只要任何非守护线程还在运行,程序就不会终止。

事实上,User Thread(用户线程)和Daemon Thread(守护线程)从本质上来说并没有什么区别,唯一的不同之处就在于虚拟机的离开:如果用户线程已经全部退出运行了,只剩下守护线程存在了,虚拟机也就退出了。因为没有了被守护者,守护线程也就没有工作可做了,也就没有继续运行程序的必要了。

守护线程并非只有虚拟机内部可以提供,用户也可以手动将一个用户线程设定/转换为守护线程。

在Thread类中提供了一个setDaemon(true)方法来将一个普通的线程(用户线程)设置为守护线程。

2 使用守护线程的注意事项

2.1 必须在线程运行前确认是否为守护线程

thread.setDaemon(true)必须在thread.start()之前设置。否则将引发IllegalThreadStateException异常。这意味着正在运行的常规线程不能设置为守护进程线程。这显然不同于操作系统中的守护进程,在创建守护进程之后,让进程摆脱对原始会话的控制+让进程摆脱对原始进程组的控制+让进程摆脱对原始控制终端的控制;因此,虚拟机上的语言机制与系统级语言本质上是不同的。

2.2 守护线程创建的新线程也是守护线程

在守护线程中生成的新线程也是守护线程。这与操作系统中的守护进程本质上是不同的:守护进程之外的子进程不再是守护进程,尽管它复制有关父进程的进程相关信息。

2.3 守护线程不要去操作固有资源

并非所有用户线程都可以分配给守护线程进行服务,例如读写操作或计算逻辑。因为这个应用程序可能在DaemonThread有时间操作之前就退出了虚拟机。这意味着守护进程线程永远不应该访问固有的资源,例如文件和数据库,因为它可以在任何时候被中断,甚至在操作的中间。

3 守护线程优雅地停止用户线程

虽然java的Thread类里面,提供了很多让线程停止和销毁的方法,但早在jdk1.2版本就不推荐使用了。主要的原因是,强制停止线程容易造成死锁。对操作中的数据非常不友好。但守护线程很好地解决了这一问题。通过守护线程,可以优雅地停止用户线程。

先上代码,截图如下:

在main方法中,启动了两个线程,t1为用户线程,如果t1_run_flag标志为true,则进行循环报数,每50毫秒报数1-5。t2为守护线程,100毫秒后,将t1_run_flag标志置为false,也就让t1停止报数。当t1停止报数后,t1线程运行结束,由于没有其他线程t2守护线程运行结束。整个程序运行结束。运行效果如下图:

在100毫秒的时间内,t1用户线程报了两轮数。然后整个线程运行结束。

4 结束语

人人都是程序员,感谢您的阅读,欢迎关注。

  • 发表于:
  • 原文链接https://kuaibao.qq.com/s/20200308A05XKT00?refer=cp_1026
  • 腾讯「腾讯云开发者社区」是腾讯内容开放平台帐号(企鹅号)传播渠道之一,根据《腾讯内容开放平台服务协议》转载发布内容。
  • 如有侵权,请联系 cloudcommunity@tencent.com 删除。

扫码

添加站长 进交流群

领取专属 10元无门槛券

私享最新 技术干货

扫码加入开发者社群
领券