<br />
当前组件销毁时,后台线程同样需要中止,否则仍会在内存中运行,这就造成内存泄漏。
线程对象属于一次性消耗品,一般线程执行完run方法之后,线程就正常结束了,线程结束之后就报废了,不能再次start,只能新建一个线程对象。但有时run方法是永远不会结束的。例如在程序中使用线程进行Socket监听请求,或是其他的需要循环处理的任务。在这种情况下,一般是将这些任务放在一个循环中,如while循环。当需要结束线程时,如何退出线程呢?
有三种方法可以结束线程:
- 使用退出标志,使线程正常退出,也就是当run方法完成后线程终止
- 使用interrupt()方法中断线程
- 使用stop方法强行终止线程(不推荐使用,可能发生不可预料的结果)
前两种方法都可以实现线程的正常退出,而第三种方法相当于电脑断电关机一样,是不安全的方法。
<br />
使用退出标志终止线程
<br />
使用一个变量来控制循环,例如最直接的方法就是设一个boolean类型的标志,并通过设置这个标志为true或false来控制while循环是否退出。
public class ThreadSafe extends Thread {
public volatile boolean exit = false;
public void run() {
while (!exit){
//do something
}
}
}
<br />
使用interrupt()方法终止线程
<br />
使用interrupt()方法来终端线程可分为两种情况
线程处于阻塞状态
如使用了sleep,同步锁的wait,socket的receiver,accept等方法时,会使线程处于阻塞状态。当调用线程的interrupt()方法时,系统会抛出一个InterruptedException异常,代码中通过捕获异常,然后break跳出循环状态,使线程正常结束。
通常很多人认为只要调用interrupt方法线程就会结束,实际上是错的,一定要先捕获InterruptedException异常之后通过break来跳出循环,才能正常结束run方法。
public class ThreadSafe extends Thread {
public void run() {
while (true) {
try {
Thread.sleep(5*1000); //阻塞5妙
} catch(InterruptedException e) {
e.printStackTrace();
break; //捕获到异常之后,执行break跳出循环。
}
}
}
}
线程未进入阻塞状态
使用isInterrupted()判断线程的中断标志来退出循环,当使用interrupt()方法时,中断标志就会置true,和使用自定义的标志来控制循环是一样的道理。
public class ThreadSafe extends Thread {
public void run() {
while (!isInterrupted()){
//do something
}
}
}
<br />
为什么要区分阻塞状态和和非阻塞状态两种情况,是因为当阻塞状态时,如果有interrupt()发生,系统除了会抛出InterruptedException异常外,还会调用interrupted()函数,调用前中断状态是true,调用之后会复位中断状态为false,所以异常抛出之后通过isInterrupted()是获取不到中断状态是true的状态,从而不能退出循环,因此在线程未进入阻塞的代码段时是可以通过isInterrupted()来判断中断是否发生来控制循环,在进入阻塞状态后要通过捕获异常来退出循环。
因此使用interrupt()来退出线程的最好的方式应该是两种情况都要考虑.
public class ThreadSafe extends Thread {
public void run() {
while (!isInterrupted()) { //非阻塞过程中通过判断中断标志来退出
try {
Thread.sleep(5*1000);//阻塞过程捕获中断异常来退出
} catch(InterruptedException e) {
e.printStackTrace();
break; //捕获到异常之后,执行break跳出循环。
}
}
}
}
<br />
问题:用interupt方法,能不能中止一个网络请求?为什么?
<br />