isInterrupted、interrupt和interrupted

本文深入探讨Java中线程的中断机制,包括isInterrupted和interrupt方法的区别及使用场景,通过实例展示如何正确处理线程中断。

由易到难开始介绍吧

isInterrupted

最容易理解的,查看线程是否被打断,没有副作用。可以自己询问自己,也可以在A线程中询问B线程。

       Thread A = new Thread(() -> {
            //自己询问自己
            Thread.currentThread().isInterrupted();
        },"A");
        A.start();
        Thread B = new Thread(() -> {
            //在B中询问A
            A.isInterrupted();
        },"B");
        B.start();

interrupt

最复杂的一个,分为几种情况:

  1. 如果线程此时正在InterruptibleChannel上执行I/O操作,线程的打断状态被设置为true,抛出异常
  2. 如果线程此时正被Selector阻塞,线程的打断状态被设置为true,线程立即返回一个非零值。
  3. 如果线程此时正被sleep、wait、join(包括这些函数带参数的版本)阻塞着呢,线程的打断状态被重置,即设为false,再抛出InterruptedException。这意味着如果在捕获InterruptedException的逻辑中不能通过isInterrupted或者interrupted来正确判断线程是否被打断,因为打断状态已经被重置了。
  4. 其他情况,打断状态被设置为true。
    注意 第一和第二都是与管道I/O相关的,平时都接触不到。当读取File时被打断不会抛出异常。

接下来演示第三和第四种情况

class Demo {

    static boolean flag = true;
    public static void main(String[] args) {
        ReentrantLock lock = new ReentrantLock();
        final Thread a = new Thread(() -> {
            synchronized (lock) {
                System.out.println(Thread.currentThread().getName() + "获得了锁");
                try {
                    flag = false;
                    lock.wait();
                } catch (InterruptedException e) {
                    //符合第三种情况,虽然a被b打断了,但这里返回的是false,即打断状态被重置了。
                    System.out.println("a捕获InterruptedException异常后Thread.currentThread().isInterrupted() " + Thread.currentThread().isInterrupted());
                    e.printStackTrace();
                }
            }
        }, "a");
        a.start();
        while (flag){}//保证a线程先于b线程执行
        new Thread(() -> {
            synchronized (lock) {
                //此时a线程还没有被打断,isInterrupted返回false,没有副作用
                System.out.println("b第一次询问a.isInterrupted() " + a.isInterrupted());
                //在b线程中打断a线程
                a.interrupt();
                //此时a线程被打断,isInterrupted返回true,没有副作用
                System.out.println("b第二次询问a.isInterrupted() " + a.isInterrupted());
                
                //没有阻塞的情况下打断线程,符合第四种情况,只是把打断状态设置为true
                Thread.currentThread().interrupt();
                System.out.println("b自我打断之后Thread.currentThread().interrupt() " + Thread.interrupted());
                lock.notify();
            }
        }, "b").start();
    }
}

结果如下图:
在这里插入图片描述
看一个读取文件的例子:

interrupted

最让人疑惑的一个。因为他又副作用。就是会重置线程的打断状态,即设置为false。可想而知,如果一个线程被interrupt了,再 马上连续多次 通过Thread.interrupted()查看,只有第一次是true,以后就是false。而isInterrupted不管调用几次都不会有副作用。
注意文中的 马上连续多次 ,要求多次Thread.interrupted()之间不能再被interrupt

Thread.currentThread().interrupt();
//连续多次调用Thread.interrupted()
System.out.println(Thread.interrupted());//true,打断状态被重置为false
System.out.println(Thread.interrupted());//false

Thread.currentThread().interrupt();
System.out.println(Thread.interrupted());//true
Thread.currentThread().interrupt();
System.out.println(Thread.interrupted());//true

总结

  1. 打断别人用a.interrupt();
  2. 打断自己用Thread.currentThread().interrupt();
  3. 查看别人的打断状态用a.isInterrupted();
  4. 查看自己的打断状态用Thread.currentThread().isInterrupted();

参考

作者:大闲人柴毛毛
链接:https://www.zhihu.com/question/41048032/answer/252905837
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

