可以终止线程的 stop() 方法已经过时,现在终止线程的更好的方法是:直接让run()方法运行结束。
package thread;
/*
如何停止线程?
stop方法已经过时。只有一种方法,run方法结束。
开启多线程运行,运行代码通常是循环结构。只要控制住循环,就可以让run方法结束,也就是线程结束。
特殊情况:
当线程处于了冻结(wait)状态。就不会读取到标记。那么线程就不会结束。
当没有指定的方式让冻结的线程恢复到运行状态时,这时需要对冻结进行清除。
强制让线程恢复到运行状态中来。这样就可以操作标记让线程结束。
Thread类提供该方法 interrupt();
其官方解释:
如果线程在调用 Object 类的 wait()、wait(long) 或 wait(long, int) 方法,
或者该类的 join()、join(long)、join(long, int)、sleep(long) 或 sleep(long, int) 方法过程中受阻,
则其中断状态将被清除,它还将收到一个 InterruptedException()。
说白了就是如果某个线程处于等待状态,interrupt方法会将其等待状态解除,变成就绪状态,这样该线程才有可能重新运行
守护线程/用户线程(后台线程):
Thread 类的 setDaemon(boolean on) 方法将该线程标记为守护线程或用户线程。
当正在运行的线程都是守护线程时,Java 虚拟机退出。
该方法必须在启动线程前调用。
*/
public class Thread_7_StopThreadDemo {
public static void main(String[] args) {
StopThread st = new StopThread();
Thread t1 = new Thread(st);
Thread t2 = new Thread(st);
//如果将这两个线程都设置成守护线程,一旦主线程(唯一的前台县城)结束,那么这两个守护线程也会被结束
//t1.setDaemon(true);
//t2.setDaemon(true);
t1.start();
t2.start();
int num = 0;
while(true) {
if(num++ == 600){
//st.stopThread();
t1.interrupt();
t2.interrupt(); //如果线程处于等待状态,调用该方法会将其强制唤醒
System.out.println("===============");
break;
}
System.out.println(Thread.currentThread().getName()+"......."+num);
}
System.out.println("over");
}
}
class StopThread implements Runnable{
private boolean flag =true;
public synchronized void run(){
//一般的线程都是死循环,将标志位修改就可以结束run方法,从而结束线程
while(flag) {
try {
wait(); //wait 方法必须在 synchronized 块内调用
} catch (InterruptedException e) {
//如果在该线程处于 wait 状态的时候调用其 interrupt 方法,则将该线程强制从等待状态唤醒并转入就绪状态,同时抛出异常
//那么可以在接受异常处修改标志位,停止线程
System.out.println(Thread.currentThread().getName()+"....Exception");
flag = false;
}
System.out.println(Thread.currentThread().getName()+"....run");
}
}
public void stopThread(){
flag = false;
}
}