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

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

那么,我们究竟应该如何停止线程呢?

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

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秒后更改退出标志的值,没有这段代码,线程就一直不能停止
    }
}

2)使用 interrupt 方法

Thread.interrupt()方法: 作用是中断线程。将会设置该线程的中断状态位,即设置为true,中断的结果线程是死亡、还是等待新的任务或是继续运行至下一步,就取决于这个程序本身。
 interrupt()方法只是改变中断状态,不会中断一个正在运行的线程。 需要用户自己去监视线程的状态为并做处理。支持线程中断的方法(也就是线程中断后会抛出interruptedException的方法)就是在监视线程的中断状态,一旦线程的中断状态被置为“中断状态”,就会抛出中断异常。这一方法实际完成的是,给受阻塞的线程发出一个中断信号,这样受阻线程检查到中断标识,就得以退出阻塞的状态。

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

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

this.interrupted():测试当前线程是否已经中断(静态方法)。如果连续调用该方法,则第二次调用将返回false。在api文档中说明interrupted()方法具有清除状态的功能。执行后具有将状态标识清除为false的功能。

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

package com.ty.day11;

public class PrimeGenerator extends Thread{
    @Override
    public void run() {
        boolean flag=true;
        while (flag){
            for(int i=2;i<=Long.MAX_VALUE;i++){
                if (isPrime(i)) {
                    System.out.println(this.getName()+"--->"+i);
                }
                if (this.isInterrupted()) {
                    System.out.println(this.getName()+"线程被终止了!");
                    flag=false;
                    break;
                }
              
            }
        }

    }
    public boolean isPrime(int num){
        boolean flag=true;
        if(num<2){
            return false;
        }else{
            for(int i=2;i<=Math.sqrt(num);i++) {
                if(num%i==0) flag=false;
            }
            return flag;
        }
    }


}

package com.ty.day11.test;

import com.ty.day11.PrimeGenerator;

public class TestAllPrime {
    public static void main(String[] args) {
        PrimeGenerator pg = new PrimeGenerator();
        pg.start();
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        pg.interrupt();//中断线程
        System.out.println("--Main thread ending--");
    }
}

②使用 interrupt() + InterruptedException来中断线程

public class PrimeGenerator extends Thread{
    @Override
    public void run() {
        boolean flag=true;
        while (flag){
            for(int i=2;i<=Long.MAX_VALUE;i++){
                if (isPrime(i)) {
                    System.out.println(this.getName()+"--->"+i);
                }
                try {
                    Thread.sleep(600);
                } catch (InterruptedException e) {
                    System.out.println(this.getName()+"线程被终止了!");
                    flag=false;
		break;                }

            }
        }

    }
    public boolean isPrime(int num){
        boolean flag=true;
        if(num<2){
            return false;
        }else{
            for(int i=2;i<=Math.sqrt(num);i++) {
                if(num%i==0) flag=false;
            }
            return flag;
        }
    }


}

package com.ty.day11.test;

import com.ty.day11.PrimeGenerator;

public class TestAllPrime {
    public static void main(String[] args) {
        PrimeGenerator pg = new PrimeGenerator();
        pg.start();
        try {
            Thread.sleep(10000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        pg.interrupt();//中断线程
        System.out.println("--Main thread ending--");
    }
}