中断线程的方式

3 篇文章 0 订阅

通过一个变量控制线程中断

示例代码:

public volatile static boolean flag = false;
public static class ThreadDemo extends Thread {
        @Override
        public void run() {
            while (true) {
                //循环处理业务
                if (flag) {
                    break;
                }
            }
        }
    }

    public static void setFlag(){
        flag = true;
    }
    
    public static void main(String[] args) throws InterruptedException {
        ThreadDemo td = new ThreadDemo();
        td.start();
        TimeUnit.SECONDS.sleep(3);
        setFlag();
    }

启动线程,线程的run方法中有个死循环,内部通过flag变量的值来控制是否退出。TimeUnit.SECONDS.sleep(3);让主线程休眠3秒。程序有个重点:volatile关键字,flag变量必须通过这个修饰,如果把这个去掉,程序无法正常退出。volatile控制了变量在多线程中的可见性。

通过线程自带的中断标志控制

示例代码:

 public static class ThreadDemo extends Thread {
        @Override
        public void run() {
            while (true) {
                //循环处理业务
                if (this.isInterrupted()) {
                    break;
                }
            }
        }
    }

    public static void main(String[] args) throws InterruptedException {
        ThreadDemo td = new ThreadDemo();
        td.start();
        TimeUnit.SECONDS.sleep(3);
        td.interrupt();
    }

运行上面的程序,程序可以正常结束。线程内部有个中断标志,当调用线程的interrupt()实例方法之后,线程的中断标志会被置为true,可以通过线程的实例方法isInterrupted()获取线程的中断标志。

线程阻塞状态中如何中断

示例代码:


public static class ThreadDemo extends Thread {
        @Override
        public void run() {
            while (true) {
                //循环处理业务
                //阻塞
                try {
                    TimeUnit.SECONDS.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    public static void main(String[] args) throws InterruptedException {
        ThreadDemo td = new ThreadDemo();
        td.start();
    }

运行上面代码,发现程序无法结束。

1. 调用线程的interrupt()实例方法,线程的中断标志会被置为true
2. 当线程处于阻塞状态时,调用线程的interrupt()实例方法,线程内部会触发InterruptedException异常,并且会清除线程内部的中断标志(即将中断标志置为false)

那么上面代码可以调用线程的interrupt()方法来引发InterruptedException异常,来中断sleep方法导致的阻塞,调整一下代码,如下:

 public static class ThreadDemo extends Thread {
        @Override
        public void run() {
            while (true) {
                //循环处理业务
                //阻塞
                try {
                    TimeUnit.SECONDS.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                    this.interrupt();
                }
                if (this.isInterrupted()) {
                    break;
                }
            }
        }
    }

    public static void main(String[] args) throws InterruptedException {
        ThreadDemo td = new ThreadDemo();
        td.start();
        TimeUnit.SECONDS.sleep(3);
        td.interrupt();
    }

运行结果:
线程中断
程序可以正常结束了,分析一下上面代码,注意几点:

1. main方法中调用了td.interrupt()方法,此时线程td内部的中断标志会置为true
2. 然后会触发run()方法内部的InterruptedException异常,所以运行结果中有异常输出,当触发InterruptedException异常时候,线程内部的中断标志又会被清除(false),所以在catch中又调用了this.interrupt();将中断标志置为true
3. run()方法中通过this.isInterrupted()来获取线程的中断标志,退出循环(break)

总结

  1. 当一个线程处于被阻塞状态或者试图执行一个阻塞操作时,可以使用Thread.interrupt()方式中断该线程,此时将会抛出一个InterruptedException的异常,同时中断状态将会被复位(由中断状态改为非中断状态)
  2. 内部有循环体,可以通过一个变量来作为一个信号控制线程是否中断,注意变量需要volatile修饰
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值