JavaSE笔记7.8-多线程-停止线程

一. 停止线程的方法

在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());
    }
}

运行结果是:
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值