Java中 Timer缺陷,用ScheduledExecutorService替代(1)

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门,即可获取!
6. public class TimerTest

  1. {

  2. private static long start;

  3. public static void main(String[] args) throws Exception

  4. {

  5. TimerTask task1 = new TimerTask()

  6. {

  7. @Override

  8. public void run()

  9. {

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

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

  12. try

  13. {

  14. Thread.sleep(3000);

  15. } catch (InterruptedException e)

  16. {

  17. e.printStackTrace();

  18. }

  19. }

  20. };

  21. TimerTask task2 = new TimerTask()

  22. {

  23. @Override

  24. public void run()

  25. {

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

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

  28. }

  29. };

  30. Timer timer = new Timer();

  31. start = System.currentTimeMillis();

  32. timer.schedule(task1, 1000);

  33. timer.schedule(task2, 3000);

  34. }

  35. }

定义了两个任务,预计是第一个任务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;

最近我根据上述的技术体系图搜集了几十套腾讯、头条、阿里、美团等公司21年的面试题,把技术点整理成了视频(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门,即可获取!

  1. import java.util.Timer;

  2. import java.util.TimerTask;

[外链图片转存中…(img-i9F5XR9t-1714676429919)]

最近我根据上述的技术体系图搜集了几十套腾讯、头条、阿里、美团等公司21年的面试题,把技术点整理成了视频(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分

[外链图片转存中…(img-GxZPAhKG-1714676429920)]

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门,即可获取!

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值