Java中 Timer缺陷,用ScheduledExecutorService替代

本文讨论了Java中的Timer类在处理任务调度时的问题,如任务执行顺序受阻,以及TimerTask抛出异常时可能导致所有任务停止。随后介绍了ScheduledThreadPoolExecutor,它能更好地管理并发任务并处理异常,确保预期的执行顺序。
  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开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

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

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

如何快速更新自己的技术积累?

  • 在现有的项目里,深挖技术,比如用到netty可以把相关底层代码和要点都看起来。
  • 如果不知道目前的努力方向,就看自己的领导或公司里技术强的人在学什么。
  • 知道努力方向后不知道该怎么学,就到处去找相关资料然后练习。
  • 学习以后不知道有没有学成,则可以通过面试去检验。

我个人觉得面试也像是一场全新的征程,失败和胜利都是平常之事。所以,劝各位不要因为面试失败而灰心、丧失斗志。也不要因为面试通过而沾沾自喜,等待你的将是更美好的未来,继续加油!

以上面试专题的答小编案整理成面试文档了,文档里有答案详解,以及其他一些大厂面试题目

八年CRUD,疫情备战三个月,三面头条、四面阿里拿offer面经分享

八年CRUD,疫情备战三个月,三面头条、四面阿里拿offer面经分享

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
,就到处去找相关资料然后练习。

  • 学习以后不知道有没有学成,则可以通过面试去检验。

我个人觉得面试也像是一场全新的征程,失败和胜利都是平常之事。所以,劝各位不要因为面试失败而灰心、丧失斗志。也不要因为面试通过而沾沾自喜,等待你的将是更美好的未来,继续加油!

以上面试专题的答小编案整理成面试文档了,文档里有答案详解,以及其他一些大厂面试题目

[外链图片转存中…(img-YCl4ngOj-1713285616199)]

[外链图片转存中…(img-PoAurFXY-1713285616199)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值