`
coolxing
  • 浏览: 874813 次
  • 性别: Icon_minigender_1
  • 来自: 北京
博客专栏
9a45b66b-c585-3a35-8680-2e466b75e3f8
Java Concurre...
浏览量:97530
社区版块
存档分类
最新评论

处理InterruptedException异常--JCIP7.1读书笔记

阅读更多

[本文是我对Java Concurrency In Practice 7.1的归纳和总结.  转载请注明作者和出处,  如有谬误, 欢迎在评论中指正. ]

 

在java的中断机制中, InterruptedException异常占据重要的位置. 处理InterruptedException异常的方式有:

 

1. 不catch直接向上层抛出, 或者catch住做一些清理工作之后重抛该异常. 这样的处理使得你的方法也成为一个可中断的阻塞方法:

// 直接向上层抛出InterruptedException, dosomething方法也是一个可中断的阻塞方法
private void dosomething() throws InterruptedException {
	Thread.sleep(1000);
}
  

2. 有时不能向上抛出InterruptedException异常(例如父类的相应方法没有声明该异常), 此时catch之后, 必须设置当前线程的中断标记为true, 以表明当前线程发生了中断, 以便调用栈上层进行处理:

public class InterruptedExceptionHandler implements Runnable {
	private Object lock = new Object();

	@Override
	public void run() {
		while (!Thread.currentThread().isInterrupted()) {
			dosomething();
		}
	}

	private void dosomething() {
		try {
			// Object.wait是一个可中断的阻塞方法, 如果在其阻塞期间检查到当前线程的中断标记为true, 会重置中断标记后从阻塞状态返回, 并抛出InterruptedException异常
			synchronized (lock) {
				lock.wait();
			}
		} catch (InterruptedException e) {
			System.out.println("InterruptedException happened");
			// catch住InterruptedException后设置当前线程的中断标记为true, 以供调用栈上层进行相应的处理
			// 在此例中, dosomething方法的调用栈上层是run方法.
			Thread.currentThread().interrupt();
		}
	}
	
	public static void main(String[] args) throws InterruptedException {
		Thread t = new Thread(new InterruptedExceptionHandler());
		t.start();
		// 启动线程1s后设置其中断标记为true
		Thread.sleep(1000);
		t.interrupt();
	}
} 

主线程启动InterruptedExceptionHandler线程1s后, 设置InterruptedExceptionHandler线程的中断标记为true. 此时InterruptedExceptionHandler线程应该阻塞在wait方法上, 由于wait方法是可中断的阻塞方法, 所以其检查到中断标记为true时, 将重置当前线程的中断标记后抛出InterruptedException, dosomething方法catch住InterruptedException异常后, 再次将当前线程的中断标记设置为true, run方法检查到中断标记为true, 循环不再继续. 假如dosomething方法catch住InterruptedException异常后没有设置中断标记, 其调用栈上层的run方法就无法得知线程曾经发生过中断, 循环也就无法终止.

 

3. 还有一种情形比较特殊: 我们希望发生了InterruptedException异常后仍然继续循环执行某阻塞方法, 此时应该将中断状态保存下来, 当循环完成后再根据保存下来的中断状态执行相应的操作:

public class InterruptedExceptionContinueHandler implements Runnable {
	private BlockingQueue<Integer> queue;

	public InterruptedExceptionContinueHandler(BlockingQueue<Integer> queue) {
		this.queue = queue;
	}

	@Override
	public void run() {
		while (!Thread.currentThread().isInterrupted()) {
			dosomething();
		}
		System.out.println(queue.size());
	}

	private void dosomething() {
		// cancelled变量用于表明线程是否发生过中断
		boolean cancelled = false;
		for (int i = 0; i < 10000; i++) {
			try {
				queue.put(i);
			} catch (InterruptedException e) {
				// 就算发生了InterruptedException, 循环也希望继续运行下去, 此时将cancelled设置为true, 以表明遍历过程中发生了中断
				System.out.println("InterruptedException happened when i = " + i);
				cancelled = true;
			}
		}
		// 如果当前线程曾经发生过中断, 就将其中断标记设置为true, 以通知dosomething方法的上层调用栈
		if (cancelled) {
			Thread.currentThread().interrupt();
		}
	}
	
	public static void main(String[] args) throws InterruptedException {
		Thread t = new Thread(new InterruptedExceptionContinueHandler(new LinkedBlockingQueue<Integer>()));
		t.start();
		
		// 启动线程2ms后设置其中断标记为true
		Thread.sleep(2);
		t.interrupt();
	}
}

在我的机器中, 输出结果如下:

InterruptedException happened when i = 936

size = 9999

队列的size是9999而不是10000, 是因为i = 936时发生了InterruptedException异常, 该次put没有成功.

为什么不在发生InterruptedException时就设置当前线程的中断标记, 而非要绕一圈? 假设将dosomething方法改为:

private void dosomething() {
	for (int i = 0; i < 10000; i++) {
		try {
			queue.put(i);
		} catch (InterruptedException e) {
			System.out.println("InterruptedException happened when i = " + i);
			Thread.currentThread().interrupt();
		}
	}
}

运行后发现结果类似为:

InterruptedException happened when i = 936

InterruptedException happened when i = 937

...

InterruptedException happened when i = 9998

InterruptedException happened when i = 9999

size = 936

catch住InterruptedException后立即将当前线程的中断标记设置为true, 就会导致put方法又抛出InterruptedException异常, 如此往复直到循环结束.

 

4. 最不可取的是catch了InterruptedException异常但是不做任何处理, 这样一来调用栈上层就无法得知当前线程是否发生过中断. 只有一种情况下可以这样处理: 当InterruptedException发生在调用栈的最上层, 如run方法, 或者main方法中, 且后续代码不检查中断状态时:

public static void main(String[] args) {
	// main方法已经是调用栈的最上层, 此时可以catchInterruptedException后不做任何处理
	try {
		Thread.sleep(2);
	} catch (InterruptedException e) {
		e.printStackTrace();
	}
}
1
0
分享到:
评论

相关推荐

    Java处理InterruptedException异常的理论与实践

    在Java编程中,`InterruptedException`是一个特殊的异常类型,主要用于处理线程中断的情况。当你在代码中调用诸如`Thread.sleep()`, `Thread.join()`, 或 `Object.wait()`等阻塞方法时,如果线程被中断,这些方法会...

    处理 InterruptedException1

    InterruptedException 是一种检查异常(checked exception),不能被忽略,因为它提供了一个机会来正确地处理中断请求。 中断机制是 Java 语言中的一种协作机制,它允许一个线程请求另一个线程停止正在做的事情。当...

    Java第二十九——三十一讲总结-异常处理.doc

    ### Java异常处理详解 #### 一、异常的概念与分类 在Java编程中,**异常**是一种用来处理程序运行过程中可能出现的错误或者不寻常情况的机制。异常不仅可以帮助开发者捕获和处理错误,还可以用于实现一些特殊的...

    JAVA InterruptedException

    1. **重新抛出异常**:如果你的代码不负责处理中断,可以选择将`InterruptedException`重新抛出,让上层调用者处理。 2. **捕获并处理**:如果你需要做一些清理工作,比如关闭文件流或释放资源,可以在`catch`块中...

    Java笔记高级部分20190819.pdf

    `catch`块用于捕获和处理异常,可以有一个或多个`catch`块。`finally`块则是无论是否发生异常都会执行的代码块,通常用于资源释放。 ```java try { // 可能抛出异常的代码 } catch (ExceptionType1 e1) { // ...

    Java异常类型-Java教程共1页.pdf.zip

    Java异常处理是编程中至关重要的一个环节,它确保了程序在遇到错误时能够优雅地运行,而不是突然崩溃。本教程将深入探讨Java异常的类型及其处理机制。 在Java中,异常是程序执行过程中发生的错误,可以是逻辑错误、...

    java异常处理

    `try`块用于包含可能抛出异常的代码,`catch`块捕获并处理异常,`finally`块确保某些代码(如资源清理)总是在程序退出前执行。 - **多 catch**:一个`try`块后面可以跟多个`catch`块,分别处理不同类型的异常。 -...

    java常见异常总结

    本文将详细介绍一些常见的Java异常及其发生场景,帮助读者更好地理解和处理这些异常。 #### 1. `java.lang.NullPointerException` (空指针异常) - **定义**:当程序试图访问一个尚未初始化的对象,或者尝试访问一...

    推荐课程 完整版优质java教程 java精品教学课件 Java语言程序设计 第9章 异常处理(共26页).ppt

    Java异常处理是编程中至关重要的一个环节,它确保了程序的稳定性和健壮性。Java异常处理机制通过异常类来实现,将程序运行时可能出现的错误情况转化为异常对象,从而提供了一种统一处理错误的方式。 Java中的异常...

    java 多线程异常处理

    Java多线程异常处理是Java编程中不可或缺的一部分,特别是在并发编程场景下,正确处理异常能够保证程序的稳定性和健壮性。本实验报告主要涵盖了Java异常处理机制、多线程概念与实现,以及多线程同步问题。 首先,...

    Java异常处理与输入输出.ppt

    try 语句块中可能会抛出异常,而 catch 语句块中则用于捕获和处理异常。 在 Java 中,输入输出是通过 IO 流来实现的。IO 流是 Java 中的一种机制,用于实现输入输出操作。IO 流可以分为两种:输入流和输出流。输入...

    InterruptedException和interrup

    这个异常通常在处理线程中断时抛出,当一个线程正在执行时被中断,如果该线程正在阻塞(例如等待I/O或者在`wait()`, `sleep()`, `join()`等方法上挂起),那么它会抛出`InterruptedException`。理解并正确处理这个...

    多线程异常处理.pdf

    在探讨多线程异常处理之前,首先要了解Java中的线程模型以及异常处理的基本概念。Java的多线程模型允许程序同时执行多个线程,以实现更高效的任务处理。每个线程都是独立执行其任务的执行路径,而异常处理是保证程序...

    黑莓学习笔记之一----进度条

    在后台线程中处理任务,并定期更新进度条的值,确保UI的响应性: ```java new Thread() { public void run() { while (true) { // 模拟任务进度 currentProgress++; UiApplication.getUiApplication()....

    Java常见异常和错误总结

    - 处理异常以优雅地退出阻塞状态。 以上总结了Java编程中常见的异常和错误类型,以及如何识别和处理这些问题。对于开发者而言,理解这些异常的含义及其解决方法至关重要,这不仅能帮助提高代码质量,还能增强程序...

    实验5异常处理.pdf

    `catch`块用于捕获并处理异常,可以有多个`catch`块,每个块对应一种类型的异常。`finally`块总是会被执行,无论是否发生异常,通常用于清理资源,如关闭文件或网络连接。 5.2.2 常见系统异常 Java提供了一系列预...

Global site tag (gtag.js) - Google Analytics