java线程中的中断

参考案例:https://www.cnblogs.com/myseries/p/10918819.html

1、线程中断

中断(Interrupt)一个线程意味着在该线程完成任务之前停止其正在进行的一切,有效地中止其当前的操作。线程是死亡、还是等待新的任务或是继续运行至下一步,就取决于这个程序。虽然初次看来它可能显得简单,但是,你必须进行一些预警以实现期望的结果。你最好还是牢记以下的几点告诫。

首先,忘掉Thread.stop方法。虽然它确实停止了一个正在运行的线程,然而,这种方法是不安全也是不受提倡的,这意味着,在未来的JAVA版本中,它将不复存在。

1.1、如何安全的结束一个正在运行的线程

Thread类相关的方法

  java.lang.Thread类包含了一些常用的方法,如:start(), stop(), stop(Throwable) ,suspend(), destroy() ,resume()。通过这些方法,我们可以对线程进行方便的操作,但是这些方法中,只有start()方法得到了保留。

在JDK帮助文档以及Sun公司的一篇文章《Why are Thread.stop, Thread.suspend and Thread.resume Deprecated?》中都讲解了舍弃这些方法的原因。

简单来说是因为:使用stop方法虽然可以强行终止正在运行或挂起的线程,但使用stop方法是很危险的,就象突然关闭计算机电源,而不是按正常程序关机一样,可能会产生不可预料的结果,因此,并不推荐使用stop方法来终止线程。

也就是说将一些没有终止掉的任务或者是正在运行中的程序由于非法关闭,可能会造成资源上的浪费和影响。

1、2 如何恰当的停止线程

  • 1、任务中一般都会有循环结构,只要用一个标记控制住循环,就可以结束任务。
  • 2、如果线程处于了冻结状态,无法读取标记,此时可以使用interrupt()方法将线程从冻结状态强制恢复到运行状态中来,让线程具备CPU的执行资格。

1.2.1、设置退出状态

当run方法执行完后,线程就会退出。但有时run方法是永远不会结束的,如在服务端程序中使用线程进行监听客户端请求,或是其他的需要循环处理的任务。  在这种情况下,一般是将这些任务放在一个循环中,如while循环。如果想使while循环在某一特定条件下退出,最直接的方法就是设一个boolean类型的标志,并通过设置这个标志为true或false来控制while循环是否退出。

public class Test1 {

    public static volatile boolean exit =false;  //退出标志
    
