Java多线程编程核心技术(第3版)
上QQ阅读APP看书,第一时间看更新

1.11.2 判断线程是不是停止状态

在介绍如何停止线程之前,先来看一下如何判断线程的状态已经是停止的。在Java的SDK中,Thread.java类里提供了两种判断方法。

1)public static boolean interrupted():测试currentThread()是否已经中断。

2)public boolean this.isInterrupted():测试this关键字所在线程类的对象是否已经中断。

其中,interrupted()方法的声明如图1-39所示。

isInterrupted ()方法的声明如图1-40所示。

图1-39 interrupted方法的声明

图1-40  isInterrupted方法的声明

这两种方法有什么区别呢?先来看看this.interrupted()方法的解释:测试当前线程是否已经中断。当前线程是指执行this.interrupted()方法的线程,为了对此方法有更深入的了解,创建项目t12,类MyThread.java代码如下:


public class MyThread extends Thread {
@Override
public void run() {
    super.run();
    for (int i = 0; i < 500000; i++) {
        System.out.println("i=" + (i + 1));
    }
}
}

类Run.java代码如下:


public class Run {
public static void main(String[] args) {
    try {
        MyThread thread = new MyThread();
        thread.start();
        Thread.sleep(1000);
        thread.interrupt();
        // Thread.currentThread().interrupt();
        System.out.println("是否停止1?="+thread.interrupted());
        System.out.println("是否停止2?="+thread.interrupted());
    } catch (InterruptedException e) {
        System.out.println("main catch");
        e.printStackTrace();
    }
    System.out.println("end!");
}
}

图1-41 运行结果

程序运行结果如图1-41所示。

类Run.java中虽然是在thread对象上调用代码:


thread.interrupt();

来停止thread对象所代表的线程,在后面又使用代码:


System.out.println("是否停止1?="+thread.interrupted());
System.out.println("是否停止2?="+thread.interrupted());

来判断thread对象所代表的线程是否停止,但从控制台输出的结果来看,线程并未停止,证明了interrupted()方法的解释:测试当前线程是否已经中断。这个当前线程是main,从未中断过,所以输出的结果是两个false。

注意

测试代码中使用thread.interrupted()来判断currentThread()是否被中断,也可以使用Thread.interrupted()进行判断,因为在Thread.java类中调用静态static方法时,大多都是针对currentThread()线程进行操作的。

如何使main线程有中断效果呢?创建Run2.java代码如下:


public class Run2 {
public static void main(String[] args) {
    Thread.currentThread().interrupt();
    System.out.println("是否停止1?=" + Thread.interrupted());
    System.out.println("是否停止2?=" + Thread.interrupted());
    System.out.println("end!");
}
}

程序运行结果如图1-42所示。

图1-42 主线程main已是停止状态

从上述结果来看,interrupted()方法的确判断了当前线程是否是停止状态,但为什么第二个Boolean布尔值是false呢? interrupted方法在官方帮助文档中的解释如下:

测试当前线程是否已经中断。线程的中断状态由该方法清除。换句话说,如果连续两次调用该方法,则第二次调用将返回false(在第一次调用已清除其中断状态之后,且第二次调用检验完中断状态前,当前线程再次被中断的情况除外)。

文档已经解释得很详细,interrupted()方法具有清除状态的功能,所以第二次调用interrupted()方法返回的值是false。

介绍完interrupted()方法后再来看一下isInterrupted()方法,声明如下:


public boolean isInterrupted()

从声明中可以看出isInterrupted()方法不是静态的,具体取决于调用这个方法的线程对象。

继续创建Run3.java类,代码如下:


public class Run3 {
public static void main(String[] args) {
    try {
        MyThread thread = new MyThread();
        thread.start();
        Thread.sleep(1000);
        thread.interrupt();
        System.out.println("是否停止1?="+thread.isInterrupted());
        System.out.println("是否停止2?="+thread.isInterrupted());
    } catch (InterruptedException e) {
        System.out.println("main catch");
        e.printStackTrace();
    }
    System.out.println("end!");
}
}

程序运行结果如图1-43所示。

图1-43 已经是停止状态

从结果可以看到,isInterrupted()方法并未清除状态标志,不具有此功能,所以输出两个true。但是也有非常非常小的概率先输出一个false,如中断的标记还未更新。

最后,再来看一下这两种方法的解释。

1)this.interrupted():测试当前线程是否已经是中断状态,执行后具有清除状态标志值为false的功能。

2)this.isInterrupted():测试所在Thread线程对象是否已经是中断状态,不清除状态标志。