搞过Java线程的人都知道,stop这个方法是臭名昭著了,早就被弃用了,但是现在任然有很多钟情与他的人,永远都放不下他,因为从他的字面意思上我们可以知道他貌似可以停止一个线程,这个需求是每个搞线程开发的人都想要的操作,但是他并非是真正意义上的停止线程,而且停止线程还会引来一些其他的麻烦事,下面就来详细的介绍一下这个方法的历史:

从SUN的官方文档可以得知,调用Thread.stop()方法是不安全的,这是因为当调用Thread.stop()方法时,会发生下面两件事:

1. 即刻抛出ThreadDeath异常,在线程的run()方法内,任何一点都有可能抛出ThreadDeath Error,包括在catch或finally语句中。

2. 会释放该线程所持有的所有的锁,而这种释放是不可控制的,非预期的。

当线程抛出ThreadDeath异常时,会导致该线程的run()方法突然返回来达到停止该线程的目的。ThreadDetath异常可以在该线程run()方法的任意一个执行点抛出。但是,线程的stop()方法一经调用线程的run()方法就会即刻返回吗?

[java] view
plain
copy

  1. package com.threadstop.demo;
  2. public class ThreadStopTest {
  3. public static void main(String[] args) {
  4. try {
  5. Thread t = new Thread() {
  6. //对于方法进行了同步操作,锁对象就是线程本身
  7. public synchronized void run() {
  8. try {
  9. long start=System.currentTimeMillis();
  10. //开始计数
  11. for (int i = 0; i < 100000; i++)
  12. System.out.println("runing.." + i);
  13. System.out.println((System.currentTimeMillis()-start)+"ms");
  14. } catch (Throwable ex) {
  15. System.out.println("Caught in run: " + ex);
  16. ex.printStackTrace();
  17. }
  18. }
  19. };
  20. //开始计数
  21. t.start();
  22. //主线程休眠100ms
  23. Thread.sleep(100);
  24. //停止线程的运行
  25. t.stop();
  26. } catch (Throwable t) {
  27. System.out.println("Caught in main: " + t);
  28. t.printStackTrace();
  29. }
  30. }
  31. }

运行结果如下:

由于打印的数据太多了,就没有全部截图了,但是我们可以看到,调用了stop方法之后,线程并没有停止,而是将run方法执行完。那这个就诡异了,多次运行之后发现每次运行的结果都表明,工作线程并没有停止,而是每次都成功的数完数(执行完run方法),然后正常中止,而不是由stop()方法进行终止的。这个是为什么呢?根据SUN的文档,原则上只要一调用thread.stop()方法,那么线程就会立即停止,并抛出ThreadDeath error,查看了Thread的源代码后才发现,原先Thread.stop(Throwable
obj)
方法是同步的,而我们工作线程的run()方法也是同步,那么这样会导致主线程和工作线程共同争用同一个锁(工作线程对象本身),由于工作线程在启动后就先获得了锁,所以无论如何,当主线程在调用t.stop()时,它必须要等到工作线程的run()方法执行结束后才能进行,结果导致了上述奇怪的现象。

下面看一下stop的源码:

[java] view
plain
copy

  1. @Deprecated
  2. public final void stop() {
  3. stop(new ThreadDeath());
  4. }

再进到stop看:

[java] view
plain
copy

  1. @Deprecated
  2. public final synchronized void stop(Throwable obj) {
  3. if (obj == null)
  4. throw new NullPointerException();
  5. SecurityManager security = System.getSecurityManager();
  6. if (security != null) {
  7. checkAccess();
  8. if ((this != Thread.currentThread()) ||
  9. (!(obj instanceof ThreadDeath))) {
  10. security.checkPermission(SecurityConstants.STOP_THREAD_PERMISSION);
  11. }
  12. }
  13. // A zero status value corresponds to "NEW", it can't change to
  14. // not-NEW because we hold the lock.
  15. if (threadStatus != 0) {
  16. resume(); // Wake up thread if it was suspended; no-op otherwise
  17. }
  18. // The VM can handle all thread states
  19. stop0(obj);
  20. }

stop0(obj)是一个native方法,我们看到stop方法是同步的,而这个同步的锁对象正好也是线程本身,所以造成上面的现象。

把上述工作线程的run()方法的同步去掉,再进行执行,结果就如上述第一点描述的那样了,运行结果如下:

从结果中我们可以看到,调用stop方法会抛出一个ThreadDeath异常,这时候run方法也就执行结束了,线程就终止了,这种是用抛异常来结束线程的,但是这种抛出线程是不安全的,因为他不可控制,不知道到在run方法中的何处就可能抛出异常,所以是危险的。下面在看一下stop的这个隐患可能造成的影响:

接下来是看看当调用thread.stop()时,被停止的线程会不会释放其所持有的锁,看如下代码:

[java] view
plain
copy

  1. public static void main(String[] args) {
  2. <span style="white-space:pre">  </span>//定义锁对象
  3. final Object lock = new Object();
  4. //定义第一个线程,首先该线程拿到锁,而后等待3s,之后释放锁
  5. try {
  6. Thread t0 = new Thread() {
  7. public void run() {
  8. try {
  9. synchronized (lock) {
  10. System.out.println("thread->" + getName()  + " acquire lock.");
  11. sleep(3*1000);
  12. System.out.println("thread->" + getName() + " 等待3s");
  13. System.out.println("thread->" + getName()  + " release lock.");
  14. }
  15. } catch (Throwable ex) {
  16. System.out.println("Caught in run: " + ex);
  17. ex.printStackTrace();
  18. }
  19. }
  20. };
  21. //定义第二个线程,等待拿到锁对象
  22. Thread t1 = new Thread() {
  23. public void run() {
  24. synchronized (lock) {
  25. System.out.println("thread->" + getName()  + " acquire lock.");
  26. }
  27. }
  28. };
  29. //线程一先运行,先拿到lock
  30. t0.start();
  31. //而后主线程等待100ms,为了做延迟
  32. Thread.sleep(100);
  33. //停止线程一
  34. //t0.stop();
  35. //这时候在开启线程二
  36. t1.start();
  37. } catch (Throwable t) {
  38. System.out.println("Caught in main: " + t);
  39. t.printStackTrace();
  40. }
  41. }

运行结果如下:

从运行结果中我们可以看到,当没有进行t0.stop()方法的调用时, 可以发现,两个线程争用锁的顺序是固定的。这个现象是正常的。

下面我们把t0.stop注释的哪行,删除注释,调用t0.stop()方法,运行结果如下:

从运行结果中我们可以看到,调用了t0.stop()方法后,可以发现,t0线程抛出了ThreadDeath error并且t0线程释放了它所占有的锁。

从上面的程序验证结果来看,thread.stop()确实是不安全的。它的不安全主要是:释放该线程所持有的所有的锁。一般任何进行加锁的代码块,都是为了保护数据的一致性,如果在调用thread.stop()后导致了该线程所持有的所有锁的突然释放(不可控制),那么被保护数据就有可能呈现不一致性,其他线程在使用这些被破坏的数据时,有可能导致一些很奇怪的应用程序错误。

下面顺便说一下:

Java中多线程锁释放的条件:

1)执行完同步代码块,就会释放锁。(synchronized)

2)在执行同步代码块的过程中,遇到异常而导致线程终止,锁也会被释放。(exception)

3)在执行同步代码块的过程中,执行了锁所属对象的wait()方法,这个线程会释放锁,进入对象的等待池。(wait)

从上面的三点我就可以看到stop方法释放锁是在第二点的,通过抛出异常来释放锁,通过证明,这种方式是不安全的,不可靠的。

好吧,Thread中的stop方法就说到这里了,后续还会有关Thread中的一些方法的详解,一定要关注奥!

Java中的线程Thread方法之---stop()的更多相关文章

  1. Java中的线程Thread方法之---interrupt()

    前几篇都介绍了Thread中的几个方法,相信大家都发现一个相似点,那就是sleep,join,wait这样的阻塞方法都必须捕获一个InterruptedException异常,顾名思义就是一个线程中断 ...

  2. Java中的线程Thread方法之---suspend()和resume()

    前篇说到了Thread中的join方法,这一篇我们就来介绍一下suspend()和resume()方法,从字面意义上可以了解到这两个方法是一对的,suspend()方法就是将一个线程挂起(暂停),re ...

  3. Java中的线程Thread方法之---join()

    上一篇我们说到了Thread中的stop方法,这一篇我们再来看一下方法join的使用,那么方法Join是干啥用的? 简单回答,同步,如何同步? 怎么实现的? 下面将逐个回答. join方法从字面上的意 ...

  4. Java中的线程Thread总结

    首先来看一张图,下面这张图很清晰的说明了线程的状态与Thread中的各个方法之间的关系,很经典的! 在Java中创建线程有两种方法:使用Thread类和使用Runnable接口. 要注意的是Threa ...

  5. Java中实现线程的方法

    Java中有几种方法可以实现一个线程?用什么关键字修饰同步方法?stop()和suspend()方法为何不推荐使用? 第一种:继承Thread类 new Thread(){}.start();这表示调 ...

  6. Java 中的线程 thread

    一.问:线程有哪些状态? new, runnable, running, waiting, dead 线程状态间的流转 二.问:线程实现方式? 实现 Runnable 接口,然后new Thread, ...

  7. JAVA中创建线程的三种方法及比较

    JAVA中创建线程的方式有三种,各有优缺点,具体如下: 一.继承Thread类来创建线程 1.创建一个任务类,继承Thread线程类,因为Thread类已经实现了Runnable接口,然后重写run( ...

  8. JAVA中创建线程池的五种方法及比较

    之前写过JAVA中创建线程的三种方法及比较.这次来说说线程池. JAVA中创建线程池主要有两类方法,一类是通过Executors工厂类提供的方法,该类提供了4种不同的线程池可供使用.另一类是通过Thr ...

  9. 【Java中的线程】java.lang.Thread 类分析

    进程和线程 联想一下现实生活中的例子--烧开水,烧开水时是不是不需要在旁边守着,交给热水机完成,烧开水这段时间可以去干一点其他的事情,例如将衣服丢到洗衣机中洗衣服.这样开水烧完,衣服洗的也差不多了.这 ...

随机推荐

  1. 【LeetCode】贪心

    [452] Minimum Number of Arrows to Burst Balloons [Medium] 给一堆线段,使用最少的arrow,穿过所有的线段.陈题,第一条线段的终点. Inpu ...

  2. matlab filtfilt 函数

    紧接上一篇,简单分析matlab中的非常好用的 filtfilt 函数,一款零相移滤波函数. 其matlab中的语法如下: y = filtfilt(data,x);1非常简单,不是一般的简单!然而, ...

  3. Vue学习笔记【11】——Vue调试工具vue-devtools的安装步骤和使用

    1.fq安装 2.本地安装: Google浏览器 chrome://extensions/ ,打开扩展程序→打开开发者模式→加载已解压的扩展程序,选择解压后的扩展程序包即可.

  4. SP6779 GSS7 - Can you answer these queries VII

    纯数据结构题,没有思维难度.直接用线段树求最大子段和的方法完成树上路径的合并.注意链上合并顺序要符合序列的前后顺序. #include <cstdio> #include <cstr ...

  5. renren-fast-vue-动态路由-添加路由-方式一(直接在原有结构上添加)

    在原有文件夹夹下新建自己的组件 在 mock/modules/sys-menu.js 中引入 实现路由的添加

  6. 搭建appium自动化测试环境

    注意:请使用不用的手机测试,appium会把微信app重新安装,记录都会清除 一.安装Java JDK JDK下载地址:https://www.oracle.com/technetwork/java/ ...

  7. windows系统使用

    1.访问局域网共享的文件,用 \\ip号 2.电脑的硬件名称(设备管理器中)是可以用软件修改的. 3.电脑中每一个连接网络的设备都有一个网卡地址(MAC地址),如无线网卡地址.有线网卡地址. 4.wi ...

  8. (转)ubuntu 下安装mysql5.5.30的过程以及遇到的问题

    转:http://blog.chinaunix.net/uid-27103408-id-3280584.html 由于实验需要安装mysql,当然我们可以通过sudo apt-get install ...

  9. I. Five Day Couple--“今日头条杯”首届湖北省大学程序设计竞赛(网络同步赛)

    题目描述:链接点此 这套题的github地址(里面包含了数据,题解,现场排名):点此 链接:https://www.nowcoder.com/acm/contest/104/H来源:牛客网 题目描述 ...

  10. 听说江苏省没有webSocket服务硬件

    听说江苏省没有webSocket服务硬件 昨天项目上线,我门开发采用的webSocket做实时轮询,然后开发部老总怒怼"江苏省没有webSocket服务硬件,江苏省没有webSocket服务 ...