首先来看一下Timer类

例子如下:

 package cn.concurrent.executor;

 import java.util.Timer;
import java.util.TimerTask; /**
* Created by spark on 17-9-25.
*/
public class TestTimer { static class Reminder {
Timer timer; public Reminder(int sec) {
timer = new Timer();
timer.schedule(new TimerTask() {
@Override
public void run() {
System.out.println("this is Timer.");
timer.cancel();
}
}, sec * 1000);
}
} public static void main(String[] args) {
System.out.println("------------------");
new Reminder(5);
System.out.println("。。。。。。。。。。");
}
}

运行结果如下:

------------------
..................
this is timer. Process finished with exit code 0

运行后,前2行很快就会输出,第三行5秒后出现。

从这个例子可以看出一个典型的利用timer执行计划任务的过程如下:

  • new一个TimerTask的子类,重写run方法来指定具体的任务
  • new一个Timer类,Timer.schedule(TimerTask),来运行具体的定时任务。
  • 调用相关调度方法执行计划。这个例子调用的是schedule方法。
  • 任务完成,结束线程。这个例子是调用cancel方法结束线程。

JDK源码来看一下:

     public void schedule(TimerTask var1, long var2) {
if(var2 < 0L) {
throw new IllegalArgumentException("Negative delay.");
} else {
this.sched(var1, System.currentTimeMillis() + var2, 0L);
}
} public void schedule(TimerTask var1, Date var2) {
this.sched(var1, var2.getTime(), 0L);
} public void schedule(TimerTask var1, long var2, long var4) {
if(var2 < 0L) {
throw new IllegalArgumentException("Negative delay.");
} else if(var4 <= 0L) {
throw new IllegalArgumentException("Non-positive period.");
} else {
this.sched(var1, System.currentTimeMillis() + var2, -var4);
}
} public void schedule(TimerTask var1, Date var2, long var3) {
if(var3 <= 0L) {
throw new IllegalArgumentException("Non-positive period.");
} else {
this.sched(var1, var2.getTime(), -var3);
}
} public void scheduleAtFixedRate(TimerTask var1, long var2, long var4) {
if(var2 < 0L) {
throw new IllegalArgumentException("Negative delay.");
} else if(var4 <= 0L) {
throw new IllegalArgumentException("Non-positive period.");
} else {
this.sched(var1, System.currentTimeMillis() + var2, var4);
}
} public void scheduleAtFixedRate(TimerTask var1, Date var2, long var3) {
if(var3 <= 0L) {
throw new IllegalArgumentException("Non-positive period.");
} else {
this.sched(var1, var2.getTime(), var3);
}
} private void sched(TimerTask var1, long var2, long var4) {
if(var2 < 0L) {
throw new IllegalArgumentException("Illegal execution time.");
} else {
if(Math.abs(var4) > 4611686018427387903L) {
var4 >>= 1;
} TaskQueue var6 = this.queue;
synchronized(this.queue) {
if(!this.thread.newTasksMayBeScheduled) {
throw new IllegalStateException("Timer already cancelled.");
} else {
Object var7 = var1.lock;
synchronized(var1.lock) {
if(var1.state != 0) {
throw new IllegalStateException("Task already scheduled or cancelled");
} var1.nextExecutionTime = var2;
var1.period = var4;
var1.state = 1;
} this.queue.add(var1);
if(this.queue.getMin() == var1) {
this.queue.notify();
} }
}
}
}

核心方法是 sched(),最终会把任务放入一个TaskQueue的队列中,然后来执行。

默认情况下,创建的timer线程会一直执行,主要有下面四种方式来终止timer线程:

  • 调用timer的cancle方法
  • 把timer线程设置成daemon线程,(new Timer(true)创建daemon线程),在jvm里,如果所有用户线程结束,那么守护线程也会被终止,不过这种方法一般不用。
  • 当所有任务执行结束后,删除对应timer对象的引用,线程也会被终止。
  • 调用System.exit方法终止程序

代码如下:

 public void cancel() {
TaskQueue var1 = this.queue;
synchronized(this.queue) {
this.thread.newTasksMayBeScheduled = false;
this.queue.clear();
this.queue.notify();
}
}