<think>嗯,用户想了解Javainterrupted()、interrupt()isInterrupted()这三个线程中断方法的区别。这个问题在并发编程中很关键,用户可能是正在学习多线程或者遇到了相关调试问题。 从提供的引用来看,用户已经收集了5个参考资料,说明ta做了功课但需要系统梳理。引用[5]的总结最清晰:interrupt()是设置中断标志位,interrupted()是静态方法会清除标志位,isInterrupted()只查询不清除。不过引用[3][4]存在表述矛盾,需要重点验证。 首先需要厘清核心机制:Java线程中断本质是协作式的,不是强制终止。关键点在于: - interrupt()仅改变中断状态(设置标记位),不会直接中止线程 - 阻塞方法(如sleep/wait)会立即响应中断,抛出InterruptedException - 非阻塞场景需要开发者手动检查中断状态 特别注意引用[2]揭示的实现细节:interrupted()isInterrupted()底层都调用native方法isInterrupted(boolean),true代表清除标记,false保留。这解释了为什么静态方法interrupted()会清除状态——因为currentThread().isInterrupted(true)的参数是true。 用户可能存在的认知误区: 1 认为interrupt()会立即停止线程(实际需要程序配合检查) 2 混淆静态方法与非静态方法(interrupted() vs isInterrupted()) 3 不清楚中断标志清除的时机 最后设计回答结构时,应该先用生活化比喻(比如"中断就像按暂停键"),再分方法对比,最后强调检查中断的两种方式(捕获异常或轮询状态)。代码示例要包含阻塞非阻塞两种场景的响应处理。 </think>以下是Java中`interrupt()`, `interrupted()``isInterrupted()`方法的详细区别总结,结合核心逻辑引用资料说明: ### 📌 核心区别总结 | **方法** | **类型** | **功能** | **是否清除中断标志** | |--------------------|----------|--------------------------------------------------------------------------|----------------------| | **`interrupt()`** | 实例方法 | 设置线程的中断状态为`true` | ❌ 不会清除 | | **`isInterrupted()`** | 实例方法 | 检查**指定线程**的中断状态(直接返回当前状态值) | ❌ 不会清除 | | **`interrupted()`** | **静态方法** | 检查**当前线程**的中断状态,并**立即清除中断标志**(重置为`false`) | ✅ 会清除 | --- ### 🔧 详细解析 1. **`interrupt()`** - **作用**:向线程发送中断请求,将其中断状态设为`true`。 - **响应机制**: - 若线程在阻塞中(如`sleep()`, `wait()`),会立即抛出`InterruptedException`并**清除中断状态**。 - 若线程在运行中,需**主动检查中断状态**来决定是否终止(通过`isInterrupted()`或`interrupted()`)。 ```java Thread thread = new Thread(() -> { while (!Thread.currentThread().isInterrupted()) { // 检查中断状态 // 执行任务... } }); thread.start(); thread.interrupt(); // 请求中断 ``` 2. **`isInterrupted()`** - **作用**:查询线程的中断状态,**不修改标志位**。 - **典型场景**:在自定义循环中检查线程是否被中断: ```java if (thread.isInterrupted()) { // 处理中断逻辑(如清理资源后退出) } ``` 3. **`interrupted()`** - **作用**:静态方法,检查**当前线程**是否被中断,并**立即清除中断状态**。 - **关键特性**:连续调用时,第一次返回`true`,后续返回`false`(除非再次中断)[^5]。 ```java Thread.currentThread().interrupt(); // 设置中断标志 System.out.println(Thread.interrupted()); // 输出 true(状态被清除) System.out.println(Thread.interrupted()); // 输出 false ``` --- ### ️ 重要注意事项 1. **中断的本质** Java中断是**协作式**的:`interrupt()`仅设置标志位,**不会强制停止线程**。线程需主动响应中断(如退出循环或抛出异常)[^5]。 2. **阻塞方法中的中断** 线程在`sleep()`, `wait()`, `join()`时被中断,会: - 抛出`InterruptedException` - **清除中断状态**(需手动恢复): ```java try { Thread.sleep(1000); } catch (InterruptedException e) { Thread.currentThread().interrupt(); // 重新设置中断标志 } ``` 3. **底层实现差异** - `interrupted()` 调用 `isInterrupted(true)`(清除标志)[^2][^4]。 - `isInterrupted()` 调用 `isInterrupted(false)`(保留标志)[^4]。 --- ### 🌟 总结对比图 ```plaintext ┌─────────────┬──────────────┬──────────────────────┬─────────────────────┐ │ 方法 │ 作用对象 │ 返回值含义 │ 清除标志位 │ ├─────────────┼──────────────┼──────────────────────┼─────────────────────┤ │ interrupt() │ 任意线程实例 │ 设置中断标志为true │ ❌ │ │ isInterrupted()│ 调用方法的实例 │ 返回当前中断状态 │ ❌ │ │ interrupted()│ **当前线程** │ 返回状态并清除标志 │ ✅ │ └─────────────┴──────────────┴──────────────────────┴─────────────────────┘ ``` > 设计中断逻辑时: > ✅ 使用`isInterrupted()`维护状态不变性(如循环退出条件) > ✅ 用`interrupted()`需警惕状态清除后的二次检查失效 > ✅ 捕获`InterruptedException`后需恢复中断状态[^3][^5] --- 📚 **相关问题** 1. 如何处理线程阻塞时抛出的`InterruptedException`? 2. 为什么Java不提供强制终止线程的方法? 3. 如何设计可安全响应中断的长时间运行任务? 4. `interrupted()`在多线程环境下是否存在线程安全问题? [^1]: 关于`isInterrupted()`不清除状态标记的说明 [^2]: `interrupted()``isInterrupted()`的底层实现差异 [^3]: 三方法的行为对比总结 [^4]: `isInterrupted(boolean)`参数含义解析 [^5]: 中断机制的核心特性与协作式设计原则
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值