Thread线程终止interrupt

作者: 激情的狼王 | 来源:发表于2017-12-19 16:17 被阅读0次

interrupt()的字面意思是中断一个线程,那么它是怎么使用来达到中断当前线程的呢?我们来看几个例子。

一、终止处于“阻塞状态”的线程

通过中断方式终止处于阻塞状态的线程,当线程由于被调用了sleep(),wait(),join()等方法而进入阻塞状态,此时调用线程的interrupt()将线程的中断标记为true。由于处于阻塞状态,中断标记会被清除,同时产生一个InterruptedException异常,InterruptedException放在适当的位置就能终止线程

@Override
public void run() 
{
    try {
    while (true) {
        // 执行任务...
    }
    } catch (InterruptedException ie) {
        // 由于产生InterruptedException异常,退出while(true)循环,线程终止!
    }
}

说明

while(true)中不断的执行任务,当线程处于阻塞状态时,调用线程的interrupt()方法会产生InterruptedException中断,中断的捕获在while(true)之外,这样就退出了while(true)循环。

InterruptException的捕获一定要放在while(true)循环体的外面,这样产生异常时就退出了while(true)循环,否则,InterruptExceptionwhile(true)循环体之外,就需要额外的添加退出处理,形式如下:

@Override
public void run() {
    while (true) {
        try {
            // 执行任务...
        } catch (InterruptedException ie) {
        // InterruptedException在while(true)循环体内。
        // 当线程产生了InterruptedException异常时,while(true)仍能继续运行!需要手动退出
        break;
        }
     }
}

上面的InterruptedException异常的捕获是在while(true)中,当产生异常被catch时,仍然在while(true)循环体内,要退出while(true)循环体,需要额外的执行操作。

二、终止处于运行状态的线程

通过标记方式终止处于运行状态的线程,其中,包括“中断标记”和“额外添加标记”

(1)通过“中断标记”终止线程,形式如下:

@Override
public void run() {
    while (!isInterrupted()) {
    // 执行任务...
    }
}

说明:

isInterrupted()是判断线程的中断标记是不是为true,当前线程处于运行状态,并且我们需要终止它时,可以调用线程的interrupt()方法,使用线程的中断标记为true,即isInterrupted()会返回true,此时,就会退出while循环。interrupt()并不会终止处于“运行状态”的线程,它会将线程的中断标记设为true

(2)通过“额外添加标记”,形式如下:

rivate volatile boolean flag= true;
    protected void stopTask() {
        flag = false;
    }

@Override
public void run() {
    while (flag) {
    // 执行任务...
    }
}

说明:

线程中有一个flag标记,它的默认值是true,并且我们提供stopTask()来设置flag标记,当我们需要终止该线程时,调用该线程的stopTask()方法就可以让线程退出while循环。其中将flag定义为volatile类型,保证flag的可见性,其他线程通过stopTask()修改了flag之后,本线程能看到修改后的flag的值。

综合终止处于“阻塞状态”和“运行状态”的终止方式。

@Override
public void run() {
    try {
        // 1. isInterrupted()保证,只要中断标记为true就终止线程。
        while (!isInterrupted()) {
        // 执行任务...
        }
     } catch (InterruptedException ie) {
        // 2. InterruptedException异常保证,当InterruptedException异常产生时,线程被终止。
     }
}

三、终止线程的示例

interrupt()常常被用来终止“阻塞状态”线程,参考示例:

class MyThread extends Thread {

    public MyThread(String name) {
        super(name);
    }

    @Override
    public void run() {
      try {
            int i=0;
            while (!isInterrupted())
            {
                System.out.println("thread is running");
                Thread.sleep(100); // 休眠100ms
                i++;
                System.out.println(Thread.currentThread().getName()+" ("+this.getState()+") loop " + i);
            }
        } catch (InterruptedException e) {
            System.out.println(Thread.currentThread().getName() +" ("+this.getState()+") catch InterruptedException.");
        }
    }
}

public class Hello {

