好好的Timer居然有坑?
2020-03-24 16:23:35来源:博客园 阅读 ()
好好的Timer居然有坑?
在做定时任务时,可能会使用到Timer+TimerTask类,但是这两个小小的类,却有大坑。
先来复现一下问题,如下,可能预期的是第一个PrintTask从1一直往后打印,直到为5时抛出异常,第二个PrintTask从100往后不间断打印。
public class Main12 {
public static void main(String[] args) {
Timer timer =new Timer();
timer.schedule(new PrintTask(1),0,1000);
timer.schedule(new PrintTask(100),0,1000);
}
static class PrintTask extends java.util.TimerTask{
private int mStart;
public PrintTask(int start) {
mStart = start;
}
@Override
public void run() {
if (mStart==5){
throw new RuntimeException();
}
System.out.println("mStart=="+mStart++);
}
}
}
mStart==1
mStart==100
mStart==101
mStart==2
mStart==3
mStart==102
mStart==4
mStart==103
Exception in thread "Timer-0" java.lang.RuntimeException
at main.Main12$PrintTask.run(Main12.java:21)
at java.util.TimerThread.mainLoop(Timer.java:555)
at java.util.TimerThread.run(Timer.java:505)
Process finished with exit code 0
但是结果却是,在第一个PrintTask到5抛出异常后,第二个PrintTask却奇迹般的终止了。
这里是不是说明某个TimerTask中的run方法向外抛出异常后,其他TimerTask会自动终止呢?
那就得看Timer的实现原理了。
Timer实现原理
Timer中维护这一个TaskQueue和一个TimerThread,TaskQueue是一个由平衡二叉树实现的优先级队列,在调用Timer的schedule方法时,也就是把TimerTask放入TaskQueue中。TimerThread则是具体执行任务的线程,它从TaskQueue中获取优先级最高的任务去执行,并且只有当前任务执行完后才会从队列中获取下一个任务,不管队列里是否有任务已经到了设置的delay时间。这个模型也就是多生产者---单消费者(只要这个消费者线程挂了,其他任务就没办法执行)。
所以,重中之重在于TimerThread,只需要搞明白TimerThread就可以了。
在TimerThread的mainLoop方法中开始获取任务并执行,当任务抛出InterruptedException以外的异常时,唯一的消费者线程就会因异常而终止,队列里的其他任务就会被清除。
public void run() {
try {
mainLoop();
} finally {
synchronized(queue) {
newTasksMayBeScheduled = false;
queue.clear();
}
}
}
private void mainLoop() {
while (true) {
try {
TimerTask task;
boolean taskFired;
//从队列里面获取任务时加锁。
synchronized(queue) {
......
}
if (taskFired)
task.run(); //执行任务
} catch(InterruptedException e) {
}
}
所以在TimerTask的run方法中,最好使用try-catch结构捕捉异常,不要向上抛。
还可以使用ScheduledThreadPoolExecutor,ScheduledThreadPoolExecutor中的一个任务抛出异常,其他任务也不受影响。
原文链接:https://www.cnblogs.com/HouXinLin/p/12560054.html
如有疑问请与原作者联系
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
- 头条面试居然跟我扯了半小时的Semaphore 2020-06-08
- 我可真是醉了,一个SpringBoot居然问了我30个问题 2020-06-08
- 面试:源码看过没?答:看过一点!好的,下一位 2020-05-26
- 腾讯面试居然跟我扯了半小时的CountDownLatch 2020-05-15
- Java连载111-timer定时器、反射机制概述 2020-04-22
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash