iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > Python >java线程中断 interrupt 和 LockSupport解析
  • 936
分享到

java线程中断 interrupt 和 LockSupport解析

java 线程中断java interrupt LockSupport 2023-02-22 18:02:42 936人浏览 薄情痞子

Python 官方文档:入门教程 => 点击学习

摘要

目录本文大纲线程中断相关方法sleep 响应中断LockSupport方法介绍park/unpark和 wait/notify区别本文大纲 本文章将要介绍的内容有以下几点,读者朋友

本文大纲

本文章将要介绍的内容有以下几点,读者朋友也可先自行思考一下相关问题:

  • 线程中断 interrupt 方法怎么理解,意思就是线程中断了吗?那当前线程还能继续执行吗?
  • 判断线程是否中断的方法有几个,它们之间有什么区别?
  • LockSupport的 park/unpark 和 wait/notify 有什么区别?
  • sleep 方法是怎么响应中断的?
  • park 方法又是怎么响应中断的?

线程中断相关方法

线程中和中断相关的方法有三个,分别介绍如下:

1) interrupt

我们一般都说这个方法是用来中断线程的,那么这个中断应该怎么理解呢? 就是说把当前正在执行的线程中断掉,不让它继续往下执行吗?

其实,不然。 此处,说的中断仅仅是给线程设置一个中断的标识(设置为true),线程还是会继续往下执行的。而线程怎么停止,则需要由我们自己去处理。 一会儿会用代码来说明这个。

2) isInterrupted

判断当前线程的中断状态,即判断线程的中断标识是true还是false。 注意,这个方法不会对线程原本的中断状态产生任何影响。

3) interrupted

也是判断线程的中断状态的。但是,需要注意的是,这个方法和 isInterrupted 有很大的不同。我们看下它们的源码

public boolean isInterrupted() {  
    return isInterrupted(false);  
}
public static boolean interrupted() {  
    return currentThread().isInterrupted(true);  
}
//调用同一个方法,只是传参不同
private native boolean isInterrupted(boolean ClearInterrupted);

首先 isInterrupted 方法是线程对象的方法,而 interrupted 是Thread类的静态方法。

其次,它们都调用了同一个本地方法 isInterrupted,不同的只是传参的值,这个参数代表的是,是否要把线程的中断状态清除(清除即不论之前的中断状态是什么值,最终都会设置为false)。

因此,interrupted 静态方法会把原本线程的中断状态清除,而 isInterrupted 则不会。所以,如果你调用两次 interrupted 方法,第二次就一定会返回false,除非中间又被中断了一次。

下面证明一下 interrupt 方法只是设置一个中断状态,而不是使当前线程中断运行:

public class TestFlag {
    static volatile boolean flag = true;
    public static void main(String[] args) throws InterruptedException {
        Thread t = new Thread(new Runnable(){
            @Override
            public void run() {
                System.out.println("线程中断标志:"+Thread.currentThread().isInterrupted());
                while (flag){
                }
                System.out.println("标志flag为:" + flag);
                System.out.println("线程中断标志:"+Thread.currentThread().isInterrupted());
                System.out.println("我还在继续执行");
            }
        });
        t.start();
        Thread.sleep(100);
        flag = false;
        t.interrupt();
    }
}

运行结果:

线程中断标志:false
标志flag为:false
线程中断标志:true
我还在继续执行

当线程启动,还没调用中断方法时,中断状态为false,然后调用中断方法,并把flag设置为false。此时,run方法跳出while死循环。我们会发现线程的中断状态为true,但是线程还是会继续往下执行,直到执行结束。

sleep 响应中断

线程中常用的阻塞方法,如sleep,join和wait 都会响应中断,然后抛出一个中断异常 InterruptedException。但是,注意此时,线程的中断状态会被清除。所以,当我们捕获到中断异常之后,应该保留中断信息,以便让上层代码知道当前线程中断了。通常有两种方法可以做到。

一种是,捕获异常之后,再重新抛出异常,让上层代码知道。另一种是,在捕获异常时,通过 interrupt 方法把中断状态重新设置为true。

