• 终止线程的2种方法。标记法和中断


    引言

    在原来的jdk版本中:
    提供了暂停,恢复,终止线程的方法,分别是suspend(),resume(),stop();
    但是它们都存在缺陷,比如暂停suspend()方法在调用后,线程不会释放资源(比如:锁),而是占有着资源进入睡眠状态。stop()方法在调用后,通常不能保证线程的资源正常的释放,因为他根本没有给予线程释放资源的机会。
    正因为这些方法带来的不良影响,使得它们被废除。

    替代

    然而聪明的程序员还是想出来了替代上述api的方法,

    原始api 替代方案
    暂停和恢复 等待-通知机制
    终止 标记法和中断法

    终止线程-法一:标记法

    通过置标记为相反的布尔值,来终止线程

    代码:

    package Interrupt;
    
    public class InterruptThread4 {
        public static void main(String args[]) throws InterruptedException {
            ThreadTest t1=new ThreadTest();
            t1.start();
            Thread.sleep(1000);
            System.out.println(System.currentTimeMillis());
            t1.stopMe();
    
        }
        static class ThreadTest extends Thread{
            private boolean stopMe=false;
    
            public void stopMe() {
                this.stopMe = true;
            }
    
            @Override
            public void run() {
                while(true){
                    if(stopMe){
                        System.out.println("interrupt!");
                        break;
                    }
                    try {
                        System.out.println("t1 sleep");
                        Thread.sleep(5000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    Thread.yield();
                }
                System.out.println(System.currentTimeMillis());
            }
        }
    }
    

    结果:

    t1 sleep
    1501152592588
    interrupt!
    1501152596589

    现象:

    通过打印的时间戳对比发现,发出中断信号和线程中断相隔的时间为5s,这正是线程休眠的时间。所以通过标志来中断线程,假设线程中有sleep()和wait()方法,它达不到立即终止的作用,必须等线程恢复正常运行或者唤醒后才能终止。

    终止线程-法二:中断法

    Thread中断是通过内部的标记字段来进行的,线程中断并不是使线程立刻退出,而是发送给线程一个通知,至于目标线程接到通知如何处理是由线程自己决定。这是和stop()方法强行退出是不同的

    api

    在演示实例之前,先来了解下Thread类中断的api

    api 含义
    public void interrupt() 中断线程
    public boolean isInterrupt() 判断线程是否中断
    public static boolean interrupted 判断线程是否中断,并且清除当前中断状态

    代码

    package Interrupt;
    
    public class InterruptThread3 {
        public static void main(String args[]) throws InterruptedException {
            Thread t1=new Thread(){
                @Override
                public void run() {
                    while(true){
                        if(Thread.currentThread().isInterrupted()){
                            System.out.println("interrupt!");
                            break;
                        }
                        try {
                            System.out.println("t1 sleep");
                            Thread.sleep(5000);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                            //设置中断状态
                            Thread.currentThread().interrupt();
                        }
                        Thread.yield();
                    }
                    System.out.println(System.currentTimeMillis());
                }
            };
            t1.start();
            Thread.sleep(1000);
            System.out.println(System.currentTimeMillis());
            t1.interrupt();
        }
    }
    

    结果

    1501153532961
    java.lang.InterruptedException: sleep interrupted
    interrupt!
    at java.lang.Thread.sleep(Native Method)
    1501153532963
    at Interrupt.InterruptThread3$1.run(InterruptThread3.java:15)

    现象

    发现interrupt()几乎可以达到立即中断的效果,但是线程必须捕获InterruptedException异常这是前提条件。否则线程会永远的执行下去。
    至于在catch代码块中为什么要通过Thread.currentThread().interrupt();来重置中断状态,因为当jvm因为sleep(),wait()等需要捕获异常的方法被中断时,在抛出InterruptedException之前,它会先清除掉线程的中断标志位,使得isInterrupted()返回false;

    中断的两种方法对比

    如果线程中有wait()和sleep()等方法,用中断法来终止线程效果更好。
    因为如果线程长久的等待下去,而没有任何其他线程对其唤醒或者长期的睡眠下去,标志法根本无法中断,而中断法可以立即中断。

    最后的两点非常重要:

    a.要使用interrupt()方法来中断线程,一定要在方法里捕获InterruptedException。
    b.如果因为wait(),或sleep()等需要捕获InterruptedException的方法被中断时,jvm会在抛出异常之前先清除中断标记位,使得isInterrupted()返回为false。素以我们要在判断之前先重新设置好标记位

    好久没写博客了,闪~

  • 相关阅读:
    课程设计之第二次冲刺----第一天
    第一个spring冲刺总结
    课程设计之第一次冲刺----第九天
    课程设计之第一次冲刺----第八天
    课程设计之第一次冲刺----第七天
    课程设计之第一次冲刺----第六天
    课程设计之第一次冲刺---第5天
    课程设计之第一次冲刺---第四天
    课程设计之第一次冲刺---第3天
    课程设计之第一次冲刺---第二天
  • 原文地址:https://www.cnblogs.com/chenny3/p/10226128.html
Copyright © 2020-2023  润新知