一.停止线程会带来什么?

对于单线程中,停止单线程就是直接使用关键字return或者break,但是在停止多线程时是让线程在完成任务前去开启另外一条线程,必须放弃当前任务,而这个过程是不可预测,所以必须去做好防备。

二.认识停止线程的几个方法

 2.1三个被弃用的方法

stop()、suspend()、resume()。

stop()方法被弃用的原因:无论线程执行到了什么位置,一旦被stop就会被马上强制中断,并且释放线程所有持有锁对象,根本就没有安全性。

suspend()和resume()这一对烂兄烂弟,因为只有其他线程调用resume这个方法时他才会释放suspend这个方法的锁,这样就极易造成死锁。

 2.2三个名字差不多的方法

interrupt()中断线程、interrupted()判断当前线程是否停止、isInterrupted()判断线程是否停止。

首先来介绍一下其中两个名字最相近的:interrupted()、isInterrupted()方法,这两个方法是用来测试线程是否被中断的。

来看一下原码:

isInterrupted():

    public boolean isInterrupted() {
return isInterrupted(false);
}

interrupted():

    public static boolean interrupted() {
return currentThread().isInterrupted(true);
}

可以明显看出interrupted()方法是静态的,而isInterrupted()是非静态的,但都是返回线程是否被中断。  

下面我们来做一个测试,代码如下:

    public class Is_Interrupt extends Thread{
@Override
public void run(){
for(int i=0;i<1000;i++){
System.out.println("当前i的值为:"+i);
}
}
public static void main(String[] args) {
Is_Interrupt is_interrupt=new Is_Interrupt();
is_interrupt.start();
is_interrupt.interrupt();//中断线程
System.out.println("线程是否已经暂停?"+is_interrupt.interrupted());
}
}

输出结果为:

发现线程没有停止。

但是这里面还有一个线程就是main线程,而interrupted()返回的就是当前线程的中断状态,那么执行这个方法的就是main线程,而main线程此时当然没有中断。

我们将interrupted()方法改为isInterrupted()试试效果,代码如下:

    public class Is_Interrupt extends Thread{
@Override
public void run(){
for(int i=0;i<1000;i++){
System.out.println("当前i的值为:"+i);
}
}
public static void main(String[] args) {
Is_Interrupt is_interrupt=new Is_Interrupt();
is_interrupt.start();
is_interrupt.interrupt();//中断线程
System.out.println("线程是否已经暂停?"+is_interrupt.isInterrupted());
}
}

结果如下:

说明线程是已经停止了的,只不过我们使用错了一个方法而已判断成了main线程的状态。

总结:isInterrupted方法是返回调用对象的中断状态,而静态方法interrupted是返回当前线程的中断状态。

既然了解了这个误区以后我们再来看看下面的代码:

public class InterruptText extends Thread{
public static void main(String[] args) {
System.out.println("main线程启动!");
System.out.println( Thread.interrupted());//判断当前线程是否中断
System.out.println(currentThread().isInterrupted());//通过currentThread().isInterrupted()同样也可以达到相同的目的,在单线程中
}
}

输出:

这个是没有问题的。

那么我们多次调用这个interrupted方法呢?

public class InterruptText extends Thread{
public static void main(String[] args) {
System.out.println("main线程启动!");
currentThread().interrupt();//中断main线程
System.out.println( Thread.interrupted());//判断当前main线程是否中断
System.out.println( Thread.interrupted());//再一次判断当main前线程是否中断
}
}

结果:按照常理应该两次返回ture,但是为什么变成了第二次变成了false了呢?

其实interrupted就是在清除状态,你两次调用当然会将true变成flase但是他还是中断状态,但是isInterrupted是不清除的。

interrupt()方法:中断线程

三.停止线程

3.1通过异常来暂停线程

首先来看一段代码:

public class InterruptText extends Thread{
@Override
public void run(){
try {
for (int i = 0; i < 100000; i++) { if (currentThread().isInterrupted()) {//如果线程中断
throw new InterruptedException();//抛出异常
} else {
System.out.println(i);
} }
System.out.println("线程没有终止");
}catch (InterruptedException e){
e.printStackTrace();
}
}
public static void main(String[] args) throws Exception {
InterruptText interruptText=new InterruptText();
interruptText.start();
Thread.sleep(100);
interruptText.interrupt();
}
}

结果:

异常停止方法的策略就是:在遇到中断时,抛出异常,扑捉异常。

3.2在睡眠中中断线程

代码如下:

public class InterruptText2 extends Thread {
@Override
public void run(){
try {
Thread.sleep(10000);//线程睡眠
} catch (InterruptedException e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
InterruptText2 interruptText2=new InterruptText2();
interruptText2.start();//开启线程,但是线程处于睡眠状态
interruptText2.interrupt();//在睡眠状态中断线程 }
}

结果为:

3.3线程让步

方法:yield(),当前线程放弃所有的资源,去执行其他的任务。但是放弃资源的时间不可以预判的。

3.4守护线程

 守护线程的定义:守护线程是一种特殊的线程,区别于非守护线程,当程序中不存在非守护线程时,守护线程退出,程序退出。

设置守护线程:setDaemon(),参数为ture则该线程为守护线程。

深入理解Java中停止线程的更多相关文章

  1. [译]线程生命周期-理解Java中的线程状态

    线程生命周期-理解Java中的线程状态 在多线程编程环境下,理解线程生命周期和线程状态非常重要. 在上一篇教程中,我们已经学习了如何创建java线程:实现Runnable接口或者成为Thread的子类 ...

  2. 逐步理解Java中的线程安全问题

    什么是Java的线程安全问题? 线程安全就是多线程访问时,采用了加锁机制,当一个线程访问该类的某个数据时,进行保护,其他线程不能进行访问直到该线程读/写完,其他线程才可使用.不会出现数据不一致或者数据 ...

  3. Java中一个线程只有六个状态。至于阻塞、可运行、挂起状态都是人们为了便于理解,自己加上去的。

    java中,线程的状态使用一个枚举类型来描述的.这个枚举一共有6个值: NEW(新建).RUNNABLE(运行).BLOCKED(锁池).TIMED_WAITING(定时等待).WAITING(等待) ...

  4. Java中的线程池用过吧?来说说你是怎么理解线程池吧?

    前言 Java中的线程池用过吧?来说说你是怎么使用线程池的?这句话在面试过程中遇到过好几次了.我甚至这次标题都想写成[Java八股文之线程池],但是有点太俗套了.虽然,线程池是一个已经被说烂的知识点了 ...

  5. Java多线程(五)停止线程 在沉睡中停止线程

    在沉睡中停止线程会抛出异常 public class SleepInterruptDemo extends Thread { public void run() { super.run(); try ...

  6. Java中的线程Thread总结

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

  7. Java多线程编程(1)--Java中的线程

    一.程序.进程和线程   程序是一组指令的有序集合,也可以将其通俗地理解为若干行代码.它本身没有任何运行的含义,它只是一个静态的实体,它可能只是一个单纯的文本文件,也有可能是经过编译之后生成的可执行文 ...

  8. Java中的线程

    http://hi.baidu.com/ochzqvztdbabcir/item/ab9758f9cfab6a5ac9f337d4 相濡以沫 Java语法总结 - 线程 一 提到线程好像是件很麻烦很复 ...

  9. 深入理解Java中的不可变对象

    深入理解Java中的不可变对象 不可变对象想必大部分朋友都不陌生,大家在平时写代码的过程中100%会使用到不可变对象,比如最常见的String对象.包装器对象等,那么到底为何Java语言要这么设计,真 ...

随机推荐

  1. go 编译:交叉编译&编译执行过程

    1. 交叉编译 编译Windows程序和mac程序 GOOS=windows GOARCH-amd64 go build main.go 转自:https://www.cnblogs.com/mafe ...

  2. Idea 提交代码到码云(提交到github也大同小异)

    1.首先下载安装git,下载地址https://git-scm.com/download/win.按默认选择安装即可,不成功的自行百度 2.安装完成后双击git.exe,调出命令界面(此处需要自己去申 ...

  3. NOI 2017 整数(线段树)

    题意 https://loj.ac/problem/2302 思路 拆分成每个二进制位的加减来考虑,维护那个整数的二进制位.不难发现,进位就是找右边第一个 \(0\) 的位置,并将其赋值为 \(1\) ...

  4. ComponentOne 产品经理:为什么要从C1Report迁移到FlexReport

    概述 如果你正在使用ComponentOne Enterprise 的Reports for WinForm 报表控件(C1Report),你一定会喜欢更为强大的FlexReport! FlexRep ...

  5. Mac cnpm安装失败及解决方案

    首先安装node 官网下载安装包,傻瓜式安装:https://nodejs.org/zh-cn/ 淘宝镜像安装cnpm, 在终端输入: npm install -g cnpm --registry=h ...

  6. lambda Helper

    /// <summary> /// 操作表达式共通类,条件并且,或者操作等 /// </summary> public static class PredicateBuilde ...

  7. Rancher2.0中部署Longhorn分布式存储实验

    目录 1.简介 2.实验环境 3.应用商店中部署longhorn 4.创建工作负载,使用longhorn存储 5.查看longhorn UI 6.注意事项 1.简介: Longhorn是Rancher ...

  8. 7.6 GRASP原则六: 多态 Polymorphism

    GRASP原则六: 多态 Polymorphism  How to handle alternative behaviors based on type 如何处理依据类型不同而有 不同行为的一类需求 ...

  9. 三款免费实用的文件夹同步/备份软件推荐 (SyncToy/FreeFileSync/Compare Advance)

    三款免费实用的本地文件夹同步/备份软件推荐 (SyncToy/FreeFileSync/Compare Advance) Microsoft SyncToy SyncToy 是由 微软 推出的一款免费 ...

  10. 『TensorFlow』读书笔记_Inception_V3_上

    1.网络背景 自2012年Alexnet提出以来,图像分类.目标检测等一系列领域都被卷积神经网络CNN统治着.接下来的时间里,人们不断设计新的深度学习网络模型来获得更好的训练效果.一般而言,许多网络结 ...