关闭

Java Timer(定时调用、实现固定时间执行)

发表于:2015-8-05 09:16

字体: | 上一篇 | 下一篇 | 我要投稿

 作者:WhyWin    来源:51Testing软件测试网采编

  最近需要用到定时调用的功能。可以通过java的Timer类来进行定时调用,下面是有关Timer的一些相关知识。
  其实就Timer来讲就是一个调度器,而TimerTask呢只是一个实现了run方法的一个类,而具体的TimerTask需要由你自己来实现,例如这样:
  Timer timer = new Timer();
  timer.schedule(new TimerTask() {
  public void run() {
  System.out.println("11232");
  }
  }, 200000 , 1000);
  这里直接实现一个TimerTask(当然,你可以实现多个TimerTask,多个TimerTask可以被一个Timer会被分配到多个 Timer中被调度,后面会说到Timer的实现机制就是说内部的调度机制),然后编写run方法,20s后开始执行,每秒执行一次,当然你通过一个 timer对象来操作多个timerTask,其实timerTask本身没什么意义,只是和timer集合操作的一个对象,实现它就必然有对应的run 方法,以被调用,他甚至于根本不需要实现Runnable,因为这样往往混淆视听了,为什么呢?也是本文要说的重点。
  在说到timer的原理时,我们先看看Timer里面的一些常见方法:
  1、这个方法是调度一个task,经过delay(ms)后开始进行调度,仅仅调度一次。
  public void schedule(TimerTask task, long delay)
  2、在指定的时间点time上调度一次。
  public void schedule(TimerTask task, Date time)
  3、这个方法是调度一个task,在delay(ms)后开始调度,每次调度完后,最少等待period(ms)后才开始调度。
  public void schedule(TimerTask task, long delay, long period)
  4、和上一个方法类似,唯一的区别就是传入的第二个参数为第一次调度的时间。
  public void schedule(TimerTask task, Date firstTime, long period)
  5、调度一个task,在delay(ms)后开始调度,然后每经过period(ms)再次调度,貌似和方法:schedule是一样的,其实不然,后面你会根据源码看到,schedule在计算下一次执行的时间的时候,是通过当前时间(在任务执行前得到) + 时间片,而scheduleAtFixedRate方法是通过当前需要执行的时间(也就是计算出现在应该执行的时间)+ 时间片,前者是运行的实际时间,而后者是理论时间点,例如:schedule时间片是5s,那么理论上会在5、10、15、20这些时间片被调度,但是如果由于某些CPU征用导致未被调度,假如等到第8s才被第一次调度,那么schedule方法计算出来的下一次时间应该是第13s而不是第10s,这样有可能下次就越到20s后而被少调度一次或多次,而scheduleAtFixedRate方法就是每次理论计算出下一次需要调度的时间用以排序,若第8s被调度,那么计算出应该是第10s,所以它距离当前时间是2s,那么再调度队列排序中,会被优先调度,那么就尽量减少漏掉调度的情况。
  public void scheduleAtFixedRate(TimerTask task, long delay, long period)
  6、方法同上,唯一的区别就是第一次调度时间设置为一个Date时间,而不是当前时间的一个时间片,我们在源码中会详细说明这些内容。
  public void scheduleAtFixedRate(TimerTask task, Date firstTime,long period)
  源码部分
  首先看Timer的构造方法有几种:
  构造方法1:无参构造方法,简单通过Tiemer为前缀构造一个线程名称:
  public Timer() {
  this("Timer-" + serialNumber());
  }
  创建的线程不为主线程,则主线程结束后,timer自动结束,而无需使用cancel来完成对timer的结束。
  构造方法2:传入了是否为后台线程,后台线程当且仅当进程结束时,自动注销掉。
  public Timer(boolean isDaemon) {
  this("Timer-" + serialNumber(), isDaemon);
  }
  另外两个构造方法负责传入名称和将timer启动:
  public Timer(String name, boolean isDaemon) {
  thread.setName(name);
  thread.setDaemon(isDaemon);
  thread.start();
  }
  这里有一个thread,这个thread很明显是一个线程,被包装在了Timer类中,我们看下这个thread的定义是:
  private TimerThread thread = new TimerThread(queue);
21/212>
《2023软件测试行业现状调查报告》独家发布~

关注51Testing

联系我们

快捷面板 站点地图 联系我们 广告服务 关于我们 站长统计 发展历程

法律顾问:上海兰迪律师事务所 项棋律师
版权所有 上海博为峰软件技术股份有限公司 Copyright©51testing.com 2003-2024
投诉及意见反馈:webmaster@51testing.com; 业务联系:service@51testing.com 021-64471599-8017

沪ICP备05003035号

沪公网安备 31010102002173号