下面,就以sleep方法为例,捕获中断异常,然后重新设置中断状态:

public class TestInterrupt {
    public static void main(String[] args) throws InterruptedException {
        Thread t = new Thread(new Runnable() {
            private int count = 0;
            @Override
            public void run() {
                try {
                    count = new Random().nextInt(1000);
                    count = count * count;
                    System.out.println("count:"+count);
                    Thread.sleep(5000);
                } catch (Exception e) {
                    System.out.println(Thread.currentThread().getName()+"线程第一次中断标志:"+Thread.currentThread().isInterrupted());
                    //重新把线程中断状态设置为true,以便上层代码判断
                    Thread.currentThread().interrupt();
                    System.out.println(Thread.currentThread().getName()+"线程第二次中断标志:"+Thread.currentThread().isInterrupted());
                }
            }
        });
        t.start();
        Thread.sleep(100);
        t.interrupt();
    }
}

结果:

count:208849
Thread-0线程第一次中断标志:false
Thread-0线程第二次中断标志:true

LockSupport方法介绍

LockSupport 方法中重要的两个方法就是park 和 unpark 。

park和interrupt中断

park方法可以阻塞当前线程,如果调用unpark方法或者中断当前线程,则会从park方法中返回。

park方法对中断方法的响应和 sleep 有一些不太一样。它不会抛出中断异常,而是从park方法直接返回,不影响线程的继续执行。我们看下代码:

public class LockSupportTest {
    public static void main(String[] args) throws InterruptedException {
        Thread t = new Thread(new ParkThread());
        t.start();
        Thread.sleep(100); //①
        System.out.println(Thread.currentThread().getName()+"开始唤醒阻塞线程");
        t.interrupt();
        System.out.println(Thread.currentThread().getName()+"结束唤醒");
    }
}
class ParkThread implements Runnable{
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName()+"开始阻塞");
        LockSupport.park();
        System.out.println(Thread.currentThread().getName()+"第一次结束阻塞");
        LockSupport.park();
        System.out.println("第二次结束阻塞");
    }
}

打印结果如下:

Thread-0开始阻塞
main开始唤醒阻塞线程
main结束唤醒
Thread-0第一次结束阻塞
第二次结束阻塞

当调用interrupt方法时,会把中断状态设置为true,然后park方法会去判断中断状态,如果为true,就直接返回,然后往下继续执行,并不会抛出异常。注意,这里并不会清除中断标志。

unpark

unpark会唤醒被park的指定线程。但是,这里要说明的是,unpark 并不是简单的直接去唤醒被park的线程。看下jdk的解释:

unpark只是给当前线程设置一个许可证。如果当前线程已经被阻塞了(即调用了park),则会转为不阻塞的状态。如若不然,下次调用park方法的时候也会保证不阻塞。这句话的意思,其实是指,park和unpark的调用顺序无所谓,只要unpark设置了这个许可证,park方法就可以在任意时刻消费许可证,从而不会阻塞方法。

还需要注意的是,许可证最多只有一个,也就是说,就算unpark方法调用多次,也不会增加许可证。 我们可以通过代码验证,只需要把上边代码修改一行即可:

//LockSupportTest类
//原代码
t.interrupt();
//修改为
LockSupport.unpark(t);
LockSupport.unpark(t);

就会发现,只有第一次阻塞会被唤醒,但是第二次依然会继续阻塞。结果如下:

Thread-0开始阻塞
main开始唤醒阻塞线程
main结束唤醒
Thread-0第一次结束阻塞
第二次结束阻塞

另外,在此基础上,把主线程的sleep方法去掉(代码中①处),让主线程先运行,也就是有可能先调用unpark方法,然后子线程才开始调用park方法阻塞。我们会发现,出现以下结果,证明了上边我说的park方法和unpark不分先后顺序,park方法可以随时消费许可证。

main开始唤醒阻塞线程
main结束唤醒
Thread-0开始阻塞
Thread-0第一次结束阻塞

park/unpark和 wait/notify区别