没有显式的线程stop方法,而是调用了queue的clear方法和queue的notify方法,clear是个自定义方法,notify是Objec自带的方法,很明显是去唤醒wait方法的。

 void clear() {
for(int var1 = 1; var1 <= this.size; ++var1) {
this.queue[var1] = null;
} this.size = 0;
}

原理是清空正个队列(底层是一个数列)。

看一下Timer的构造器:

 public Timer(String var1) {
this.queue = new TaskQueue();
this.thread = new TimerThread(this.queue);
this.threadReaper = new Object() {
protected void finalize() throws Throwable {
synchronized(Timer.this.queue) {
Timer.this.thread.newTasksMayBeScheduled = false;
Timer.this.queue.notify();
}
}
};
this.thread.setName(var1);
this.thread.start();
}

里面维护一个线程,然后看一下这个线程,如下:

继承Thread类,重写run()方法,看看run()方法的逻辑:

  public void run() {
boolean var9 = false; try {
var9 = true;
this.mainLoop();
var9 = false;
} finally {
if(var9) {
TaskQueue var4 = this.queue;
synchronized(this.queue) {
this.newTasksMayBeScheduled = false;
this.queue.clear();
}
}
} TaskQueue var1 = this.queue;
synchronized(this.queue) {
this.newTasksMayBeScheduled = false;
this.queue.clear();
}
}

看到调用了一个方法mainLoop();其他的操作都是初始化TaskQueue,并清空TaskQueue.

看一下mainLoop()方法。如下:

 private void mainLoop() {
while(true) {
while(true) {
try {
TaskQueue var3 = this.queue;
TimerTask var1;
boolean var2;
synchronized(this.queue) {
while(this.queue.isEmpty() && this.newTasksMayBeScheduled) {
this.queue.wait();
} if(this.queue.isEmpty()) {
return;
} var1 = this.queue.getMin();
Object var8 = var1.lock;
long var4;
long var6;
synchronized(var1.lock) {
if(var1.state == 3) {
this.queue.removeMin();
continue;
} var4 = System.currentTimeMillis();
var6 = var1.nextExecutionTime;
if(var2 = var6 <= var4) {
if(var1.period == 0L) {
this.queue.removeMin();
var1.state = 2;
} else {
this.queue.rescheduleMin(var1.period < 0L?var4 - var1.period:var6 + var1.period);
}
}
} if(!var2) {
this.queue.wait(var6 - var4);
}
} if(var2) {
var1.run();
}
} catch (InterruptedException var13) {
;
}
}
}
}

可以看到wait方法,之前的notify就是通知到这个wait,然后clear方法在notify之前做了清空数组的操作,所以会break,线程执行结束,退出。

schedule VS. scheduleAtFixedRate

这两个方法都是任务调度方法,他们之间区别是,schedule会保证任务的间隔是按照定义的period参数严格执行的,如果某一次调度时间比较长,那么后面的时间会顺延,保证调度间隔都是period,而scheduleAtFixedRate是严格按照调度时间来的,如果某次调度时间太长了,那么会通过缩短间隔的方式保证下一次调度在预定时间执行。举个栗子:你每个3秒调度一次,那么正常就是0,3,6,9s这样的时间,如果第二次调度花了2s的时间,如果是schedule,就会变成0,3+2,8,11这样的时间,保证间隔,而scheduleAtFixedRate就会变成0,3+2,6,9,压缩间隔,保证调度时间。

