java多线程编程核心技术——第五章总结

定时器Timer的使用

  在JDK中Timer类主要负责计划任务的功能,也就是在指定的时间开始执行某一个任务。

  Timer类的主要作用是设置计划任务,但封装任务的类却是TimerTask类。

  执行计划任务的代码要放入TimerTask的子类中,因为TimerTask是一个抽象类。

  注:在做Demo期间阿里的IDEA插件爆出建议:

  多线程并发处理定时任务时,Timer运行多个TimerTask时,只要其中之一没有捕获抛出的异常,其它任务就会终止,使用ScheduledExecutorService则没有这个问题。

方法schedule(TimerTask task, Date time)的测试

 1.执行任务的时间晚于当前时间:在未来执行的效果。

Timer timer = new Timer();

   这个构造是获取Timer对象的方式直接,这个构造方法创建的是非守护线程。

   TimerTask类似于线程(但并不是),他所执行的的定时任务是重写在run()方法中的(自定义任务类继承TimerTask然后重写了run())。

class MyTask extends TimerTask {
  public void run(){}
}
MyTask task = new MyTask();
//通过下面方法实现执行定时任务,date为要执行run()的时间。
timer.schedule(task,date)

  只要将date设置为未来的时间,且重写run()方法,就可以在制定时间执行想要做的定时任务了。

  这个Timer的构造是创建的是一个守护线程。

Timer timer = new Timer(true);

  源码如下:

  这个空参构造最终会调用下面的构造

  可以看出空参构造的作用:创建一个Timer就是启动一个新的线程,而且这个新线程并不是守护线程一直运行。

  使用另一种构造,则可以获取守护线程:

2.计划时间早于当前时间,提前运行的效果

  注:如果执行任务的时间早于当前时间,那么立即执行task任务。

3.多个TimerTask任务及延时的测试

  多个TimerTask可以被一个Timer执行。调用的方法是:schedule(TimerTask task, Date time)。

  TimerTask是以队列的方式一个一个被顺序执行的,所以可能出现执行的时间与预期的时间不一致的情况,因为前面的任务有可能消耗较长的时间,则后面的任务会被延迟。

方法schedule(TimerTask task, Date firstTime, long period)的测试

  该方法是在指定的日期后(firstTime)按指定的间隔(period)周期性地无限地执行某一任务。

1.计划时间晚于当前时间:在未来执行的结果

  一旦当前时间(本地计算机时间)到达指定时间(firstTime),TimerTask中的任务会被立即执行,随后每间隔一段时间(period)执行一次。

2.计划时间早于当前时间:提前运行的效果 

  在当前时间立即执行TimerTask中的任务,且每个一段时间(period)执行一次。

  注:起点以当前线程启动的时间为准,即调用schedule()方法为准,间隔也是以这个时间为起点。

3.任务执行时间被延时

  如果程序执行的消耗时间大于间隔时间(period),那么任务的执行就会延后,谁大以谁为准。

  注:消耗时间大,就以消耗时间为准;间隔时间大,就以间隔时间为准。

4.TimerTask类的cancel()方法  

  TimerTask类的cancle()方法是将自身从任务队列中移除,其他任务依旧在任务队列不收影响。

5.Timer类的cancel()方法

  Timer类的cancel()方法作用是将任务队列中的全部任务清空。

  注:调用cancel()后,全部任务被清空,进程被销毁。

  注:Timer类的cancel()的注意事项:Timer类中的cancel()方法有时并不一定会停止执行计划任务,而是正常执行。

    这是因为Timer类中的cancel()方法有时并没有争抢到queue锁,所以TimerTask类中的任务继续正常执行。

    (这个注意事项,我没有想太明白,以后会再看的)

方法schedule(TimerTask task, long delay)的测试

  该方法的作用是以当前时间为参考时间(这个当前时间指的是调用schedule(TimerTask task,long delay)的时间),在此基础上延迟指定毫秒数后执行一次TimerTask任务。 

