JAVA并发编程(四)任务的取消与关闭

概述

之前三篇文章介绍了多线程的同步,加锁机制,同步容器,并发容器,以及多线程工具类,线程池等。

详细内容请看 线程池  内置锁与多线程数据共享 线程安全工具类 

大多数时候,我们会让线程运行直到结束。然而,有时候我们希望提前结束任务,或者因为用户取消了操作,需要终止线程。我们来看一下如何使一个线程安全、快速、可靠的停下来。

任务取消的原因

1、用户主动取消。如点击图形界面的取消按钮等。

2、有时间限制的操作。当超过时间限制时,需要结束任务。

3、应用程序事件。当一个任务找到解决方案时,需要结束操作。

4、出现错误。当任务执行过程中出现不可恢复的错误,需要结束任务。

5、关闭。当一个程序被关闭时,需要做一些清理操作。

使用简单标志位保存取消状态

为了保证标志位可靠,标志必须为volatile类型。
/**
 * 
 * @author chao
 *
 */
public class SafeCancel extends Thread {
	private volatile boolean cancelled;

	@Override
	public void run() {
		while (!cancelled) {
			// todo something
		}
	}

	public void cancel() {
		cancelled = true;
	}
}
一个可取消的任务必须拥有取消策略。这个策略需要详细定义如何取消改任务,何时检查是否已经取消以及在响应取消请求之后该执行哪些操作。

中断

使用标志位来取消任务是不及时的,如果中间调用了阻塞方法,有可能永远都无法结束。
/**
 * 
 * @author chao
 *
 */
class InterruptCancel extends Thread {
	BlockingQueue<Object> queue = new LinkedBlockingQueue<>();

	@Override
	public void run() {
		try {
			while (!isInterrupted()) {
				Object info = new Object();
				// do something
				queue.put(info);
			}
		} catch (InterruptedException e) {
		}
	}

	public void cancel() {
		interrupt();
	}
}
每个线程都有一个boolean类型的中断状态,interrupt方法能中断目标线程,而isInterrupted方法能返回目标线程的中断状态,静态的interrupted方法将清除当前线程的中断状态,也是清除中断状态的唯一一个方法。

中断响应

当调用可中断的阻塞方法时,有两种策略可用于处理InterruptedException
1、传递异常,从而使你的方法也成为可中断的阻塞方法。
2、恢复中断状态,从而使调用栈上中的上层代码能够对其进行处理。
只有实现了线程中断策略的代码才可以屏蔽中断请求,常规任务中不应该屏蔽中断请求。

通过Future实现取消

private static ExecutorService service = Executors.newCachedThreadPool();

	public static void timedRun(Runnable r, long timeout, TimeUnit unit) {
		Future<?> task = service.submit(r);
		try {
			task.get(timeout, unit);
		} catch (InterruptedException | ExecutionException | TimeoutException e) {

		} finally {
			task.cancel(true);
		}
	}

关闭ExecutorService

ExecutorService提供了两种关闭方法:使用shutdown正常关闭,使用shutdownNow强行关闭。在进行强行关闭时,首先关闭当前正在进行的任务,然后返回所有尚未启动的任务清单。

处理未捕获的异常

Thread API提供了UncaughtExceptionHandler,检测出某个线程由于未捕获异常而终结的情况。在Android开发中,可以用来上报异常,发现问题及时修复。
Thread的方法
  public static void setDefaultUncaughtExceptionHandler(UncaughtExceptionHandler paramUncaughtExceptionHandler)
  {
    SecurityManager localSecurityManager = System.getSecurityManager();
    if (localSecurityManager != null) {
      localSecurityManager.checkPermission(new RuntimePermission("setDefaultUncaughtExceptionHandler"));
    }
    defaultUncaughtExceptionHandler = paramUncaughtExceptionHandler;
  }
  
  
  public static abstract interface UncaughtExceptionHandler
  {
    public abstract void uncaughtException(Thread paramThread, Throwable paramThrowable);
  }
  

Android中提供了另一个方法,单独设置每个线程的未捕获异常处理,更加灵活。
 public void setUncaughtExceptionHandler(UncaughtExceptionHandler handler) {
        uncaughtHandler = handler;
    }

JVM关闭钩子

在线上Java程序中经常遇到进程程挂掉,一些状态没有正确的保存下来,这时候就需要在JVM关掉的时候执行一些清理现场的代码。Java中得ShutdownHook提供了比较好的方案。
  JDK在1.3之后提供了Java Runtime.addShutdownHook(Thread hook)方法,可以注册一个JVM关闭的钩子,这个钩子可以在以下几种场景被调用:
1)程序正常退出
2)使用System.exit()
3)终端使用Ctrl+C触发的中断
4)系统关闭
5)使用Kill pid命令干掉进程




欢迎扫描二维码,关注公众号


  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值