Java中 Timer缺陷,用ScheduledExecutorService替代

  1. try

  2. {

  3. Thread.sleep(3000);

  4. } catch (InterruptedException e)

  5. {

  6. e.printStackTrace();

  7. }

  8. }

  9. };

  10. TimerTask task2 = new TimerTask()

  11. {

  12. @Override

  13. public void run()

  14. {

  15. System.out.println("task2 invoked ! "

  16. + (System.currentTimeMillis() - start));

  17. }

  18. };

  19. Timer timer = new Timer();

  20. start = System.currentTimeMillis();

  21. timer.schedule(task1, 1000);

  22. timer.schedule(task2, 3000);

  23. }

  24. }

定义了两个任务,预计是第一个任务1s后执行,第二个任务3s后执行,但是看运行结果:

[java]  view plain copy

  1. task1 invoked ! 1000

  2. task2 invoked ! 4000

task2实际上是4后才执行,正因为Timer内部是一个线程,而任务1所需的时间超过了两个任务间的间隔导致。下面使用ScheduledThreadPool解决这个问题:

[java]  view plain copy

  1. package com.zhy.concurrency.timer;

  2. import java.util.TimerTask;

  3. import java.util.concurrent.Executors;

  4. import java.util.concurrent.ScheduledExecutorService;

  5. import java.util.concurrent.TimeUnit;

  6. public class ScheduledThreadPoolExecutorTest

  7. {

  8. private static long start;

  9. public static void main(String[] args)

  10. {

  11. /**

  12. * 使用工厂方法初始化一个ScheduledThreadPool

  13. */

  14. ScheduledExecutorService newScheduledThreadPool = Executors

  15. .newScheduledThreadPool(2);

  16. TimerTask task1 = new TimerTask()

  17. {

  18. @Override

  19. public void run()

  20. {

  21. try

  22. {

  23. System.out.println("task1 invoked ! "

  24. + (System.currentTimeMillis() - start));

  25. Thread.sleep(3000);

  26. } catch (Exception e)

  27. {

  28. e.printStackTrace();

  29. }

  30. }

  31. };

  32. TimerTask task2 = new TimerTask()

  33. {

  34. @Override

  35. public void run()

  36. {

  37. System.out.println("task2 invoked ! "

  38. + (System.currentTimeMillis() - start));

  39. }

  40. };

  41. start = System.currentTimeMillis();

  42. newScheduledThreadPool.schedule(task1, 1000, TimeUnit.MILLISECONDS);

  43. newScheduledThreadPool.schedule(task2, 3000, TimeUnit.MILLISECONDS);

  44. }

  45. }

输出结果:

[java]  view plain copy

  1. task1 invoked ! 1001

  2. task2 invoked ! 3001

符合我们的预期结果。因为ScheduledThreadPool内部是个线程池,所以可以支持多个任务并发执行。

2、Timer当任务抛出异常时的缺陷

如果TimerTask抛出RuntimeException,Timer会停止所有任务的运行:

[java]  view plain copy

  1. package com.zhy.concurrency.timer;

  2. import java.util.Date;

  3. import java.util.Timer;

  4. import java.util.TimerTask;

  5. public class ScheduledThreadPoolDemo01

  6. {

  7. public static void main(String[] args) throws InterruptedException

  8. {

  9. final TimerTask task1 = new TimerTask()

  10. {

  11. @Override

  12. public void run()

  13. {

  14. throw new RuntimeException();

  15. }

  16. };

  17. final TimerTask task2 = new TimerTask()

  18. {

  19. @Override

  20. public void run()

  21. {

  22. System.out.println(“task2 invoked!”);

  23. }

  24. };

  25. Timer timer = new Timer();

  26. timer.schedule(task1, 100);

  27. timer.scheduleAtFixedRate(task2, new Date(), 1000);

  28. }

  29. }

上面有两个任务,任务1抛出一个运行时的异常,任务2周期性的执行某个操作,输出结果:

[java]  view plain copy

  1. task2 invoked!

  2. Exception in thread “Timer-0” java.lang.RuntimeException

  3. at com.zhy.concurrency.timer.ScheduledThreadPoolDemo01$1.run(ScheduledThreadPoolDemo01.java:24)

  4. at java.util.TimerThread.mainLoop(Timer.java:512)

  5. at java.util.TimerThread.run(Timer.java:462)

由于任务1的一次,任务2也停止运行了。。。下面使用ScheduledExecutorService解决这个问题:

[java]  view plain copy

  1. package com.zhy.concurrency.timer;

  2. import java.util.Date;

  3. import java.util.Timer;

  4. import java.util.TimerTask;

  5. import java.util.concurrent.Executors;

  6. import java.util.concurrent.ScheduledExecutorService;

  7. import java.util.concurrent.TimeUnit;

  8. public class ScheduledThreadPoolDemo01

  9. {

  10. public static void main(String[] args) throws InterruptedException

  11. {

  12. final TimerTask task1 = new TimerTask()

  13. {

  14. @Override

  15. public void run()

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
I-1715718220524)]

[外链图片转存中…(img-lJZzUqQF-1715718220525)]

[外链图片转存中…(img-o8NsQmHZ-1715718220525)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

  • 21
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值