一. 停止线程的方法
在Java中有3种方法可以停止正在运行的线程
(1)使用stop( )方法强行停止线程,已被弃用
(2)在run( )方法的循环结构中使用退出标志,当run( )方法完成后线程停止
(3)使用interrupt( )方法中断线程
二. 在run( )方法的循环结构中使用退出标志
1. 使用方法
让run方法结束
因为开启多线程运行,运行代码通常都是循环结构,
所以只要控制住循环结构,就可以让run方法结束,也就是线程结束。
class StopThread implements Runnable{
//标记flag用来控制循环的停止
private boolean flag=true;
public synchronized void run(){
while (flag){
System.out.println(Thread.currentThread().getName()+"...run");
}
}
public void changeFlag(){
flag=false;
}
}
class StopThreadDemo
{
public static void main(String[] args)
{
StopThread st=new StopThread();
Thread t1=new Thread(st);
Thread t2=new Thread(st);
t1.start();
t2.start();
//主线程sleep,避免出现线程t1或t2未判断flag,flag已变为false
try { Thread.sleep(10); } catch (Exception e){}
int num=0;
while(true)
{
if(num++==60)//1.自增(后),所以主线程会打印num=60
{
st.changeFlag();//2. flag标记置为假后,t1和t2都不再打印
break;
}
System.out.println(Thread.currentThread().getName()+"...run"+num);
}
}
}
运行结果是:
2. 特殊情况下存在的问题
当线程处于阻塞状态,就不会读取到标记,那么线程就不会结束
class StopThread implements Runnable{
//标记flag用来控制循环的停止
private boolean flag=true;
public synchronized void run(){
while (flag){
try{
//线程t1或t2进来,wait(),被冻结
wait();
}
catch (InterruptedException e){
System.out.println(Thread.currentThread().getName()+"...Exception");
}
System.out.println(Thread.currentThread().getName()+"...run");
}
}
public void changeFlag(){
flag=false;
}
}
class StopThreadDemo
{
public static void main(String[] args) {
StopThread st=new StopThread();
Thread t1=new Thread(st);
Thread t2=new Thread(st);
t1.start();
t2.start();
//主线程sleep,避免出现线程t1或t2未判断flag,flag已变为false
try { Thread.sleep(10); } catch (Exception e){}
int num=0;
while(true)
{
if(num++==60)//1.自增(后),所以主线程会打印num=60
{
st.changeFlag();//2. flag标记置为假后,t1和t2都不再打印
break;
}
System.out.println(Thread.currentThread().getName()+"...run"+num);
}
}
}
运行结果是:线程不能结束
3. 用interrupt( )解决特殊情况下的问题
用interrupt( )方法清除线程的阻塞状态
class StopThread implements Runnable{
//标记flag用来控制循环的停止
private boolean flag=true;
public synchronized void run(){
while (flag){
try{
//线程t1或t2进来,wait(),被冻结
wait();
}
catch (InterruptedException e){
System.out.println(Thread.currentThread().getName()+"...Exception");
}
System.out.println(Thread.currentThread().getName()+"...run");
}
}
public void changeFlag(){
flag=false;
}
}
class StopThreadDemo
{
public static void main(String[] args) {
StopThread st=new StopThread();
Thread t1=new Thread(st);
Thread t2=new Thread(st);
t1.start();
t2.start();
//主线程sleep,避免出现线程t1或t2未判断flag,flag已变为false
try { Thread.sleep(10); } catch (Exception e){}
int num=0;
while(true)
{
if(num++==60)//1.自增(后),所以主线程会打印num=60
{
//st.changeFlag();//2. flag标记置为假后,t1和t2都不再打印
t1.interrupt();
t2.interrupt();
break;
}
System.out.println(Thread.currentThread().getName()+"...run"+num);
}
}
}
运行结果是:
线程t1和t2被强制唤醒,分别处理异常后,打印了一次;然而flag仍为真,线程t1和t2再次被wait(),线程无法结束
class StopThread implements Runnable{
//标记flag用来控制循环的停止
private boolean flag=true;
public synchronized void run(){
while (flag){
try{
//线程t1或t2进来,wait(),被冻结
wait();
}
catch (InterruptedException e){
System.out.println(Thread.currentThread().getName()+"...Exception");
//当interrupt处理后,线程被强制唤醒,异常被处理,flag就改为false来结束线程
flag=false;
}
System.out.println(Thread.currentThread().getName()+"...run");
}
}
public void changeFlag(){
flag=false;
}
}
class StopThreadDemo
{
public static void main(String[] args) {
StopThread st=new StopThread();
Thread t1=new Thread(st);
Thread t2=new Thread(st);
t1.start();
t2.start();
//主线程sleep,避免出现线程t1或t2未判断flag,flag已变为false
try { Thread.sleep(10); } catch (Exception e){}
int num=0;
while(true)
{
if(num++==60)//1.自增(后),所以主线程会打印num=60
{
//st.changeFlag();//2. flag标记置为假后,t1和t2都不再打印
t1.interrupt();
t2.interrupt();
break;
}
System.out.println(Thread.currentThread().getName()+"...run"+num);
}
}
}
运行结果是:
线程t1和t2被强制唤醒,分别处理异常后,打印了一次,flag被改为假,线程结束。
4. 总结
- 特殊情况: 当线程处于阻塞状态,就不会读取到标记,那么线程就不会结束。
- 当没有指定的方式让阻塞的线程恢复到运行状态时,这时就需要对阻塞状态进行清除,强制让线程恢复到运行状态中来,这样就可以操作标记让线程结束。
- Thread类中提供了该方法interrupt()
三. 使用interrupt( )方法中断线程
1. 所涉及的方法
定义在Thread类中
- interrupt( ):作用是中断此线程(指调用该方法的线程),但实际上只是给线程设置一个中断状态,线程仍会继续运行。
- interrupted( ):作用是测试当前线程是否被中断(检查中断标志),返回一个boolean并清除中断状态。
- isInterrupted( ):作用是测试此线程(指调用该方法的线程)是否被中断(检查中断标志),返回一个boolean但不清除中断状态。
class MyThread implements Runnable{
public void run(){
for(int i=0;i<10;i++){
System.out.println("i="+(i+1));
}
}
}
class MyThreadDemo{
public static void main(String[] args){
MyThread mt=new MyThread();
Thread t1=new Thread(mt);
t1.start();
//给线程t1设置中断状态
t1.interrupt();
//判断线程t1是否被中断,但不清除中断标志:返回true
System.out.println("第一次调用t1.isInterrupted():"+t1.isInterrupted());
//判断线程t1是否被中断,但不清除中断标志:返回true
System.out.println("第二次调用t1.isInterrupted():"+t1.isInterrupted());
//判断当前线程(主线程)是否被中断:返回true
System.out.println("第一次调用Thread.interrupted():"+ Thread.interrupted());
//判断当前线程(主线程)是否被中断:返回true
System.out.println("第二次调用Thread.interrupted():"+Thread.interrupted());
//判断线程t1是否存活:线程t1被设置了中断标志,但并没有被中断,返回true
System.out.println("t1是否存活:"+t1.isAlive());
//给当前线程(主线程)设置中断标志
Thread.currentThread().interrupt();
//判断当前线程(主线程)是否被中断,但不清除中断标志:返回true
System.out.println("第一次调用Thread.currentThread().isInterrupted():"+Thread.currentThread().isInterrupted());
//判断当前线程(主线程)是否被中断,并清除中断标志:返回true
System.out.println("第三次调用Thread.interrupted():"+Thread.interrupted());
//判断当前线程(主线程)是否被中断,因为中断标志已被清除:返回false
System.out.println("第四次调用Thread.interrupted():"+Thread.interrupted());
}
}
运行结果是:
2. 调用interrupt( )+对run( )做处理真正终止线程
class MyThread implements Runnable{
public void run(){
for(int i=0;i<1000;i++){
System.out.println("i="+(i+1));
if(Thread.currentThread().isInterrupted()){
System.out.println("通过isInterrupted检测到中断");
System.out.println("第一个interrupted():"+Thread.interrupted());
System.out.println("第二个interrupted():"+Thread.interrupted());
//检测到中断,退出循环,结束run()
break;
}
}
System.out.println("因为检测到中断,所以跳出循环,线程到这里结束");
}
}
class MyThreadDemo{
public static void main(String[] args) throws InterruptedException {
MyThread mt=new MyThread();
Thread t1=new Thread(mt);
t1.start();
t1.interrupt();
//主线程等待1s,等线程t1运行完
Thread.sleep(1000);
//判断线程t1是否存活
System.out.println("线程t1是否存活:"+t1.isAlive());
}
}
运行结果是: