Java线程中的interrupt()方法

如何中断线程?

已经抛弃的方法

  • 通过调用stop()方法停止线程,该方法有点暴力停止的意思,比如线程A去停止线程B,停止的时候根本不知道线程B的执行情况,还有就是执行stop()之后,线程B会马上释放锁,可能会引发数据不同步的问题
  • 类似的被抛弃的还有suspend()和resume()方法

目前用的方法

  • 调用interrupt(),通知线程应该中断了
    • 如果线程处于被阻塞状态,那么线程将立即退出被阻塞状态,并抛出一个InterruptedException异常
    • 如果线程处于正常活动状态,那么会将该线程的中断标志设置为true。被设置中断标志的线程将继续正常运行,不受影响
  • 需要被调用的线程配合中断
    • 在正常运行任务时,可以自行检查本线程的中断标志位(Thread.currentThread().interrupt();),如果被设置了中断标志可以由我们自行停止线程。

测试

 public static void main(String[] args) throws InterruptedException {
            Runnable interruptTask = new Runnable() {
                @Override
                public void run() {
                    int i = 0;
                    try {
                        //在正常运行任务时,经常检查本线程的中断标志位,如果被设置了中断标志就自行停止线程
                        while (!Thread.currentThread().isInterrupted()) {
                            Thread.sleep(100); // 休眠100ms
                            i++;
                            System.out.println(Thread.currentThread().getName() + " (" + Thread.currentThread().getState() + ") 循环次数: " + i);
                        }
                    } catch (InterruptedException e) {
                        //在调用阻塞方法时正确处理InterruptedException异常。(例如,catch异常后就结束线程。)
                        System.out.println(Thread.currentThread().getName() + " (" + Thread.currentThread().getState() + ") 捕获中断异常.");
                    }
                }
            };
            Thread t1 = new Thread(interruptTask, "t1");
            System.out.println(t1.getName() +" ("+t1.getState()+") 新建......");

            t1.start();                      // 启动“线程t1”
            System.out.println(t1.getName() +" ("+t1.getState()+") 已启动......");

            // 主线程休眠300ms,然后主线程给t1发“中断”指令。
            Thread.sleep(300);
            t1.interrupt();
            System.out.println(t1.getName() +" ("+t1.getState()+") 被中断中......");

            // 主线程休眠300ms,然后查看t1的状态。
            Thread.sleep(300);
            System.out.println(t1.getName() +" ("+t1.getState()+") 线程终止......");
        }

// 测试结果

t1 (NEW) 新建......
t1 (RUNNABLE) 已启动......
t1 (RUNNABLE) 循环次数: 1
t1 (RUNNABLE) 循环次数: 2
t1 (TIMED_WAITING) 被中断中......
t1 (RUNNABLE) 捕获中断异常.
t1 (TERMINATED) 线程终止......

  • 这个测试便是一个简单的中断操作,将线程的中断标识改为true,我们在线程中设置了循环检测器,一旦中断标识被修改,则由我们自行进行停止。
  • 这里解释一下为什么会抛出异常,因为主线程中断300ms后去中断线程,线程中的while极大可能在执行第四次循环,位于sleep阻塞,故要抛出异常,然后结束线程。还有一点可以说明,结果倒数第三行,线程状态为(TIMED_WAITING) 是限期等待,说明此时线程在执行sleep方法。

线程的六个状态

上边的测试里面我们用到了getState()来获取线程的状态,下边介绍一下线程的状态:

  • 新建(New):创建后未启动的线程的状态

  • 运行(Runnable):包含Running和Ready,在操作系统中,它有可能正在执行也有可能正在等待cpu为它分配执行时间

  • 无限期等待(Waiting):不会被分配CPU执行时间,需要显示地被唤醒

    • 没有设置Timeout参数的Object.wait()方法。
    • 没有设置Timeout参数的Thread.join()方法。
    • LockSupport.park()方法
  • 限期等待(Timed Waiting):在一定时间后会由系统自动唤醒

    • Thread.sleep()
    • 设置Timeout参数的Object.wait()方法。
    • 设置Timeout参数的Thread.join()方法。
  • 阻塞(Blocked):等待获取排他锁

  • 结束(Terminated):已终止线程的状态,线程已结束执行。终止的线程无法再次复生。

附上一张线程状态图

在这里插入图片描述

以上纯为个人理解,如有不对,请各位看官及时指出(轻喷)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章