• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

关于java:为什么强烈不推荐使用stopsuspend方法来中断线程

java 搞代码 4年前 (2022-01-27) 35次浏览 已收录 0个评论

咱们晓得像stop、suspend这几种中断或者阻塞线程的办法在较高java版本中曾经被标记上了@Deprecated过期标签,那么为什么她们已经登上了java的历史舞台而又慢慢的推出了舞台呢,到底是兽性的扭曲还是道德的沦丧呢,亦或是她们不思进取被取而代之呢,如果是被取而代之,那么取而代之的又是何方人也,本文咱们将一探到底。

一、stop的闭幕

首先stop办法的作用是什么呢,用java源码中的一句正文来理解一下:Forces the thread to stop executing.,即强制线程进行执行,’Forces’仿佛曾经透漏出了stop办法的蛮狠无理。那么咱们再看看java开发者是怎们解释stop被淘汰了的:

咱们从中能够看出以下几点:

1.stop这种办法实质上是不平安的
2.应用Thread.stop进行线程会导致它解锁所有已锁定的监视器,即间接开释以后线程曾经获取到的所有锁,使得以后线程间接进入阻塞状态

咱们举例来看一下上边提到的两点:

public static void main(String[] args) throws InterruptedException {
        Object o1=new Object();
        Object o2=new Object();
        Thread t1=new Thread(()->{
              synchronized (o1)
              {
                  synchronized (o2)
                  {
                      try {
                          System.out.println("t1获取到锁");
                          Thread.sleep(5000);
                          System.out.println("t1完结");
                      } catch (InterruptedException e) {
                          e.printStackTrace();
                      }
                  }
              }
        });
        t1.start();
        Thread.sleep(1000);
        Thread t2=new Thread(()->{
            synchronized (o1)
            {
                synchronized (o2)
                {
                    try {
                        System.out.println("t2获取到锁");
                        Thread.sleep(5000);
                        System.out.println("t2完结");
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        });
        t2.start();
        t1.stop();
    }

运行后果:

能够看到,当线程t1在获取到o1和o2两个锁开始执行,在还没有执行完结的时候,主线程调用了t1的stop办法中断了t1的执行,开释了t1线程获取到的所有锁,中断后t2获取到了o1和o2锁,开始执行直到完结,而t1却夭折在了sleep的时候,sleep后的代码没有执行。

因而应用stop咱们在不晓得线程到底运行到了什么中央,暴力的中断了线程,如果sleep后的代码是资源开释、重要业务逻辑等比拟重要的代码的话,亦或是其余线程依赖t1线程的运行后果,那间接中断将可能造成很重大的结果。

那么不倡议应用stop中断线程咱们应该怎么去优雅的完结一个线程呢,咱们能够存java开发者的正文中窥探到一种解决方案:

能够看到java开发者举荐咱们应用以下两种办法来优雅的进行线程:

1.定义一个变量,由指标线程去一直的查看变量的状态,当变量达到某个状态时进行线程。

代码举例如下:

volatile static boolean flag=false;
public static void main(String[] args) throws InterruptedException {
        Object o1=new Object();
        Thread t1=new Thread(()->{
              synchronized (o1)
              {
                  try {
                      System.out.println("t1获取到锁");
                      while (!flag)
                          Thread.sleep(5000);//执行业务逻辑
                      System.out.println("t1完结");
                  } catch (InterruptedException e) {
                      e.printStackTrace();
                  }
              }
        });
        t1.start();
        Thread.sleep(1000);
        Thread t2=new Thread(()->{
            synchronized (o1)
            {
                try {
                    System.out.println("t2获取到锁");
                    Thread.sleep(5000);//执行业务逻辑
                    System.out.println("t2完结");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
        t2.start();
        flag=true;
    }

运行后果:

2.应用interrupt办法中断线程。

代码举例如下:

public static void main(String[] args) throws InterruptedException {
        Object o1=new Object();
        Thread t1=new Thread(()->{
              synchronized (o1)
              {
                  System.out.println("t1获取到锁");
                  while (!Thread.currentThread().isInterrupted()) {
                      for (int i = 0; i < 100; i++) {
                          if(i==50)
                              System.out.println();
                          System.out.print(i+" ");
                      }
                      System.out.println();
                  }
                  System.out.println("t1完结");
              }
        });
        t1.start();
        Thread t2=new Thread(()->{
            synchronized (o1)
            {
                try {
                    System.out.println("t2获取到锁");
                    Thread.sleep(5000);//执行业务逻辑
                    System.out.println("t2完结");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
        t2.start();
        t1.interrupt();
    }

运行后果:

咱们用while (!Thread.currentThread().isInterrupted())来一直判断以后线程是否被中断,中断的话则让线程天然沦亡并开释锁。能够看到调用interrupt办法后并不会像stop那样暴力的中断线程,会等到以后运行的逻辑完结后再查看是否中断,十分的优雅。

注:运行举例代码可能不会打印出数字,这是因为t1线程运行到while(!Thread.currentThread().isInterrupted())时,主线程曾经调了interrupt办法,因而屡次运行可能会打印出数字。

二、suspend的闭幕

suspend办法的作用是挂起某个线程直到调用resume办法来复原该线程,然而调用了suspend办法后并不会开释被挂起线程获取到的锁,正因如此就给suspend和resume这哥俩贴上了容易引发死锁的标签,当然这也正是导致suspend和resume退出历史舞台的罪魁祸首。同样咱们看看java开发者为suspend的淘汰给出的理由:

从中咱们能够得出以下论断:

1.suspend具备人造的死锁偏向
2.当某个线程被suspend后,该线程持有的锁不会被开释,其余线程也就不能拜访这些资源
3.suspend某个线程后,如果在resume的过程中出现异常导致resume办法执行失败,则lock无奈开释,导致死锁

接下来模仿一下由suspend引起的死锁场景,Talk is cheap,show my code:

public static void main(String[] args) throws InterruptedException {
        Object o1=new Object();
        Object o2=new Object();
        Thread t1=new Thread(()->{
              synchronized (o1)
              {
                  System.out.println("t1获取到o1锁开始执行");
                  try {
                      Thread.sleep(5000);//模仿执行业务逻辑
                  } catch (InterruptedException e) {
                      e.printStackTrace();
                  }
                  System.out.println("t1执行完结");
              }
        });
        t1.start();
        Thread t2=new Thread(()->{
            synchronized (o2)
            {
                System.out.println("t2获取到o2开始执行");
                try {
                    Thread.sleep(2000);//执行耗时业务
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                synchronized (o1)
                {
                    System.out.println("t2获取到o1锁开始继续执行");
                }
                System.out.println("t2执行完结");
            }
        });
        t2.start();

        Thread.sleep(1000);
        t1.suspend();
        //假如抛出了一个未知异样
        int i=1/0;
        t1.resume();
    }

运行后果:

能够看到,整个程序卡的死死的,在调用resume复原t1线程之前抛出了一个未知异样,导致t1始终挂起进而无奈开释o1锁,而t2须要获取到o1锁后能力继续执行,但苦苦期待,奈何o1被t1拿捏的死死的,从此整个程序就陷入了无尽的期待中—-死锁。

作者:浪舟子
http://www.gaodaima.com/qq_40400960/article/details/112651249


搞代码网(gaodaima.com)提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发送到邮箱[email protected],我们会在看到邮件的第一时间内为您处理,或直接联系QQ:872152909。本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:关于java:为什么强烈不推荐使用stopsuspend方法来中断线程

喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址