了解了 park/unpark的用法之后,想必你也能分析出来它们和 wait、notify有什么不同之处了。

  • wait和notify方法必须和同步 synchronized一块儿使用。而park/unpark使用就比较灵活了,没有这个限制,可以在任何地方使用。
  • park/unpark 使用时没有先后顺序,都可以使线程不阻塞(前面代码已验证)。而wait必须在notify前先使用,如果先notify,再wait,则线程会一直等待。
  • notify只能随机释放一个线程,并不能指定某个特定线程,notifyAll是释放锁对象中的所有线程。而unpark方法可以唤醒指定的线程。
  • 调用wait方法会使当前线程释放锁资源,但使用的前提是必须已经获得了锁。 而park不会释放锁资源。(以下代码验证)
public class LockSyncTest {
    private static Object lock = new Object();
    //保存调用park的线程,以便后续唤醒
    private static Thread parkedThread;
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(()->{
             synchronized (lock){
                 System.out.println("unpark前");
                 LockSupport.unpark(parkedThread);
                 System.out.println("unpark后");
             }
        });
        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                //和t1线程用同一把锁时,park不会释放锁资源,若换成this锁,则会释放锁
                synchronized (lock){
                    System.out.println("park前");
                    parkedThread = Thread.currentThread();
                    LockSupport.park();
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println("park后");
                }
            }
        });
        t2.start();
        Thread.sleep(100);
        t1.start();
    }
}
//打印结果
//park前

以上代码,会一直卡在t2线程,因为park不会释放锁,因此t1也无法执行。

如果把t2的锁换成this锁,即只要和t1不是同一把锁,则t1就会正常执行,然后把t2线程唤醒。打印结果如下:

park前
unpark前
unpark后
park后

以上就是java线程中断 interrupt 和 LockSupport的详细内容,更多关于java 线程中断的资料请关注编程网其它相关文章!

--结束END--

本文标题: java线程中断 interrupt 和 LockSupport解析

