java 多线程编程中interrupt(),interrupted()和isInterrupted()用法

最近,笔者在学习高洪岩老师的《java多线程编程核心技术》一书时,在1.7停止线程这一小节时,敲击p27的代码时得到的Run3类的运行结果和书上总是不一致。

System.out.println("是否停止1? ="+ThreadInterrupt.interrupted());得到的一直是false。并且在请群里的网友和同事运行时也是得到false。但是看网上的博文,大家得到的结果都是true,于是很纳闷。

在今天查阅资料和动手调试之后才发现问题所在。故将代码粘贴于下:

ThreadInterrupt类:

package com.wang.thread;

import org.apache.log4j.Logger;

public class ThreadInterrupt extends Thread{
	private static Logger logger = Logger.getLogger(ThreadInterrupt.class);
	
	@Override
	public void run(){
		super.run();
		 for(int i =1; i<=500000; i++){
			 System.out.println("i= "+i);
			 if(i==100){
				 this.interrupt();//此线程interrupt标记位设置为true
				 logger.info("ThreadInterrupt线程中段位标记--------->"+this.isInterrupted());
				 System.out.println("ThreadInterrupt线程中段位标记--------->"+this.isInterrupted());
			 }
		 }
	}

}

测试类:

package com.wang.thread;

import org.apache.log4j.Logger;

public class TestInterruptedTest {
	
	private static Logger logger = Logger.getLogger(TestInterruptedTest.class);

	public static void main(String[] args) {
		// TODO 自动生成的方法存根
		
		try {
			ThreadInterrupt thread6 = new ThreadInterrupt();
			thread6.start();
			System.out.println(thread6.isAlive());
			//sleep是让"当前正在执行的线程"休眠,
			//也就是Thread.currentThread()返回的线程 在这里是main线程
			ThreadInterrupt.sleep(1000);
			//在主线程休眠的这1s内,thread6可能已经执行完了。
			System.out.println(thread6.isAlive());
			//thread6.interrupt();
			//当前线程是main 
			System.out.println("当前线程是:"+Thread.currentThread().getName());
			logger.info("当前线程是:"+Thread.currentThread().getName());
			//这个为什么是false
			System.out.println("是否停止3? ="+ thread6.isInterrupted());
			logger.info("是否停止3? ="+ thread6.isInterrupted());
			//虽然是Thread5.interrupted(),但是interrupted()是当前线程的状态,与谁来调用它没有关系
			System.out.println("是否停止1? ="+ThreadInterrupt.interrupted());
			logger.info("是否停止1? ="+ThreadInterrupt.interrupted());
			System.out.println("是否停止2? ="+ThreadInterrupt.interrupted());
			logger.info("是否停止2? ="+ThreadInterrupt.interrupted());
		} catch (InterruptedException e) {
			// TODO 自动生成的 catch 块
			System.out.println("main catch");
			e.printStackTrace();
		}

	}

}

输出的log.log内容如下:

2018-02-12 13:51:28  [ Thread-0:1 ] - [ INFO ]  ThreadInterrupt线程中段位标记--------->true
2018-02-12 13:51:29  [ main:1045 ] - [ INFO ]  当前线程是:main
2018-02-12 13:51:29  [ main:1046 ] - [ INFO ]  是否停止3? =true
2018-02-12 13:51:29  [ main:1047 ] - [ INFO ]  是否停止1? =false

2018-02-12 13:51:29  [ main:1048 ] - [ INFO ]  是否停止2? =false


这下结果就对上了。

sleep(Long long) 是休眠 “当前正在运行的”线程,当前运行的线程也就是Thread.currentThread()返回的方法,在这里是main主方法,虽然是ThreadInterrupt类调用的,但是休眠的是main线程。

在main线程休眠1s之后,让出cpu等资源后,原来的ThreadInterrupt类对象的run()方法可能早就运行完了。所以在这里我把循环的范围从50000改为500000。如果你的电脑运行更快,请把值改的再大点,确保1s内运行不完就行了。

ThreadInterrupt的run()执行完后线程就结束了,所以再用interrupt()方法设置中断位就没有任何效果了。

所以用isInterrupted()再去检查就会一直得到false。

所以 在 我把i的范围改为500000之后,在i等于100时就用interrupt()方法把中断标记置为false,在main方法中再去用isInterrupted()检查就得到true了。

但是如果是用innterrupted()方法检查,会返回ture,但是第二次再去检查时就会变成false。因为innterrupted()会在返回中段位后同时清除原来的中断位标记(恢复为默认的false)。而isInterrupted()则只检查不改变。这一部分的内容其他博文里讲的很多了。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值