本文记录:

1,使用ScheduledExecutorService的 scheduleAtFixedRate 方法执行周期性任务的过程,讨论了在任务周期执行过程中出现了异常,会导致周期任务失败。

2,使用普通的Thread类来执行任务,在main线程中周期性创建线程,提交任务。然后,使用UncaughtExceptionHandler来处理异常。

一,正常任务执行

负责执行任务的线程类如下:(一个计算阶乘的任务,计算5以上的阶乘,就会抛出异常)

  1. public class FactorialCalc implements Runnable {
  2.  
  3. private Integer number;
  4.  
  5. public FactorialCalc(Integer number) {
  6. this.number = number;
  7. }
  8.  
  9. public void run() {
  10.  
  11. int result = 1;
  12.  
  13. if (number == 0) {
  14. System.out.println("0!=" + "1");
  15. }
  16.  
  17. if (number > 5) {
  18. System.out.println("exception");
  19. throw new IllegalArgumentException(">5");
  20. }
  21.  
  22. for(int i = 1; i <= number; i++) {
  23. result *= i;
  24.  
  25. }
  26. System.out.println(number + "!=" + result);
  27. }
  28. }

测试的Main类如下:

  1. public class MainPeriod {
  2.  
  3. public static void main(String[] args) {
  4.  
  5. ScheduledExecutorService executorService = Executors.newScheduledThreadPool(2);
  6.  
  7. FactorialCalc task1 = new FactorialCalc(6);
  8. FactorialCalc task2 = new FactorialCalc(3);
  9.  
  10. ScheduledFuture<?> result = executorService.scheduleAtFixedRate(task1, 0, 1, TimeUnit.SECONDS);
  11. // ScheduledFuture<?> result = executorService.scheduleAtFixedRate(task2, 0, 2, TimeUnit.SECONDS);
  12.  
  13. try {
  14. TimeUnit.SECONDS.sleep(5);
  15. executorService.shutdown();
  16. } catch (InterruptedException e) {
  17. e.printStackTrace();
  18. }
  19. }
  20. }
  1. ScheduledFuture<?> result = executorService.scheduleAtFixedRate(task1, 0, 1, TimeUnit.SECONDS);

提交一个Runnable任务,延迟为0,每1秒钟执行一次。

二,线程 执行过程中出现异常

当提交 task1 时,线程在执行过程中会抛出异常。

  1. FactorialCalc task1 = new FactorialCalc(6);//计算6的阶乘,6大于5
  2.  
  3. ScheduledFuture<?> result = executorService.scheduleAtFixedRate(task1, 0, 1, TimeUnit.SECONDS);

此时,如果注释掉 //executorService.shutdown(); 则线程池不会中止,因为这是一个线程池,所有的异常都由线程池catch住了但是由于线程执行过程中抛出了异常,任务也不会周期性地执行了。参考JDK里面的scheduleAtFixedRate注释:

  1. * If any execution of the task
  2. * encounters an exception, subsequent executions are suppressed.
  3. public ScheduledFuture<?> scheduleAtFixedRate(Runnable command,
  4. long initialDelay,
  5. long period,
  6. TimeUnit unit);

三,关闭线程池 ,执行 executorService.shutdown() 语句

若线程池线程 执行任务过程中抛出了异常,但是在 主线程中 执行了executorService.shutdown() 语句,则会正常关闭 线程池。

四,总结

使用ScheduledExecutorService的 scheduleAtFixedRate 方法执行周期性任务时,如果任务一直正常执行,则任务会按设定的执行周期一直在运行(前提是,主线程内不要调用executorService.shutdown() ,比如需要 执行 永久性的周期性任务,那就不能调用 executorService.shutdown() )。

如果任务在某次执行过程中抛出了异常,则周期性任务会被中断,后续也不会再生成任务了,如果主线程 也没有 调用 executorService.shutdown() ,那线程池就不会关闭了。

五,使用Thread类执行任务,在Main线程中通过while循环周期性提交任务,使用UncaughtExceptionHandler来处理异常

  1. import java.util.Random;
  2. import java.util.concurrent.TimeUnit;
  3.  
  4. public class Main {
  5.  
  6. public static void main(String[] args) {
  7.  
  8. Random rand = new Random();
  9.  
  10. while(true){
  11. int number = rand.nextInt(10);
  12. Thread t = new Thread(new FactorialCalc(number));
  13. t.setUncaughtExceptionHandler(new ExceptionHandler());
  14. t.start();
  15. try{
  16. System.out.println("sleep 4s for next task");
  17. TimeUnit.SECONDS.sleep(4);
  18. }catch(InterruptedException e){
  19.  
  20. }
  21. }
  22. }
  23. }

在main方法中使用一个while(true)循环,周期性地创建线程 提交任务。

第12-13行,每创建一个线程,调用setUncaughtExceptionHandler方法设置异常处理。关于异常处理,可参考:JAVA线程未捕获异常处理

第15-18行,线程每隔4s提交一次任务,从而实现任务的周期性执行。

异常处理类ExceptionHandler类实现了UncaughtExceptionHandler接口,然后在uncaughtException方法里面定义具体的异常处理过程即可。

  1. import java.lang.Thread.UncaughtExceptionHandler;
  2.  
  3. public class ExceptionHandler implements UncaughtExceptionHandler{
  4.  
  5. @Override
  6. public void uncaughtException(Thread t, Throwable e) {
  7. System.out.println("illegal exception: 计算的阶乘大于5了," + e.getMessage());
  8. }
  9. }