本文链接: https://www.lsjlt.com/news/197182.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • java线程中断 interrupt 和 LockSupport解析
    目录本文大纲线程中断相关方法sleep 响应中断LockSupport方法介绍park/unpark和 wait/notify区别本文大纲 本文章将要介绍的内容有以下几点,读者朋友...
    99+
    2023-02-22
    java 线程中断 java interrupt LockSupport
  • 线程中断 interrupt 和 LockSupport
    本文章将要介绍的内容有以下几点,读者朋友也可先自行思考一下相关问题:线程中断 interrupt 方法怎么理解,意思就是线程中断了吗?那当前线程还能继续执行吗?判断线程是否中断的方法有几个,它们之间有什么区别?LockSupport的 pa...
    99+
    2023-06-02
  • java线程中断interrupt和LockSupport的方法是什么
    本文小编为大家详细介绍“java线程中断interrupt和LockSupport的方法是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“java线程中断interrupt和LockSupport的方法是什么”文章能帮助大家解决疑惑,下...
    99+
    2023-07-05
  • java线程中断之interrupt和stop
    interrupt方法当我们调用线程的sleep方法或者join方法时,都可以让某一些线程处于等待状态,调用当前线程的interrupt()方法就可以打断该阻塞状态。interrupt方法不会让线程结束。public void interr...
    99+
    2019-09-17
    java教程 java 线程中断 interrup stop
  • Java多线程之Interrupt中断线程详解
    目录一、测试代码二、测试三、执行过程描述四、输出日志五、结论六、主要方法释义七、DEMO八、拓展程序九、实战一、测试代码 https://gitee.com/zture/sprin...
    99+
    2022-11-12
  • Java多线程之Interrupt中断线程的示例分析
    小编给大家分享一下Java多线程之Interrupt中断线程的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!一、测试代码https://gitee.com/zture/spring-test/blob/master...
    99+
    2023-06-15
  • java 中断线程的几种方式 interrupt()详解
    中断   中断(Interrupt)一个线程意味着在该线程完成任务之前停止其正在进行的一切,有效地中止其当前的操作。线程是死亡、还是等待新的任务或是继续运行至下一步,就取决于这个程序...
    99+
    2022-11-12
  • Java线程中断interrupt的常用方法
    目录前言示例说明注意事项Object#wait 和 Thread.sleep 差异在哪里总结前言 这里主要探讨中断常用的三个方法: interrupt()。在一个线程中调用需要中断现...
    99+
    2022-11-13
  • java中使用interrupt通知线程停止详析
    目录普通情况停止线程在可能被阻塞情况下停止线程在每次迭代后都阻塞的情况下停止线程两种停止线程最佳方法总结前言: 使用 interrupt 来通知线程停止运行,而不...
    99+
    2022-11-13
  • Java线程池的分析和使用详解
    目录1.   引言2.线程池的使用线程池的创建线程池的关闭3.   线程池的分析4.   合...
    99+
    2022-11-12
  • SpringBoot线程池和Java线程池的使用和实现原理解析
    目录SpringBoot线程池和Java线程池的用法和实现原理使用默认的线程池方式一:通过@Async注解调用方式二:直接注入 ThreadPoolTaskExecutor...
    99+
    2023-05-15
    SpringBoot线程池和Java线程池用法 SpringBoot线程池
  • java中进程和线程的示例分析
    小编给大家分享一下java中进程和线程的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!为什么会有进程在简单的批处理操作系统中,作业时串行执行的,即一个作业...
    99+
    2023-06-20
  • Java高级-解析Java中的多线程机制(转)
    Java高级-解析Java中的多线程机制(转)[@more@]进程与应用程序的区别进程(Process)是最初定义在Unix等多用户、多任务操作系统环境下用于表示应用程序在内存环境中基本执行单元的概念。以Unix操作系统为例,进程是Unix...
    99+
    2023-06-03
  • 详解Java线程池和Executor原理的分析
    详解Java线程池和Executor原理的分析线程池作用与基本知识在开始之前,我们先来讨论下“线程池”这个概念。“线程池”,顾名思义就是一个线程缓存。它是一个或者多个线程的集合,用户可以把需要执行的任务简单地扔给线程池,而不用过多的纠结与执...
    99+
    2023-05-31
    java 线程池 executor
  • 怎么解析Redis6中的单线程和多线程模型
    这篇文章的内容主要围绕怎么解析Redis6中的单线程和多线程模型进行讲述,文章内容清晰易懂,条理清晰,非常适合新手学习,值得大家去阅读。感兴趣的朋友可以跟随小编一起阅读吧。希望大家通过这篇文章有所收获!1....
    99+
    2022-10-19
  • java高并发的线程中断的几种方式详解
    目录通过一个变量控制线程中断通过线程自带的中断标志控制线程阻塞状态中如何中断?总结通过一个变量控制线程中断 代码: package com.itsoku.chat05; imp...
    99+
    2022-11-12
  • 如何解析Java线程池在使用中的问题
    本篇文章为大家展示了如何解析Java线程池在使用中的问题,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。Java线程池需要我们不断的学习,其实我们在使用的时候还是有不少问题需要我们解决。我们实现了一个...
    99+
    2023-06-17
  • java高并发中怎样理解进程和线程
    java高并发中怎样理解进程和线程,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。进程进程(Process)是计算机中的程序关于某数据集合上的一次运行活动,是系统进行资源分配和调...
    99+
    2023-06-25
  • java多线程编程同步器Future和FutureTask解析及代码示例
    publicinterfaceFuture<V>Future表示异步计算的结果。它提供了检查计算是否完成的方法,以等待计算的完成,并获取计算的结果。计算完成后只能使用get方法来获取结果,如有必要,计算完成前可以阻塞此方法。取消...
    99+
    2023-05-30
    java 多线程 ava
  • Java线程中关键字和方法的示例分析
    这篇文章主要为大家展示了“Java线程中关键字和方法的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Java线程中关键字和方法的示例分析”这篇文章吧。一、volatile关键字1,vol...
    99+
    2023-06-29
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作