ScheduledThreadPoolExecutor与Timer的更多相关文章

  1. 【高并发】ScheduledThreadPoolExecutor与Timer的区别和简单示例

    JDK 1.5开始提供ScheduledThreadPoolExecutor类,ScheduledThreadPoolExecutor类继承ThreadPoolExecutor类重用线程池实现了任务的 ...

  2. Android定时器,推荐ScheduledThreadPoolExecutor

    Android定时器,推荐ScheduledThreadPoolExecutor 官方网址:http://developer.android.com/reference/java/util/Timer ...

  3. [转载] java多线程学习-java.util.concurrent详解(三)ScheduledThreadPoolExecutor

    转载自http://janeky.iteye.com/blog/770441 ------------------------------------------------------------- ...

  4. ScheduledThreadPoolExecutor Usage

    refs: https://blog.csdn.net/wenzhi20102321/article/details/78681379 对比一下Timer和ScheduledThreadPoolExe ...

  5. 深入理解Java线程池:ScheduledThreadPoolExecutor

    介绍 自JDK1.5开始,JDK提供了ScheduledThreadPoolExecutor类来支持周期性任务的调度.在这之前的实现需要依靠Timer和TimerTask或者其它第三方工具来完成.但T ...

  6. ScheduledThreadPoolExecutor使用指南

    ScheduledThreadPoolExecutor是Timer的多线程实现版本,JDK官方推荐使用.ScheduledThreadPoolExecutor用于替代Timer.是接口Schedule ...

  7. JUC源码分析-线程池篇(三)ScheduledThreadPoolExecutor

    JUC源码分析-线程池篇(三)ScheduledThreadPoolExecutor ScheduledThreadPoolExecutor 继承自 ThreadPoolExecutor.它主要用来在 ...

  8. 实例分析Scheduled Thread Pool Executor与Timer的区别

    摘要:JDK 1.5开始提供Scheduled Thread PoolExecutor类,Scheduled Thread Pool Executor类继承Thread Pool Executor类重 ...

  9. Java定时任务的常用实现

    Java的定时任务有以下几种常用的实现方式: 1)Timer 2)ScheduledThreadPoolExecutor 3)Spring中集成Cron Quartz 接下来依次介绍这几类具体实现的方 ...

随机推荐

  1. mysql-sql分析策略及优化

    tranlation事务:从失败中回复到正常状态的方法多个应用并发访问数据库时,提供隔离方法 acid原子性:要么成功.要么失败一致性:数据保持“合理性”隔离型:多个事务同时并发执行,每个事务就像各自 ...

  2. windows重建ESP分区修复引导

    开始 装在虚拟机里面的win7实在是太卡了,所以准备把虚拟磁盘文件复制到固态硬盘,,,但是,,, 我只有128GB固态... 那就只能卸载之前通宵装的linux 好气 首先需要装进入PE UEFI + ...

  3. 题解【BZOJ4472】[JSOI2015]salesman

    题面 树形\(\text{DP}\)与贪心的结合. 首先考虑树形\(\text{DP}\). 设\(dp_i\)表示从\(i\)出发,访问\(i\)的子树,并且最后回到\(i\)能获得的最大收益. 转 ...

  4. php对数组排序 关联数组功能比较

    用php在国家统计局中抓取 省市区县 代码.名称.排序order id,处理方式是通过curl请求网址,正则匹配 获取信息,并保存为json文件,以便后期读取文件. 过程中或遇到对json文件转化为数 ...

  5. 使用Limit实现分页

    limit语法 #语法 SELECT * FROM table LIMIT stratIndex,pageSize SELECT * FROM table LIMIT 5,10; // 检索记录行 6 ...

  6. Java-POJ1006-Biorhythms(中国剩余定理)

    https://blog.csdn.net/shanshanpt/article/details/8724769 有中文题面,就不解释了. 妥妥的中国剩余定理没跑了. Java跑得慢,一点办法也没有, ...

  7. JDK的卸载和安装

    Java入门 Java最大优势:基于JVM,跨平台 Java的几个版本 JavaSE:标准版,占领桌面,桌面程序,控制台开发等. JavaME:嵌入式开发,占领手机,手机,小家电等.(几乎死掉了) J ...

  8. 题解【UVA10054】The Necklace

    题目描述 输入格式 输出格式 题意简述 有一种由彩色珠子连接而成的项链.每个珠子的两半由不同颜色组成.如图所示,相邻两个珠子在接触的地方颜色相同.现在有一些零碎的珠子,需要确认它们是否可以复原成完整的 ...

  9. 题解【UVA12003】Array Transformer

    题目描述 输入输出格式 输入格式 输出格式 输入输出样例 输入样例#1 10 1 11 1 2 3 4 5 6 7 8 9 10 2 8 6 10 输出样例#1 1 2 3 4 5 6 7 8 9 6 ...

  10. Vue-cli3 项目配置 Vue.config.js( 代替vue-cli2 build config)

    Vue-cli3 搭建的项目 界面相对之前较为简洁 之前的build和config文件夹不见了,那么应该如何配置 如webpack等的配那 只需要在项目的根目录下新建 vue.config.js 文件 ...