Java中 Timer缺陷,用ScheduledExecutorService替代

本文对比了Java中的Timer与ScheduledThreadPool在执行任务时的区别,特别强调了Timer的线程单线程特性可能导致的任务执行顺序问题,以及如何使用ScheduledThreadPool避免此类问题。同时提到了面试准备中的策略,包括刷题和掌握常见知识点如MySQL、Redis等。
摘要由CSDN通过智能技术生成

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

  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()

总结

谈到面试,其实说白了就是刷题刷题刷题,天天作死的刷。。。。。

为了准备这个“金三银四”的春招,狂刷一个月的题,狂补超多的漏洞知识,像这次美团面试问的算法、数据库、Redis、设计模式等这些题目都是我刷到过的

并且我也将自己刷的题全部整理成了PDF或者Word文档(含详细答案解析)

我的美团offer凉凉了?开发工程师(Java岗)三面结束等通知...

66个Java面试知识点

架构专题(MySQL,Java,Redis,线程,并发,设计模式,Nginx,Linux,框架,微服务等)+大厂面试题详解(百度,阿里,腾讯,华为,迅雷,网易,中兴,北京中软等)

我的美团offer凉凉了?开发工程师(Java岗)三面结束等通知...

算法刷题(PDF)

我的美团offer凉凉了?开发工程师(Java岗)三面结束等通知...

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门,即可获取!
成了PDF或者Word文档(含详细答案解析)**

[外链图片转存中…(img-rKZO4HPG-1714676464644)]

66个Java面试知识点

架构专题(MySQL,Java,Redis,线程,并发,设计模式,Nginx,Linux,框架,微服务等)+大厂面试题详解(百度,阿里,腾讯,华为,迅雷,网易,中兴,北京中软等)

[外链图片转存中…(img-uR4rN8p5-1714676464645)]

算法刷题(PDF)

[外链图片转存中…(img-pNSUmFco-1714676464645)]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值