    public static void main(String[] args) {
        new Thread() {
            public void run() {
                System.out.println("线程启动了");
                while (!exit) {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                System.out.println("线程结束了");
            }
        }.start();
        
        try {
            Thread.sleep(1000 * 5);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        exit = true;//5秒后更改退出标志的值,没有这段代码,线程就一直不能停止
    }
}

1.2.2、使用 interrupt 方法

Thread.interrupt()方法: 作用是中断线程。将会设置该线程的中断状态位,即设置为true,中断的结果线程是死亡、还是等待新的任务或是继续运行至下一步,就取决于这个程序本身。线程会不时地检测这个中断标示位,以判断线程是否应该被中断(中断标示值是否为true)。它并不像stop方法那样会中断一个正在运行的线程

interrupt()方法只是改变中断状态,不会中断一个正在运行的线程

interrupt()方法只是改变中断状态,不会中断一个正在运行的线程

interrupt()方法只是改变中断状态,不会中断一个正在运行的线程

(重要的事情说三遍)。

需要用户自己去监视线程的状态为并做处理。支持线程中断的方法(也就是线程中断后会抛出interruptedException的方法)就是在监视线程的中断状态,一旦线程的中断状态被置为“中断状态”,就会抛出中断异常。这一方法实际完成的是,给受阻塞的线程发出一个中断信号,这样受阻线程检查到中断标识,就得以退出阻塞的状态。

更确切的说,如果线程被Object.wait, Thread.join和Thread.sleep三种方法之一阻塞,此时调用该线程的interrupt()方法,那么该线程将抛出一个 InterruptedException中断异常(该线程必须事先预备好处理此异常),从而提早地终结被阻塞状态。如果线程没有被阻塞,这时调用 interrupt()将不起作用,直到执行到wait(),sleep(),join()时,才马上会抛出 InterruptedException。

Demo:

public static void main(String[] args) {
        Thread thread = new Thread() {
            public void run() {
                System.out.println("线程启动了");
                try {
                    Thread.sleep(1000 * 100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("线程结束了");
            }
        };
        thread.start();

        try {
            Thread.sleep(1000 * 5);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        //作用是:在线程阻塞时抛出一个中断信号,这样线程就得以退出阻塞的状态
        thread.interrupt();
    }

在线程处于sleep、join和wait状态中的时候,此时若给线程打上了中断标记之后,那么线程就会抛出一样。

看对应的输出信息:

线程启动了
java.lang.InterruptedException: sleep interrupted
	at java.lang.Thread.sleep(Native Method)
	at com.guang.thread.interupt.Test2$1.run(Test2.java:14)
线程结束了

1.2.3、使用 interrupt() + isInterrupted()来中断线程

this.interrupted():测试当前线程是否已经中断(静态方法)。

如果连续调用该方法,则第二次调用将返回false。在api文档中说明interrupted()方法具有清除状态的功能。执行后具有将状态标识清除为false的功能。

(这段话的意思也就是说,对于一个新的线程来说,刚创建出的时候,线程是没有中断过的,那么对应的是否中断状态应该是false;

如果当前线程中断状态是true,那么调用了这个方法之后,中断状态会被清除掉,成为false,也就是擦除了线程中断状态。)

this.isInterrupted():测试线程是否已经中断,但是不能清除状态标识。

Demo:

public class TestInterrupted {
    public static void main(String[] args) {
        System.out.println("当前线程是否已经被中断过1?    "+Thread.currentThread().isInterrupted());
        // true---->false;false不发生改变
        Thread.interrupted();
        System.out.println("当前线程是否已经被中断过2?    "+Thread.currentThread().isInterrupted());
        // 设置中断标记
        Thread.currentThread().interrupt();
        System.out.println("当前线程是否已经被中断过3?    "+Thread.currentThread().isInterrupted());
        // 擦除标记
        Thread.interrupted();
        System.out.println("当前线程是否已经被中断过4?    "+Thread.currentThread().isInterrupted());
    }
}

查看对应的输出信息:

当前线程是否已经被中断过1?    false
当前线程是否已经被中断过2?    false
当前线程是否已经被中断过3?    true
当前线程是否已经被中断过4?    false

可以看到当线程中断状态被设置成true之后,调用interrupted()方法之后,将其中断标记给擦除掉了,中断状态为false。

再来看一下使用案例:

public static void main(String[] args) {
        Thread thread = new Thread() {
            public void run() {
                System.out.println("线程启动了");
                while (!isInterrupted()) {
                    System.out.println(isInterrupted());//调用 interrupt 之后为true
                }
                System.out.println("线程结束了");
            }
        };
        thread.start();

        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        thread.interrupt();
        System.out.println("线程是否被中断:" + thread.isInterrupted());//true
    }

看控制台输出信息:

....
false
false
线程结束了
线程是否被中断:true

1.2.4、线程不中断擦除状态和线程中断擦除状态

发送线程中断并擦除状态

    private static void demo1() {
        new Thread(() -> {
            Thread thread = Thread.currentThread();
            System.out.println("当前线程状态:" + thread.isInterrupted());
            // 发送中断信号,给线程打上标记状态
            thread.interrupt();
            System.out.println("当前线程状态:" + thread.isInterrupted());
            // 消除线程被标记状态,这里返回的是线程标记上的类容
            boolean interrupted = Thread.interrupted();
            System.out.println("消除后的状态是:" + interrupted);
            // 上一个方法中已经被消除了,所以这里显示不为中断
            System.out.println("当前线程状态:" + thread.isInterrupted());
        }).start();
    }

输出:

当前线程状态:false
当前线程状态:true
消除后的状态是:true
当前线程状态:false

线程不中断而去擦除

    private static void demo2() {
        new Thread(() -> {
            Thread thread = Thread.currentThread();
            System.out.println("当前线程状态:" + thread.isInterrupted()); // false
            // 线程如果没有中断,那么擦除方法有无变化
            boolean interrupted = Thread.interrupted();
            System.out.println("消除后的状态是:" + interrupted); // false
            // 上一个方法中已经被消除了,所以这里显示不为中断
            System.out.println("当前线程状态:" + thread.isInterrupted()); // false
        }).start();
    }

输出:

当前线程状态:false
消除后的状态是:false
当前线程状态:false

1.2.4、线程无法终止

public class Test {
    public static void main(String[] args) {
        Thread thread = new Thread() {
            public void run() {
                System.out.println("线程启动了");
                while (true) {//对于这种情况,即使线程调用了intentrupt()方法并且isInterrupted(),但线程还是会继续运行,根本停不下来!
                    System.out.println(isInterrupted());//调用interrupt之后为true
                }
            }
        };
        thread.start();
        thread.interrupt();//注意,此方法不会中断一个正在运行的线程,它的作用是:在线程受到阻塞时抛出一个中断信号,这样线程就得以退出阻塞的状态
        while (true) {
            System.out.println("是否isInterrupted:" + thread.isInterrupted());//true
        }
    }
}

下面这种情况:即使打上了中断标记状态,但是无法让线程停止运行(如果没有遇到sleep、join和wait方法的时候)

但是上面的demo中并没有设置任何wait、sleep和wait方法,所以线程会一直运行下去,结果就是线程无法终止掉。

1.2.5、isInterrupted补充说明

public class TestInterrupted {
    public static void main(String[] args) {
        Thread thread = new Thread(() -> {
            try {
                Thread.sleep(1000*10);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        thread.start();
        thread.interrupt();
        System.out.println("当前的线程状态是:"+Thread.currentThread().isInterrupted());
        System.out.println("当前的线程状态是:"+Thread.currentThread().isInterrupted());
    }
}

输出控制台信息:

当前的线程状态是:false
当前的线程状态是:false
java.lang.InterruptedException: sleep interrupted
	at java.lang.Thread.sleep(Native Method)
	at com.guang.thread.interupt.TestInterrupted.lambda$main$0(TestInterrupted.java:13)
	at java.lang.Thread.run(Thread.java:748)

之所以输出异常信息,是因为新起的线程中跑出来了对应的异常,而主线程的中断状态没有改变过,所以打印出来了两个false。

2、总结

正确中断线程对应的逻辑应该是:

public class MakeUserOfThread {
    public static void main(String[] args) {
        Thread thread = new Thread(() -> {
            int i = 0;
            while (!Thread.currentThread().isInterrupted()) {
                i++;
                // doSomething.............
                if (i == 5) {
                    // 将其设置为中断标记
                    // 可以使用wait、sleep和join方法调用
                    // 但是我觉得是用break的方式更加优雅
                    Thread.currentThread().interrupt();
                }
                try {
                    Thread.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                    // 从这里可以看到,中断之后,状态又恢复了
                    System.out.println("当前的线程中断标记是:"+Thread.currentThread().isInterrupted());
                   /* Thread.interrupted();
                    System.out.println("当前的线程中断标记是:"+Thread.currentThread().isInterrupted());*/
                    // 让其进行终止
                    break;
                }
            }
        });
        thread.start();
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

这里只是一种实现思路而已,需要搭配着实际的场景来进行实现。

posted @ 2022-01-26 13:43  写的代码很烂  阅读(271)  评论(0编辑  收藏  举报