与线程池方式相比 ,这种方式是每个周期,都要new一个线程。而线程池则是每个周期new一个任务,把任务提交给线程池即可。

原文:http://www.cnblogs.com/hapjin/p/7616068.html

JAVA线程池ScheduledExecutorService周期性地执行任务 与单个Thread周期性执行任务的异常处理的更多相关文章

  1. 让Java线程池实现任务阻塞执行的一种可行方案

    Java的线程池一般是基于concurrent包下的ThreadPoolExecutor类实现的, 不过当我们基于spring框架开发程序时, 通常会使用其包装类ThreadPoolTaskExecu ...

  2. java 线程池 ScheduledExecutorService

    ScheduledExecutorService执行周期性或定时任务 https://blog.csdn.net/u013851082/article/details/70207640

  3. Java线程池实现原理及其在美团业务中的实践

    本文转载自Java线程池实现原理及其在美团业务中的实践 导语 随着计算机行业的飞速发展,摩尔定律逐渐失效,多核CPU成为主流.使用多线程并行计算逐渐成为开发人员提升服务器性能的基本武器.J.U.C提供 ...

  4. 解决Java线程池任务执行完毕后线程回收问题

      转载请注明出处:http://www.cnblogs.com/pengineer/p/5011965.html         对于经常使用第三方框架进行web开发的程序员来说,Java线程池理所 ...

  5. 捕获Java线程池执行任务抛出的异常

    捕获Java线程池执行任务抛出的异常Java中线程执行的任务接口java.lang.Runnable 要求不抛出Checked异常, public interface Runnable { publi ...

  6. java 线程池 并行 执行

    https://github.com/donaldlee2008/JerryMultiThread/blob/master/src/com/jerry/threadpool/ThreadPoolTes ...

  7. Java如何判断线程池所有任务是否执行完毕

    import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class Tes ...

  8. 由浅入深理解Java线程池及线程池的如何使用

    前言 多线程的异步执行方式,虽然能够最大限度发挥多核计算机的计算能力,但是如果不加控制,反而会对系统造成负担.线程本身也要占用内存空间,大量的线程会占用内存资源并且可能会导致Out of Memory ...

  9. 【java线程池】

    一.概述 1.线程池的优点 ①降低系统资源消耗,通过重用已存在的线程,降低线程创建和销毁造成的消耗: ②提高系统响应速度,当有任务到达时,无需等待新线程的创建便能立即执行: ③方便线程并发数的管控,线 ...

随机推荐

  1. 【BZOJ2242】计算器(BSGS,快速幂)

    [BZOJ2242]计算器(BSGS,快速幂) 题面 BZOJ 洛谷 1.给定y.z.p,计算y^z mod p 的值: 2.给定y.z.p,计算满足xy ≡z(mod p)的最小非负整数x: 3.给 ...

  2. 【mysql】mysql尾部空格

    mysql 字段为varchar类型的在查询时候胡忽略尾部空格. 先看表结构 插入一条数据包含空格 在查询是可以查到的 所有在插入数据的时候要对插入字段的数据处理下,php可以用函数trim()去掉两 ...

  3. Codeforces1096G Lucky Tickets(NTT优化dp)

    设\(f[i][j]\)表示填了\(i\)个数,数位和为\(j\)的方案数 于是方程为: \[f[i][j]=\sum_{k=0}^9 f[i-1][j-k]*[CanUse[k]==1]\] 其中\ ...

  4. CANOE入门(一)

    CANoe是Vector公司的针对汽车电子行业的总线分析工具,现在我用CANoe7.6版本进行介绍,其他版本功能基本差不多. 硬件我使用的是CAN case XL. 1,CANoe软件的安装很简单,先 ...

  5. shell中的source和直接执行sh的区别

    首先我们知道我们执行shell有这么几种方法 1. sh/bash使用其内置的命令集来执行一些命令,例如如下 sh demo.sh bash demo.sh 2. 使用./或者/$SHELLPATH/ ...

  6. Ubuntu16.04下的NetCore环境搭建(附录含Ubuntu 18.04 安装 NetCore2.1)

    跨平台系列汇总:http://www.cnblogs.com/dunitian/p/4822808.html#linux VSCode安装:http://www.cnblogs.com/dunitia ...

  7. nuxt.js实战之开发环境配置

    一.创建项目 1.使用如下命令生成项目 vue init nuxt-community/starter-template testPro --testPro为项目名称 2.进入到项目根目录下,使用np ...

  8. C#项目获取当前时间的农历时间

    https://blog.csdn.net/cstester/article/details/7407044 using System.Globalization; class CnCanlendar ...

  9. Day29--Python--缓冲区, 粘包

    tcp: 属于长连接,与一个客户端进行连接了以后,其他的客户端要等待.要想连接另外一个客户端,需要优雅地断开当前客户端的连接 允许地址重用:server.setsockopt(socket.SOL_S ...

  10. C语言进阶--Day2

    今天主要讲解的是函数的压栈与出栈 1. 要实现一个数组的逆置,用栈的压栈出栈观点: reverseArr(int *parr,int i,int len) { if(i != len-1) rever ...