    public static void main(String[] args) {
        try {
            Thread t1 = new MyThread("t1");  // 新建“线程t1”
            System.out.println(t1.getName() +" ("+t1.getState()+") is new.");

            t1.start();                      // 启动“线程t1”
            System.out.println(t1.getName() +" ("+t1.getState()+") is started.");
            // 主线程休眠300ms,然后主线程给t1发“中断”指令。
            System.out.println("MainThread sleep");
            Thread.sleep(300);
            System.out.println("Thread interrupt");
            t1.interrupt();
            System.out.println(t1.getName() +" ("+t1.getState()+") is interrupted.");

            // 主线程休眠300ms,然后查看t1的状态。
            System.out.println("MainThread sleep");
            Thread.sleep(300);
            System.out.println(t1.getName() +" ("+t1.getState()+") is interrupted now.");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

运行结果:

t1 (NEW) is new.
t1 (RUNNABLE) is started.
MainThread sleep
thread is running
t1 (RUNNABLE) loop 1
thread is running
t1 (RUNNABLE) loop 2
thread is running
t1 (RUNNABLE) loop 3
thread is running
Thread interrupt
t1 (TIMED_WAITING) is interrupted.
MainThread sleep
t1 (RUNNABLE) catch InterruptedException.
t1 (TERMINATED) is interrupted now.

结果说明:

(1)主线程main中会通过new MyThread("t1")创建线程t1,之后通过t1.start()启动线程t1
(2)t1启动之后,会不断的检查他的中断标记,如果中断标记为false,则休眠100ms
(3)t1休眠之后会切换到主线程main,主线程再次运行时,会执行t1.interrupt()中断线程t1。t1收到中断指令之后,会将t1的中断标志设置为false,而且会抛出InterruptedException异常,在t1的run()方法中,是在循环体之外捕获的异常,因此循环被终止。

通过“额外添加标记”的方式终止“运行状态”的线程的示例:

class MyThread extends Thread {

    private volatile boolean flag= true;
    public void stopTask() {
        flag = false;
    }

    public MyThread(String name) {
        super(name);
    }

    @Override
    public void run() {
        synchronized(this) {
            try {
                int i=0;
                while (flag) {
                    Thread.sleep(100); // 休眠100ms
                    i++;
                    System.out.println(Thread.currentThread().getName()+" ("+this.getState()+") loop " + i);
                }
            } catch (InterruptedException ie) {
                System.out.println(Thread.currentThread().getName() +" ("+this.getState()+") catch InterruptedException.");
            }
        }
    }
}
public class Hello {

    public static void main(String[] args) {
        try {
            MyThread t1 = new MyThread("t1");  // 新建“线程t1”
            System.out.println(t1.getName() +" ("+t1.getState()+") is new.");

            t1.start();                      // 启动“线程t1”
            System.out.println(t1.getName() +" ("+t1.getState()+") is started.");

            // 主线程休眠300ms,然后主线程给t1发“中断”指令。
            Thread.sleep(300);
            System.out.println("stopTask");
            t1.stopTask();
            System.out.println(t1.getName() +" ("+t1.getState()+") is interrupted.");

            // 主线程休眠300ms,然后查看t1的状态。
            Thread.sleep(300);
            System.out.println(t1.getName() +" ("+t1.getState()+") is interrupted now.");
            } catch (InterruptedException e) {
            e.printStackTrace();
            }
    }
}

interrupted()isInterrupted()都能够用于检测对象的“中断标记”,区别是interrupted()除了返回中断标记外,它还会清除中断标记(即将中断标记设为false),而isInterrupted()仅仅返回中断标记,关于这两个方法的详细解释请看后续文章。

相关文章

  • Thread线程终止interrupt

    interrupt()的字面意思是中断一个线程,那么它是怎么使用来达到中断当前线程的呢?我们来看几个例子。 一、终...

  • java并发之线程终止

    说到线程终止,在Thread中有stop(),destroy(),interrupt(),destroy方法JDK...

  • 线程中断interrupt()、interrupted()和is

    interrupt() thread.interrupt(),该方法用于中断Thread线程,此线程并非当前线程,...

  • Java线程的中断

    引言 Java没有提供任何机制来安全地终止线程,但提供了中断机制,即thread.interrupt()方法。线程...

  • 关闭线程

    终止线程 stop方法,已经被弃用,无法预期 interrupt interrupt方法其作用是中断此线程(此线程...

  • 线程 interrupt() 中断

    一、正常运行状态的线程 interrupt() 中断 Thread 类 interrupt() 方法,一个线程正常...

  • 线程终止

    线程终止 通过 stop 终止 已被 jdk 弃用,它可能导致线程安全问题。 通过 interrupt 终止 推荐...

  • 多线程基础

    多线程基础总结 一、线程中断 interrupt方法用来请求终止线程。 1. interrupt置位中断标志位 当...

  • Thread

    Thread-中断 interrupt() interrupt()用于中断线程,调用该方法的线程的状态将被置为"中...

  • 深入分析Java线程中断机制

    THread.interrupt真的能中断线程吗?

网友评论

    本文标题:Thread线程终止interrupt

    本文链接:https://www.haomeiwen.com/subject/jifiwxtx.html