方法schedule(TimerTask task, long delay, long period)的测试

  该方法的作用是以当前时间为参考时间(这个当前时间指的是调用schedule(TimerTask task,long delay)的时间),在此基础上延迟指定毫秒数,再以某一间隔时间无限次地执行某一任务。

方法scheduleAtFixedRate(TimerTask task, Date firstTime, long period)的测试

  方法schedule与方法scheduleAtFixedRate都会按顺序执行,所以不需要考虑非线程安全的情况。

  方法schedule与方法scheduleAtFixedRate的主要区别在于不延时的情况。

  使用schedule方法,如果执行任务没有被延时,那么下一次任务的执行时间参考的是上一次任务的“开始”时的时间计算的。

  使用scheduleAtFixedRate方法,如果执行任务没有被延时,那么下一次任务的执行时间参考的是上一次任务的“结束”的时间计算的。

  延时的情况则没有区别,也就是说如果两者都被延时,那么都已上一次任务结束的时间计算。

注:schedule不具有追赶性。

  scheduleAtFixedRate具有追赶性。

  追赶性:起始时间早于当前时间,会将两段时间内的任务进行追赶式(弥补错过的时间)的执行。

  这只是定时器简单的API实现。

  实际上Spring通过了注解实现定时器,会找时间对其进行总结的。

  注解式定时器——SpringTask的实现(未完成)

本文内容是书中内容兼具自己的个人看法所成。可能在个人看法上会有诸多问题(毕竟知识量有限,导致认知也有限),如果读者觉得有问题请大胆提出,我们可以相互交流、相互学习,欢迎你们的到来,心成意足,等待您的评价。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏noteless

[十三]JavaIO之PushBackInputStream

看得出来,如果你从来不曾回退过,那么好像什么都一样,还是使用原来的InputStream 进行读取

1182
来自专栏老付的网络博客

程序如何运行的

在写代码的时候,我们直接在没有编译报错的时候,直接点击运行后,ide会直接把程序的结果输出到控制台上,代码如下:

1055
来自专栏V站

如何提高PHP编程效率,提升程序编译质量(53条)

用单引号代替双引号来包含字符串,这样做会更快一些。因为PHP会在双引号包围的字符串中搜寻变量,单引号则不会,注意:只有echo能这么做,它是一种可以把多个字符串...

1485
来自专栏架构之路

Java的反射与RTTI(运行时类型信息)的区别

重要的事情说三遍: 反射并没有什么神奇之处!反射并没有什么神奇之处!反射并没有什么神奇之处! ---- 当通过反射与一个未知类型的对象打交道时,JVM只是简单地...

2596
来自专栏林德熙的博客

dotnet core 编程规范

本文实际只是翻译 .NET Core foundational libraries 官方文档的编码风格。

712
来自专栏Python专栏

Python | 19个优雅的Python编程技巧

http://lovesoo.org/pythonic-python-programming.html

1492
来自专栏青玉伏案

算法与数据结构(十二) 散列(哈希)表的创建与查找(Swift版)

散列表又称为哈希表(Hash Table), 是为了方便查找而生的数据结构。关于散列的表的解释,我想引用维基百科上的解释,如下所示: 散列表(Hash tab...

21910
来自专栏lgp20151222

@Transient 理解

transient使用小结 1)一旦变量被transient修饰,变量将不再是对象持久化的一部分,该变量内容在序列化后无法获得访问。 2)transient...

871
来自专栏Python小屋

暴力测试也疯狂——论Python代码优化

问题描述:在123456789这9个数字中间插入任意多个+和-的组合,使得表达式的值为100,输出所有符合条件的表达式。 对于该问题,前天推送了一个暴力测试的代...

3314
来自专栏企鹅号快讯

PHP中被忽略的性能优化利器:生成器

如果是做Python或者其他语言的小伙伴,对于生成器应该不陌生。但很多PHP开发者或许都不知道生成器这个功能,可能是因为生成器是PHP5.5.0才引入的功能,也...

36914

扫码关注云+社区