Java线程核心01-中断线程的理论原理

  • 启停原理:使用interrupt来通知,而不是强制
    • 最好停止线程的方式是使用中断interrupt,但这仅仅是通知到被终止的线程"你该停止运行了",被终止的线程自身拥有决定权(决定中否,以及何时停止),这依赖于请求停止方和被停止方都遵守一种约定好的编码规范
    • 任务和线程的启动很容易。大多数情况下,我们都会让它们运行直到结束,或者让它们自行停止。然而,有时候我们希望提前结束任务或线程,或许是因为用户取消了操作,或者服务需要被快速关闭,或者运行超时,或者出错了
    • 要使任务和线程能安全、快速、可靠地停止下来,并不是一件容易的事
    • Java没有提供任何机制来安全地终止线程。但它提供了中断,这是一种协作机制,能够使一个线程终止另一个线程的当前工作
    • 这种协作方式是必要的,几乎很少希望某个任务、线程或服务立即停止,因为这种立即停止会使共享的数据结构处于不一致的状态
    • 所以当需要停止时,会首先清除当前正在执行的工作,然后再结束。这提供了更好的灵活性,因为任务本身的代码比发出取消请求的代码更清楚如何正确执行清除工作
    • 使命周期结束(End-of-Lifecycle)的问题会使任务、服务以及程序的设计和实现等过程变得复杂,而这个在程序设计中非常重要的要素却经常被忽略
    • 一个行为良好的软件与勉强运行的软件之间最主要区别是,行为良好的软件能很完善地处理失败、关闭和取消等过程
  • 停止的情况
    • 线程代码执行完毕
    • 线程内代码抛出异常
    • 线程被阻塞
      • 当线程内部使用代码TimeUnit.SECONDS.sleep(1);进入休眠后,这时如果外部有使用thread.interrupt();来中断时,休眠代码会抛出异常
        • InterruptedException: sleep interrupted
        • 需要注意的是,一旦睡眠被中断会抛出异常,同时会消除线程内的interrupt的标记位,也就是说无法再使用!Thread.cuurentThread.isInterrupted()来判断线程是否被中断
          • 解决这种中断标记位被清除的问题,可以使用传递中断、恢复中断和不屏蔽异常
            • 传递中断:
              • 在方法签名中抛出异常,这样会在run方法中被强制要求对异常进行处理
              • 不要在方法内部try...catch异常,这样会导致异常不能向上传递给run方法,即便方法内部出现异常,也不会被调用者来接收并终止,因为处理异常就会重置中断标记位
              • run方法不允许在签名中抛出异常,必须要在其内部处理
            • 恢复中断:try...catchcatch部分加入代码:Thread.currentThread().interrupted()来恢复被try...catch语句中清除的中断标志,这样就可以在run方法中来判断是否中断来决定程序运行逻辑
    • 响应中断的方法列表
      • Object wait() wait(long) wait(long, int)
      • Thread sleep(long) sleep(long, int)
      • Thread join() join(long) join(long, int)
      • java.util.concurrent.BlockingQueue take() put(E)
      • java.util.concurrent.locks.Lock lockInterruptibly()
      • java.util.concurrent.CountDownLatch await()
      • java.util.concurrent.CyclicBarrier await()
      • java.util.concurrent.Exchanger exchange(V)
      • java.nio.channels.InterruptibleChannel 相关方法
      • java.nio.channels.Selector 相关方法
  • 错误停止线程的方法
    • 使用了已经被弃用的方法:stop() suspend() resume()
      • stop()会释放所有的监视器,但会以一种破坏原子性操作的方式停止线程运行
      • suspend()会挂起线程,但不会释放锁,如果不会被唤醒,或者唤醒它的线程需要它的锁,就容易导致死锁
    • volatile设置boolean标记位
      • 陷入阻塞状态时volatile无法支持终止线程
相关推荐
.生产的驴8 分钟前
SpringBoot 对接第三方登录 手机号登录 手机号验证 微信小程序登录 结合Redis SaToken
java·spring boot·redis·后端·缓存·微信小程序·maven
我曾经是个程序员10 分钟前
鸿蒙学习记录
开发语言·前端·javascript
爱上语文10 分钟前
宠物管理系统:Dao层
java·开发语言·宠物
王ASC42 分钟前
SpringMVC的URL组成,以及URI中对/斜杠的处理,解决IllegalStateException: Ambiguous mapping
java·mvc·springboot·web
是小崔啊44 分钟前
开源轮子 - Apache Common
java·开源·apache
因我你好久不见1 小时前
springboot java ffmpeg 视频压缩、提取视频帧图片、获取视频分辨率
java·spring boot·ffmpeg
程序员shen1616111 小时前
抖音短视频saas矩阵源码系统开发所需掌握的技术
java·前端·数据库·python·算法
小老鼠不吃猫1 小时前
力学笃行(二)Qt 示例程序运行
开发语言·qt
长潇若雪1 小时前
《类和对象:基础原理全解析(上篇)》
开发语言·c++·经验分享·类和对象
Ling_suu1 小时前
SpringBoot3——Web开发
java·服务器·前端