欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 文旅 > 艺术 > Java线程核心01-中断线程的理论原理

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

2025/2/10 22:41:00 来源:https://blog.csdn.net/taoyong001/article/details/144655104  浏览:    关键词: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无法支持终止线程

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com