Java 底层大揭秘系列:如何实现定时任务
yund56 2025-05-02 20:53 23 浏览
定时器已经是现代软件中不可缺少的一部分,例如每隔5秒去查询一下状态,是否有新邮件,实现一个闹钟等, Java 中已经有现成的 api 供使用,但是如果你想设计更高效,更精准的定时器任务,就需要了解底层的硬件知识,比如实现一个分布式任务调度中间件,你可能要考虑到各个应用间时钟同步的问题。
Java 中我们要实现定时任务,有两种方式,一种通过 timer 类, 另外一种是 JUC 中的 ScheduledExecutorService ,不知道大家有没有好奇 JVM 是如何实现定时任务的,难道一直轮询时间,看是否时间到了,如果到了就调用对应的处理任务,但是这种一直轮询不释放 CPU 肯定是不可取的,要么就是线程阻塞,等到时间到了在来唤醒线程,那么 JVM 怎么知道时间到了,如何唤醒呢?
首先我们翻一下 JDK ,发现和时间相关的 API 大概有3处,而且这 3 处还都对时间的精度做了区分:
object.wait(long millisecond) 参数是毫秒,必须大于等于 0 ,如果等于 0 ,就一直阻塞直到其他线程来唤醒 ,timer 类就是通过 wait() 方法来实现,下面我们看一下wait的另外一个方法:
public final void wait(long timeout, int nanos) throws InterruptedException {
if (timeout < 0) {
throw new IllegalArgumentException("timeout value is negative");
}
if (nanos < 0 || nanos > 999999) {
throw new IllegalArgumentException(
"nanosecond timeout value out of range");
}
if (nanos > 0) {
timeout++;
}
wait(timeout);
}
这个方法是想提供一个可以支持纳秒级的超时时间,然而只是粗暴的加 1 毫秒。
Thread.sleep(long millisecond) 目前一般通过这种方式释放 CPU ,如果参数为 0 ,表示释放 CPU 给更高优先级的线程,自己从运行状态转换为可运行态等待 CPU 调度,他也提供了一个可以支持纳秒级的方法实现,跟 wait 额区别是它通过 500000 来分隔是否要加 1 毫秒。
public static void sleep(long millis, int nanos)
throws InterruptedException {
if (millis < 0) {
throw new IllegalArgumentException("timeout value is negative");
}
if (nanos < 0 || nanos > 999999) {
throw new IllegalArgumentException(
"nanosecond timeout value out of range");
}
if (nanos >= 500000 || (nanos != 0 && millis == 0)) {
millis++;
}
sleep(millis);
}
LockSupport.park(long nans) Condition.await()调用的该方法, ScheduledExecutorService 用的 condition.await() 来实现阻塞一定的超时时间,其他带超时参数的方法也都通过他来实现,目前大多定时器都是通过这个方法来实现的,该方法也提供了一个布尔值来确定时间的精度。
System.currentTimeMillis() 以及 System.nanoTime() 这两种方式都依赖于底层操作系统,前者是毫秒级,经测试 windows 平台的频率可能超过 10ms ,而后者是纳秒级别,频率在 100ns 左右,所以如果要获取更精准的时间建议用后者好了,api 了解完了,我们来看下定时器的底层是怎么实现的,现代PC机中有三种硬件时钟的实现,他们都是通过晶体振动产生的方波信号输入来完成时钟信号同步的。
实时时钟 RTC ,用于长时间存放系统时间的设备,即使关机也可以依靠主板中的电池继续计时。Linux 启动的时候会从 RTC 中读取时间和日期作为初始值,之后在运行期间通过其他计时器去维护系统时间。
可编程间隔定时器 PIT ,该计数器会有一个初始值,每过一个时钟周期,该初始值会减1,当该初始值被减到0时,就通过导线向 CPU 发送一个时钟中断, CPU 就可以执行对应的中断程序,也就是回调对应的任务
时间戳计数器 TSC , 所有的 Intel8086 CPU 中都包含一个时间戳计数器对应的寄存器,该寄存器的值会在每次 CPU 收到一个时钟周期的中断信号后就会加 1 。他比 PIT 精度高,但是不能编程,只能读取。
时钟周期:硬件计时器在多长时间内产生时钟脉冲,而时钟周期频率为1秒内产生时钟脉冲的个数。目前通常为1193180。
时钟滴答:当PIT中的初始值减到0的时候,就会产生一次时钟中断,这个初始值由编程的时候指定。
Linux启动的时候,先通过 RTC 获取初始时间,之后内核通过 PIT 中的定时器的时钟滴答来维护日期,并且会定时将该日期写入 RTC,而应用程序的定时器主要是通过设置 PIT 的初始值设置的,当初始值减到0的时候,就表示要执行回调函数了,这里大家会不会有疑问,这样同一时刻只能有一个定时器程序了,而我们在应用程序中,以及多个应用程序之间有好多定时器任务,我们可以参考 ScheduledExecutorService 的实现。
相关推荐
- Frequency函数用法详解—6个示例
-
Hello,大家好!今天和大家分享Frequency函数的使用方法。Frequency函数可以用于统计数据的频率分布,其语法为frequency(data_array,bins_array)。其中参数...
- 期刊论文网:查重复值excel的函数
-
问:如何在excel表格中用公式查找重复值答:打开需要查找重复值的excel表格,找到需要查找的A列。2、将鼠标定位在C1单元格中,在编辑栏中输入公式“=IF(COUNTIF($A:$A,A1)>...
- MATCH 函数的几个典型应用
-
MATCH函数,用于在一行或一列的查询区域中搜索指定内容,然后返回该内容在查询区域中的相对位置。MATCH函数有三个参数,第一个参数是查找对象,第二参数指定查找的范围,第三参数用数字来指定匹配方式...
- LOOKUP+INDIRECT组合封神!合并单元格查询从此无压力!
-
你是不是经常遇到这样的问题?“A列部门是合并单元格,B列是员工姓名,怎么根据姓名快速查部门?”手动翻找?眼睛看花!用VLOOKUP直接查?合并单元格全是坑!今天教你一个神级公式,用LOOKUP+IN...
- Match函数的使用方法
-
一、Match函数。功能:返回符合特定值特定顺序的项在数组中的相对位置。语法结构:=Match(查询值,数据范围,[匹配方式])。其中“匹配方式”有三种,分别为:-1(大于)、0(精准匹配)、1(小于...
- 两个表格如何匹配出相同的数据!用一个表查询另一个表进行填充!
-
Excel秘籍大全,前言两个表格匹配出相同的数据,可以通过匹配数据然后复制粘贴即可。当然,也可以直接使用VLOOKUP函数或者INDEX和MATCH函数组合来实现。具体使用哪种方法,可以根据自身需求来...
- INDEX+MATCH函数一组合,赛过VLOOKUP函数
-
提到查询数据,首先想到的就是VLOOKUP函数。但VLOOKUP查询有两个不便之处:1、查找值(第一个参数)始终位于查找区域(第二个参数)的第一列;2、从左向右查找容易,但从右向左查找需要IF或CHO...
- Excel模糊查找学会这5个方法,工作简直开了挂
-
模糊查找向来都是一个比较难的点,即使你对VLOOKUP、LOOKUP函数很熟悉,但有些问题也是要想很久。但当你学会下面这5个方法时,工作简直跟开了挂一样!下图中,我们想要通过E列的姓名在B:C表数据区...
- PART19:MATCH+HLOOKUP查找匹配直接套用!
-
公式解析HLOOKUP:在表格或数值数组的首行查找指定的数值,并由此返回表格或数组当前列中指定行处的数值。(默认情况下,表是升序的)数据表的首行必须为查找值所在行FALSE-精确匹配TRUE-近似匹配...
- 拒绝加班!快速完成Excel表格核对重复数据!
-
如何快速完成两个表格或者两组数据对比工作,在日常工作中非常重要,你用的是哪一种方法?平凡的世界平凡的你,努力学习使我们变得不平凡,今天要介绍的知识是excel核对表格差异的相关知识,你准备好学习exc...
- 如何筛选两个电子表格中重复的数据
-
#如何筛选两个电子表格中重复的数据#在Excel中筛选两个电子表格中的重复数据,可以通过多种方法实现。以下是几种常用的方法,以及它们的操作步骤:方法一:使用条件格式突出显示重复项打开Excel,并加载...
- Excel-函数技巧,COUNTIF+MATCH+INDEX提取不重复物料
-
要求:用函数求出B列当中,不重复的物料,也就是物料的唯一值。函数公式如下:{=IFERROR(INDEX($B$3:$B$142,MATCH(0,COUNTIF($G$2:G2,$B$3:$B$142...
- excel如何快速提取不重复值?提供2个经典用法
-
当excel有很多数据,而且有很多是重复数据的时候,如何快速提取不重复值就成为了需要解决的难题!如图中案例,A列是学校的列表,由于很多学员都出自同一所大学,所以就会有很多重复的。现在,需要提取不重复的...
- 如何根据多个条件查找数据,我整理了三种方法
-
一、案例如下图所示,A1:C8为某小区住户信息表,要求根据楼栋和房号查询住户姓名。二、解决方法方法一、INDEX+MATCH函数在G2单元格输入公式=INDEX($C$2:$C$8,MATCH(1,(...
- 利用match函数让两个表格排序一致
-
利用match函数让两个表格排序一致一线老师经常统计学生的情况,有时两个不同的表格由于姓名顺序不一致,手动调整往往需要耗费大量的时间与精力,当我们需要将两个表格的数据需要按照相同的顺序进行对比时,我们...
- 一周热门
- 最近发表
- 标签列表
-
- filter函数js (37)
- filter函数excel用不了 (73)
- 商城开发 (40)
- 影视网站免费源码最新版 (57)
- 影视资源api接口 (46)
- 网站留言板代码大全 (56)
- java版软件下载 (52)
- java教材电子课本下载 (48)
- 0基础编程从什么开始学 (50)
- java是用来干嘛的 (51)
- it入门应该学什么 (55)
- java线上课程 (55)
- 学java的软件叫什么软件 (38)
- 程序开发软件有哪些 (53)
- 软件培训 (59)
- 机器人编程代码大全 (50)
- 少儿编程教程免费 (45)
- 新代系统编程教学 (61)
- 共创世界编程网站 (38)
- 亲测源码 (36)
- 三角函数积分公式表 (35)
- 函数的表示方法 (34)
- 表格乘法的公式怎么设置 (34)
- sumif函数的例子 (34)
- 图片素材 (36)