欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

java高并发系列 - 第11天:线程中断的几种方式

程序员文章站 2022-06-14 17:34:39
java高并发系列第11篇文章。 本文主要探讨一下中断线程的几种方式。 通过一个变量控制线程中断 代码: 代码中启动了一个线程,线程的run方法中有个死循环,内部通过exit变量的值来控制是否退出。 让主线程休眠3秒,此处为什么使用TimeUnit?TimeUnit使用更方便一些,能够很清晰的控制休 ......

java高并发系列第11篇文章。

本文主要探讨一下中断线程的几种方式。

通过一个变量控制线程中断

代码:

package com.itsoku.chat05;

import java.util.concurrent.timeunit;

/**
 * 微信公众号:路人甲java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
 */
public class demo1 {

    public volatile static boolean exit = false;

    public static class t extends thread {
        @override
        public void run() {
            while (true) {
                //循环处理业务
                if (exit) {
                    break;
                }
            }
        }
    }

    public static void setexit() {
        exit = true;
    }

    public static void main(string[] args) throws interruptedexception {
        t t = new t();
        t.start();
        timeunit.seconds.sleep(3);
        setexit();
    }
}

代码中启动了一个线程,线程的run方法中有个死循环,内部通过exit变量的值来控制是否退出。timeunit.seconds.sleep(3);让主线程休眠3秒,此处为什么使用timeunit?timeunit使用更方便一些,能够很清晰的控制休眠时间,底层还是转换为thread.sleep实现的。程序有个重点:volatile关键字,exit变量必须通过这个修饰,如果把这个去掉,程序无法正常退出。volatile控制了变量在多线程中的可见性,关于volatile前面的文章中有介绍,此处就不再说了。

通过线程自带的中断标志控制

示例代码:

package com.itsoku.chat05;

import java.util.concurrent.timeunit;

/**
 * 微信公众号:路人甲java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
 */
public class demo2 {

    public static class t extends thread {
        @override
        public void run() {
            while (true) {
                //循环处理业务
                if (this.isinterrupted()) {
                    break;
                }
            }
        }
    }


    public static void main(string[] args) throws interruptedexception {
        t t = new t();
        t.start();
        timeunit.seconds.sleep(3);
        t.interrupt();
    }
}

运行上面的程序,程序可以正常结束。线程内部有个中断标志,当调用线程的interrupt()实例方法之后,线程的中断标志会被置为true,可以通过线程的实例方法isinterrupted()获取线程的中断标志。

线程阻塞状态中如何中断

示例代码:

package com.itsoku.chat05;

import java.util.concurrent.timeunit;

/**
 * 微信公众号:路人甲java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
 */
public class demo3 {

    public static class t extends thread {
        @override
        public void run() {
            while (true) {
                //循环处理业务
                //下面模拟阻塞代码
                try {
                    timeunit.seconds.sleep(1000);
                } catch (interruptedexception e) {
                    e.printstacktrace();
                }
            }
        }
    }


    public static void main(string[] args) throws interruptedexception {
        t t = new t();
        t.start();
    }
}

运行上面代码,发现程序无法结束。

在此先补充几点知识:

  1. 调用线程的interrupt()实例方法,线程的中断标志会被置为true
  2. 当线程处于阻塞状态时,调用线程的interrupt()实例方法,线程内部会触发interruptedexception异常,并且会清除线程内部的中断标志(即将中断标志置为false)

那么上面代码可以调用线程的interrupt()方法来引发interruptedexception异常,来中断sleep方法导致的阻塞,调整一下代码,如下:

package com.itsoku.chat05;

import java.util.concurrent.timeunit;

/**
 * 微信公众号:路人甲java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
 */
public class demo3 {

    public static class t extends thread {
        @override
        public void run() {
            while (true) {
                //循环处理业务
                //下面模拟阻塞代码
                try {
                    timeunit.seconds.sleep(1000);
                } catch (interruptedexception e) {
                    e.printstacktrace();
                    this.interrupt();
                }
                if (this.isinterrupted()) {
                    break;
                }
            }
        }
    }


    public static void main(string[] args) throws interruptedexception {
        t t = new t();
        t.start();
        timeunit.seconds.sleep(3);
        t.interrupt();
    }
}

运行结果:

java.lang.interruptedexception: sleep interrupted
    at java.lang.thread.sleep(native method)
    at java.lang.thread.sleep(thread.java:340)
    at java.util.concurrent.timeunit.sleep(timeunit.java:386)
    at com.itsoku.chat05.demo3$t.run(demo3.java:17)

程序可以正常结束了,分析一下上面代码,注意几点:

  1. main方法中调用了t.interrupt()方法,此时线程t内部的中断标志会置为true
  2. 然后会触发run()方法内部的interruptedexception异常,所以运行结果中有异常输出,上面说了,当触发interruptedexception异常时候,线程内部的中断标志又会被清除(变为false),所以在catch中又调用了this.interrupt();一次,将中断标志置为false
  3. run()方法中通过this.isinterrupted()来获取线程的中断标志,退出循环(break)

总结

  1. 当一个线程处于被阻塞状态或者试图执行一个阻塞操作时,可以使用thread.interrupt()方式中断该线程,注意此时将会抛出一个interruptedexception的异常,同时中断状态将会被复位(由中断状态改为非中断状态)
  2. 内部有循环体,可以通过一个变量来作为一个信号控制线程是否中断,注意变量需要volatile修饰
  3. 文中的几种方式可以结合起来灵活使用控制线程的中断

java高并发系列

java高并发系列连载中,总计估计会有四五十篇文章,可以关注公众号:javacode2018,获取最新文章。
java高并发系列 - 第11天:线程中断的几种方式