并发编程大师系列之:线程的定义和中断 interrupt
1.启动线程的三种方式:
1.1继承Thread类
public static class UseThread extends Thread {
public void run() {
System.out.println("thread run 执行!");
}
}
启动线程:
UseThread ut = new UseThread();
ut.start();
1.2实现Runable接口
public static class UseRun implements Runnable {
@Override
public void run() {
System.out.println("runable run 执行!");
}
}
启动线程
UseRun ur = new UseRun();
new Thread(ur).start();
1.3实现Callable接口
public static class UseCall implements Callable<String> {
@Override
public String call() throws Exception {
return "callable call 执行!";
}
}
启动线程
它是有返回值的,需要用FutureTask来包装,如:
UseCall uc = new UseCall();
FutureTask<String> ft = new FutureTask<>(uc);
new Tread(ft).start();
System.out.println(ft.get());
返回值需要用Future的get方法来获取。 执行结果:
callable call 执行!
2.使用interrupt来中断线程
先看一段代码
public class InterruptTest01 {
public static class m1 implements Runnable {
public void run() {
try {
System.out.println("in run() - 睡眠20秒");
Thread.sleep(20000);
System.out.println("in run() - 线程唤醒");
} catch (Exception e) {
System.out.println("in run() - 线程睡眠中被打断了");
// 如果没有return,线程不会实际被中断,它会继续打印下面的信息
System.out.println("t线程sleep中被打断后中断标志位状态:" + Thread.interrupted());
return;
}
System.out.println("in run() - 线程正常的消亡了");
}
}
public static void main(String[] args) {
m1 si = new m1();
Thread t = new Thread(si);
// 开启t线程
t.start();
System.out.println("调用interrupt方法前,t线程的中断状态:" + t.isInterrupted());
try {
// 仅仅是做标记
t.interrupt();
System.out.println("调用interrupt方法后,t线程的中断状态:" + t.isInterrupted());
// 主线程休眠2秒,从而确保刚才启动的线程有机会执行一段时间
Thread.sleep(2000);
} catch (InterruptedException e) {
System.err.println("主线程catch块出问题了");
}
System.out.println("in main() - 主线程终止了");
}
}
运行的结果:
调用interrupt方法前,t线程的中断状态:false
调用interrupt方法后,t线程的中断状态:true
in run() - 睡眠20秒
in run() - 线程睡眠中被打断了
t线程sleep中被打断后中断标志位状态:false
in main() - 主线程终止了
分析:
分析:Thread.sleep(2000)这个方法需放在interrupt方法之后,如果放在之前,会出现中断标志位全为false的情况。如果只是单纯的调用interrupt()方法,线程并没有实际被中断,会继续往下执行。
比如这样:
public class InterruptTest01 {
public static class m1 implements Runnable {
public void run() {
while (true){
System.out.println("我是不会中断的了");
}
}
}
public static void main(String[] args) {
try {
m1 si = new m1();
Thread t = new Thread(si);
// 开启t线程
t.start();
System.out.println("调用interrupt方法前,t线程的中断状态:" + t.isInterrupted());
// 仅仅是做标记
t.interrupt();
System.out.println("调用interrupt方法后,t线程的中断状态:" + t.isInterrupted());
// 主线程休眠2秒,从而确保刚才启动的线程有机会执行一段时间
Thread.sleep(2000);
} catch (InterruptedException e) {
System.err.println("主线程catch块出问题了");
}
System.out.println("in main() - 主线程终止了");
}
}
它的运行结果就是自旋,应当在run方法中加上中断标志位的判断,如下:
public static class m1 implements Runnable {
public void run() {
while (true){
if(Thread.interrupted()){
System.out.println("我竟然中断了");
return;
}
System.out.println("我是不会中断的了");
}
}
}
再次运行的结果:
调用interrupt方法前,t线程的中断状态:false
调用interrupt方法后,t线程的中断状态:true
我竟然中断了
in main() - 主线程终止了
3.待决中断
意思就是:如果线程在调用sleep()方法前被中断,那么该中断称为待决中断,它会在刚调用sleep()方法时,立即抛出InterruptedException异常。
如下:
public class PendingInterruptTest {
public static void main(String[] args) {
Thread.currentThread().interrupt();
// 获取当前时间
long startTime = System.currentTimeMillis();
try {
Thread.sleep(2000);
System.out.println("主线程没有被打断");
} catch (InterruptedException x) {
System.out.println("主线程被打断,进入catch块");
}
// 计算中间代码执行的时间
System.out.println("时间差 =" + (System.currentTimeMillis() - startTime));
}
}
运行代码的结果:
主线程被打断,进入catch块
时间差 =1
4.interrupted和isInterrupted的区别?
首先看一段代码
public class InterruptTest01 {
public static class m1 implements Runnable {
public void run() {
System.out.println("我是不会中断的了");
}
}
public static void main(String[] args) {
try {
m1 si = new m1();
Thread t = new Thread(si);
// 开启t线程
t.start();
System.out.println("调用interrupt方法前0,t线程的中断状态:" + t.interrupted());
t.interrupt();
System.out.println("调用interrupt方法后1,t线程的中断状态:" + t.interrupted());
System.out.println("调用interrupt方法后2,t线程的中断状态:" + t.interrupted());
// 主线程休眠2秒,从而确保刚才启动的线程有机会执行一段时间
Thread.sleep(2000);
} catch (InterruptedException e) {
System.err.println("主线程catch块出问题了");
}
System.out.println("in main() - 主线程终止了");
}
}
运行的结果为:
调用interrupt方法前0,t线程的中断状态:false
调用interrupt方法后1,t线程的中断状态:false
调用interrupt方法后2,t线程的中断状态:false
我是不会中断的了
in main() - 主线程终止了
分析:
结果竟然都是false,可见thread 线程并没有停止,而且调用 thread.interrupted() 结果是两个 false 表示线程一直在运行过程中。官方解释:当前线程是指运行 this.interrupted() 方法的线程 。也就是说,当前线程并不是 t,并不是因为 t 调用了 interrupted() 方法就是当前线程。当前线程一直是 main 线程,它从未中断过,所以打印结果就是两个 false。
改变一下代码:
public class InterruptTest01 {
public static class m1 implements Runnable {
public void run() {
System.out.println("我是不会中断的了");
}
}
public static void main(String[] args) {
try {
m1 si = new m1();
Thread t = new Thread(si);
// 开启t线程
t.start();
System.out.println("调用interrupt方法前0,t线程的中断状态:" + Thread.interrupted());
Thread.currentThread().interrupt();
System.out.println("调用interrupt方法后1,t线程的中断状态:" + Thread.interrupted());
System.out.println("调用interrupt方法后2,t线程的中断状态:" + Thread.interrupted());
// 主线程休眠2秒,从而确保刚才启动的线程有机会执行一段时间
Thread.sleep(2000);
} catch (InterruptedException e) {
System.err.println("主线程catch块出问题了");
}
System.out.println("in main() - 主线程终止了");
}
}
结果:
调用interrupt方法前0,t线程的中断状态:false
调用interrupt方法后1,t线程的中断状态:true
调用interrupt方法后2,t线程的中断状态:false
我是不会中断的了
in main() - 主线程终止了
分析:
从上述结果中可以看出,方法 interrupted() 的确判断出当前线程是否已经停止,但是为什么第 2 个布尔值是 false 呢?文档中说的很详细,interrupted() 方法具有清除状态的功能,所以第二次的时候返回值是 false。
isInterrupt()方法代码
public static void main(String[] args) {
try {
Thread t = Thread.currentThread();
System.out.println("调用interrupt方法前0,t线程的中断状态:" + t.isInterrupted());
t.interrupt();
System.out.println("调用interrupt方法后1,t线程的中断状态:" + t.isInterrupted());
System.out.println("调用interrupt方法后2,t线程的中断状态:" + t.isInterrupted());
// 主线程休眠2秒,从而确保刚才启动的线程有机会执行一段时间
Thread.sleep(2000);
System.out.println("还在输出吗?");
} catch (InterruptedException e) {
System.err.println("主线程catch块出问题了");
}
System.out.println("in main() - 主线程终止了");
}
结果:
调用interrupt方法前0,t线程的中断状态:false
主线程catch块出问题了
调用interrupt方法后1,t线程的中断状态:true
调用interrupt方法后2,t线程的中断状态:true
in main() - 主线程终止了
分析:
线程一旦被中断,isInterrupted()方法便会返回true,而一旦sleep()方法抛出异常,它将清空中断标志,此时isInterrupted()方法将返回false
总结以上:
总结:interrupted():测试 当前线程 是否已经是中断状态,执行后具有清除状态功能。isInterrupted():测试线程 Thread 对象 是否已经是中断状态,但不清除状态标志。
并发编程大师系列之:线程的定义和中断 interrupt的更多相关文章
- 并发编程大师系列之:Synchronized的类锁和对象锁
说到并发编程,感觉跟大多数人一样,谈之色变,说它简单把,其实很有内容,说难吧,用起来也挺容易,最近我硬着头皮,决心要把并发编程好好的搞一遍.以前,面试的时候,面试官问,并发编程会吗?嗯,接触过,就加一 ...
- 并发编程大师系列之:wait/notify/notifyAll/condition
1. wait().notify()和notifyAll()方法是本地方法,并且为final方法,无法被重写. 2. 调用某个对象的wait()方法能让当前线程阻塞,并且当前线程必须拥有此对象的mon ...
- 并发编程大师系列之:CountDownLatch和Join
业务场景描述:假设一条流水线上有三个工作者:worker1,worker2,worker3.有一个任务的完成需要他们三者协作完成,worker3可以开始这个任务的前提是worker1和worker2完 ...
- Java 并发编程——Executor框架和线程池原理
Eexecutor作为灵活且强大的异步执行框架,其支持多种不同类型的任务执行策略,提供了一种标准的方法将任务的提交过程和执行过程解耦开发,基于生产者-消费者模式,其提交任务的线程相当于生产者,执行任务 ...
- Java 并发编程——Executor框架和线程池原理
Java 并发编程系列文章 Java 并发基础——线程安全性 Java 并发编程——Callable+Future+FutureTask java 并发编程——Thread 源码重新学习 java并发 ...
- linux高级编程基础系列:线程间通信
linux高级编程基础系列:线程间通信 转载:原文地址http://blog.163.com/jimking_2010/blog/static/1716015352013102510748824/ 线 ...
- Java并发编程:如何创建线程?
Java并发编程:如何创建线程? 在前面一篇文章中已经讲述了在进程和线程的由来,今天就来讲一下在Java中如何创建线程,让线程去执行一个子任务.下面先讲述一下Java中的应用程序和进程相关的概念知识, ...
- 【转】Java并发编程:如何创建线程?
一.Java中关于应用程序和进程相关的概念 在Java中,一个应用程序对应着一个JVM实例(也有地方称为JVM进程),一般来说名字默认是java.exe或者javaw.exe(windows下可以通过 ...
- python并发编程之进程、线程、协程的调度原理(六)
进程.线程和协程的调度和运行原理总结. 系列文章 python并发编程之threading线程(一) python并发编程之multiprocessing进程(二) python并发编程之asynci ...
随机推荐
- python 爬虫实例(四)
环境: OS:Window10 python:3.7 爬取链家地产上面的数据,两个画面上的数据的爬取 效果,下面的两个网页中的数据取出来 代码 import datetime import threa ...
- python线程队列Queue-FIFO(35)
之前的文章中讲解很多关于线程间通信的知识,比如:线程互斥锁lock,线程事件event,线程条件变量condition 等等,这些都是在开发中经常使用的内容,而今天继续给大家讲解一个更重要的知识点 — ...
- 解决Ubuntu14.04不能打正确拼音--无法选择第二个拼音
这时候 我们回到桌面 按"ctrl"+"Alt"+"T",打开系统终端,在终端里面输入ibus-daemon -drx并回车 Ref: ht ...
- Kylin系列(一)—— 入门
因为平常只会使用kylin而不知其原理,故写下此篇文章.文章不是自己原创,是看过很多资料,查过很多博客,有自己的理解,觉得精华的部分的一个集合.算是自己对Kylin学习完的一个总结和概括吧 ...
- 012 Android 动画效果(补间动画) +去掉App默认自带的标题+更改应用的图标
1.介绍 补间动画开发者只需指定动画开始,以及动画结束"关键帧", 而动画变化的"中间帧"则由系统计算并补齐! 2.去掉App的标题 (1)将AndroidMa ...
- 关于 Visual Studio 的代码度量值
查看方式:Visual Studio -> Analyze -> Calculate code metrics feature 代码度量(Code Metrics)是用来测量专业标准的软件 ...
- 公钥、私钥、数字签名、数字证书、对称与非对称算法、HTTPS
作者: yoyoso https://my.oschina.net/ioslighter/blog/359207 对公钥和私钥有点稀里糊涂的,搜索了一些资料,作一些整理吧,先看这个: 加密--公钥 看 ...
- centos 6.10 oracle 19c安装
centos 7以下版本安装oracle 19c 问题较多,centos 以上版本没有任何问题.记录如下. hosts文件,否则图形界面无法启动 127.0.0.1 localhost localho ...
- PAT(B) 1029 旧键盘(Java)字符串
题目链接:1029 旧键盘 (20 point(s)) 题目描述 旧键盘上坏了几个键,于是在敲一段文字的时候,对应的字符就不会出现.现在给出应该输入的一段文字.以及实际被输入的文字,请你列出肯定坏掉的 ...
- 2019牛客多校八 H. How Many Schemes (AC自动机,树链剖分)
大意: 给定树, 每条边有一个字符集合, 给定$m$个模式串, $q$个询问$(u,v)$, 对于路径$(u,v)$中的所有边, 每条边从对应字符集合中取一个字符, 得到一个串$s$, 求$s$至少包 ...