5. 中断的使用
通常,中断的使用场景有以下几个:
- 点击某个桌面应用中的取消按钮时;
- 某个操作超过了一定的执行时间限制需要中止时;
- 多个线程做相同的事情,只要一个线程成功其它线程都可以取消时;
- 一组线程中的一个或多个出现错误导致整组都无法继续时;
- 当一个应用或服务需要停止时。
下面来看一个具体的例子。这个例子里,本打算采用 GUI 形式,但考虑到 GUI 代码会使程序复杂化,就使用控制台来模拟下核心的逻辑。这里新建了一个磁盘文件扫描的任务,扫描某个目录下的所有文件并将文件路径打印到控制台,扫描的过程可能会很长。若需要中止该任务,只需在控制台键入 quit 并回车即可。
package com.ticmy.interrupt; import java.io.BufferedReader; import java.io.File; import java.io.InputStreamReader; public class FileScanner { private static void listFile(File f) throws InterruptedException { if(f == null) { throw new IllegalArgumentException(); } if(f.isFile()) { System.out.println(f); return; } File[] allFiles = f.listFiles(); if(Thread.interrupted()) { throw new InterruptedException("文件扫描任务被中断"); } for(File file : allFiles) { // 还可以将中断检测放到这里 listFile(file); } } public static String readFromConsole() { BufferedReader reader = new BufferedReader(new InputStreamReader(System.in)); try { return reader.readLine(); } catch (Exception e) { e.printStackTrace(); return ""; } } public static void main(String[] args) throws Exception { final Thread fileIteratorThread = new Thread() { public void run() { try { listFile(new File("c:\\")); } catch (InterruptedException e) { e.printStackTrace(); } } }; new Thread() { public void run() { while(true) { if("quit".equalsIgnoreCase(readFromConsole())) { if(fileIteratorThread.isAlive()) { fileIteratorThread.interrupt(); return; } } else { System.out.println("输入 quit 退出文件扫描"); } } } }.start(); fileIteratorThread.start(); } }
在扫描文件的过程中,对于中断的检测这里采用的策略是,如果碰到的是文件就不检测中断,是目录才检测中断,因为文件可能是非常多的,每次遇到文件都检测一次会降低程序执行效率。此外,在 fileIteratorThread 线程中,仅是捕获了 InterruptedException,没有重设中断状态也没有继续抛出异常,因为我非常清楚它的使用环境,run 方法的调用栈上层已经没有可能需要检测中断状态的方法了。
在这个程序中,输入 quit 完全可以执行 System.exit(0) 操作来退出程序,但正如前面提到的,这是个 GUI 程序核心逻辑的模拟,在 GUI 中,执行 System.exit(0) 会使得整个程序退出。
6. 参考资料
- 《Java Concurrency in Practice》
- 《Concurrent Programming in Java Design principles and patterns》
- http://docs.oracle.com/javase/1.4.2/docs/guide/misc/threadPrimitiveDeprecation.html