当前位置: 首页 > article >正文

Java——定时任务

定时任务1、Timer和TimerTask1.1、基本用法1.2、基本示例1.3、基本原理1.4、死循环1.5、异常任务1.6、总结2、ScheduledExecutorService2.1、基本用法2.2、基本示例2.3、基本原理在Java中主要有两种方式实现定时任务使用java.util包中的Timer和TimerTask。使用Java并发包中的ScheduledExecutorService。1、Timer和TimerTask1.1、基本用法TimerTask表示一个定时任务它是一个抽象类实现了Runnable具体的定时任务需要继承该类实现run方法。Timer是一个具体类它负责定时任务的调度和执行主要方法有//在指定绝对时间time运行任务taskpublicvoidschedule(TimerTasktask,Datetime)//在当前时间延时delay毫秒后运行任务taskpublicvoidschedule(TimerTasktask,longdelay)//固定延时重复执行第一次计划执行时间为firstTime//后一次的计划执行时间为前一次实际执行时间加上periodpublicvoidschedule(TimerTasktask,DatefirstTime,longperiod)//同样是固定延时重复执行第一次执行时间为当前时间加上delaypublicvoidschedule(TimerTasktask,longdelay,longperiod)//固定频率重复执行第一次计划执行时间为firstTime//后一次的计划执行时间为前一次计划执行时间加上periodpublicvoidscheduleAtFixedRate(TimerTasktask,DatefirstTime,longperiod)//同样是固定频率重复执行第一次计划执行时间为当前时间加上delaypublicvoidscheduleAtFixedRate(TimerTasktask,longdelay,longperiod)需要注意固定延时fixed-delay与固定频率fixed-rate的区别二者都是重复执行但后一次任务执行相对的时间是不一样的对于固定延时它是基于上次任务的“实际”执行时间来算的如果由于某种原因上次任务延时了则本次任务也会延时而固定频率会尽量补够运行次数。另外需要注意的是如果第一次计划执行的时间firstTime是一个过去的时间则任务会立即运行对于固定延时的任务下次任务会基于第一次执行时间计算而对于固定频率的任务则会从firstTime开始算有可能加上period后还是一个过去时间从而连续运行很多次直到时间超过当前时间。我们通过一些简单的例子具体来看下。1.2、基本示例看一个最简单的例子如代码所示。publicclassBasicTimer{staticclassDelayTaskextendsTimerTask{Overridepublicvoidrun(){System.out.println(delayed task);}}publicstaticvoidmain(String[]args)throwsInterruptedException{TimertimernewTimer();timer.schedule(newDelayTask(),1000);Thread.sleep(2000);timer.cancel();}}创建一个Timer对象1秒钟后运行DelayTask最后调用Timer的cancel方法取消所有定时任务。publicclassTimerFixedDelay{staticclassLongRunningTaskextendsTimerTask{Overridepublicvoidrun(){try{Thread.sleep(5000);}catch(InterruptedExceptione){}System.out.println(long running finished);}}staticclassFixedDelayTaskextendsTimerTask{Overridepublicvoidrun(){System.out.println(System.currentTimeMillis());}}publicstaticvoidmain(String[]args)throwsInterruptedException{TimertimernewTimer();timer.schedule(newLongRunningTask(),10);timer.schedule(newFixedDelayTask(),100,1000);}}有两个定时任务第一个运行一次但耗时5秒第二个是重复执行1秒一次第一个先运行。运行该程序会发现第二个任务只有在第一个任务运行结束后才会开始运行运行后1秒一次。如果替换上面的代码为固定频率即变为代码所示。publicclassTimerFixedRate{staticclassLongRunningTaskextendsTimerTask{//省略与代码清单18-4一样}staticclassFixedRateTaskextendsTimerTask{//省略与代码清单18-4一样}publicstaticvoidmain(String[]args)throwsInterruptedException{TimertimernewTimer();timer.schedule(newLongRunningTask(),10);timer.scheduleAtFixedRate(newFixedRateTask(),100,1000);}}运行该程序第二个任务同样只有在第一个任务运行结束后才会运行但它会把之前没有运行的次数补过来一下子运行5次输出类似下面这样long running finished 1489467662330 1489467662330 1489467662330 1489467662330 1489467662330 14894676624191.3、基本原理Timer内部主要由任务队列和Timer线程两部分组成。任务队列是一个基于堆实现的优先级队列按照下次执行的时间排优先级。Timer线程负责执行所有的定时任务需要强调的是一个Timer对象只有一个Timer线程所以对于上面的例子任务会被延迟。Timer线程主体是一个循环从队列中获取任务如果队列中有任务且计划执行时间小于等于当前时间就执行它如果队列中没有任务或第一个任务延时还没到就睡眠。如果睡眠过程中队列上添加了新任务且新任务是第一个任务Timer线程会被唤醒重新进行检查。在执行任务之前Timer线程判断任务是否为周期任务如果是就设置下次执行的时间并添加到优先级队列中对于固定延时的任务下次执行时间为当前时间加上period对于固定频率的任务下次执行时间为上次计划执行时间加上period。1.4、死循环一个Timer对象只有一个Timer线程这意味着定时任务不能耗时太长更不能是无限循环。看个例子如代码所示。publicclassEndlessLoopTimer{staticclassLoopTaskextendsTimerTask{Overridepublicvoidrun(){while(true){try{//模拟执行任务Thread.sleep(1000);}catch(InterruptedExceptione){e.printStackTrace();}}}}//永远也没有机会执行staticclassExampleTaskextendsTimerTask{Overridepublicvoidrun(){System.out.println(hello);}}publicstaticvoidmain(String[]args)throwsInterruptedException{TimertimernewTimer();timer.schedule(newLoopTask(),10);timer.schedule(newExampleTask(),100);}}第一个定时任务是一个无限循环其后的定时任务ExampleTask将永远没有机会执行。1.5、异常任务关于Timer线程还需要强调非常重要的一点在执行任何一个任务的run方法时一旦run抛出异常Timer线程就会退出从而所有定时任务都会被取消。我们看个简单的示例如代码所示。publicclassTimerException{staticclassTaskAextendsTimerTask{Overridepublicvoidrun(){System.out.println(task A);}}staticclassTaskBextendsTimerTask{Overridepublicvoidrun(){System.out.println(task B);thrownewRuntimeException();}}publicstaticvoidmain(String[]args)throwsInterruptedException{TimertimernewTimer();timer.schedule(newTaskA(),1,1000);timer.schedule(newTaskB(),2000,1000);}}期望TaskA每秒执行一次但TaskB会抛出异常导致整个定时任务被取消程序终止屏幕输出为taskAtaskAtaskBExceptionin threadTimer-0java.lang.RuntimeExceptionatlaoma.demo.timer.TimerException$TaskB.run(TimerException.java:21)atjava.util.TimerThread.mainLoop(Timer.java:555)atjava.util.TimerThread.run(Timer.java:505)所以如果希望各个定时任务不互相干扰一定要在run方法内捕获所有异常。1.6、总结可以看到Timer/TimerTask的基本使用是比较简单的但我们需要注意后台只有一个线程在运行固定频率的任务被延迟后可能会立即执行多次将次数补够固定延时任务的延时相对的是任务执行前的时间不要在定时任务中使用无限循环一个定时任务的未处理异常会导致所有定时任务被取消。2、ScheduledExecutorService由于Timer/TimerTask的一些问题Java并发包引入了ScheduledExecutorService下面我们介绍它的基本用法、基本示例和基本原理。2.1、基本用法ScheduledExecutorService是一个接口其定义为publicinterfaceScheduledExecutorServiceextendsExecutorService{//单次执行在指定延时delay后运行commandpublicScheduledFuture?schedule(Runnablecommand,longdelay,TimeUnitunit);//单次执行在指定延时delay后运行callablepublicVScheduledFutureVschedule(CallableVcallable,longdelay,TimeUnitunit);//固定频率重复执行publicScheduledFuture?scheduleAtFixedRate(Runnablecommand,longinitialDelay,longperiod,TimeUnitunit);//固定延时重复执行publicScheduledFuture?scheduleWithFixedDelay(Runnablecommand,longinitialDelay,longdelay,TimeUnitunit);}它们的返回类型都是ScheduledFuture它是一个接口扩展了Future和Delayed没有定义额外方法。这些方法的大部分语义与Timer中的基本是类似的。对于固定频率的任务第一次执行时间为initialDelay后第二次为initialDelayperiod第三次为initial-Delay2*period以此类推。不过对于固定延时的任务它是从任务执行后开始算的第一次为initialDelay后第二次为第一次任务执行结束后再加上delay。与Timer不同它不支持以绝对时间作为首次运行的时间。ScheduledExecutorService的主要实现类是ScheduledThreadPoolExecutor它是线程池ThreadPoolExecutor的子类是基于线程池实现的它的主要构造方法是publicScheduledThreadPoolExecutor(intcorePoolSize)此外还有构造方法可以接受参数ThreadFactory和RejectedExecutionHandler含义与ThreadPoolExecutor一样我们就不赘述了。它的任务队列是一个无界的优先级队列所以最大线程数对它没有作用即使core-PoolSize设为0它也会至少运行一个线程。工厂类Executors也提供了一些方便的方法以方便创建ScheduledThreadPoolExecutor如下所示//单线程的定时任务执行服务publicstaticScheduledExecutorServicenewSingleThreadScheduledExecutor()publicstaticScheduledExecutorServicenewSingleThreadScheduledExecutor(ThreadFactorythreadFactory)//多线程的定时任务执行服务publicstaticScheduledExecutorServicenewScheduledThreadPool(intcorePoolSize)publicstaticScheduledExecutorServicenewScheduledThreadPool(intcorePoolSize,ThreadFactorythreadFactory)2.2、基本示例由于可以有多个线程执行定时任务一般任务就不会被某个长时间运行的任务所延迟了。比如对于TimerFixedDelay如果改为代码所示publicclassScheduledFixedDelay{staticclassLongRunningTaskimplementsRunnable{//省略与代码清单18-4一样}staticclassFixedDelayTaskimplementsRunnable{//省略与代码清单18-4一样}publicstaticvoidmain(String[]args)throwsInterruptedException{ScheduledExecutorServicetimerExecutors.newScheduledThreadPool(10);timer.schedule(newLongRunningTask(),10,TimeUnit.MILLISECONDS);timer.scheduleWithFixedDelay(newFixedDelayTask(),100,1000,TimeUnit.MILLISECONDS);}}再次执行第二个任务就不会被第一个任务延迟了。另外与Timer不同单个定时任务的异常不会再导致整个定时任务被取消即使后台只有一个线程执行任务。我们看个例子如代码所示。publicclassScheduledException{staticclassTaskAimplementsRunnable{Overridepublicvoidrun(){System.out.println(task A);}}staticclassTaskBimplementsRunnable{Overridepublicvoidrun(){System.out.println(task B);thrownewRuntimeException();}}publicstaticvoidmain(String[]args)throwsInterruptedException{ScheduledExecutorServicetimerExecutors.newSingleThreadScheduledExecutor();timer.scheduleWithFixedDelay(newTaskA(),0,1,TimeUnit.SECONDS);timer.scheduleWithFixedDelay(newTaskB(),2,1,TimeUnit.SECONDS);}}TaskA和TaskB都是每秒执行一次TaskB两秒后执行但一执行就抛出异常屏幕的输出类似如下taskAtaskAtaskBtaskAtaskA…这说明定时任务TaskB被取消了但TaskA不受影响即使它们是由同一个线程执行的。不过需要强调的是与Timer不同没有异常被抛出TaskB的异常没有在任何地方体现。所以与Timer中的任务类似应该捕获所有异常。2.3、基本原理ScheduledThreadPoolExecutor的实现思路与Timer基本是类似的都有一个基于堆的优先级队列保存待执行的定时任务它的主要不同是它的背后是线程池可以有多个线程执行任务。它在任务执行后再设置下次执行的时间对于固定延时的任务更为合理。任务执行线程会捕获任务执行过程中的所有异常一个定时任务的异常不会影响其他定时任务不过发生异常的任务即使是一个重复任务不会再被调度。

相关文章:

Java——定时任务

定时任务1、Timer和TimerTask1.1、基本用法1.2、基本示例1.3、基本原理1.4、死循环1.5、异常任务1.6、总结2、ScheduledExecutorService2.1、基本用法2.2、基本示例2.3、基本原理在Java中,主要有两种方式实现定时任务: 使用java.util包中的Timer和Timer…...

555定时器深度解析:从RC电路到三种工作模式的原理与应用

1. 项目概述在电子设计的工具箱里,有那么几颗芯片,你几乎可以在任何时代的电路板上找到它们的身影。它们可能不是性能最强的,但一定是应用最广、最经久不衰的。今天要聊的555定时器,就是这样一个“活化石”级别的存在。自上世纪70…...

AI 越火,存储越关键:一颗存储藏着设备稳定运行的秘密

很多人看芯片,第一眼喜欢看“大件”。CPU、GPU、主控、屏幕、电池、无线模组,好像这些才是产品的主角。但真正做过硬件的人都知道:一个设备能不能稳定开机,程序能不能快速读取,系统能不能在复杂环境下长期跑得住&#…...

终极虚拟显示器解决方案:ParsecVDisplay完全指南

终极虚拟显示器解决方案:ParsecVDisplay完全指南 【免费下载链接】parsec-vdd ✨ Perfect virtual display for game streaming 项目地址: https://gitcode.com/gh_mirrors/pa/parsec-vdd ParsecVDisplay是一款基于Parsec虚拟显示驱动(VDD&#x…...

FTDI Friend:从USB转串口到AVR编程的嵌入式开发利器

1. 项目概述:为什么你需要一个FTDI Friend?在嵌入式开发和单片机玩家里,串口通信就像空气和水一样基础且不可或缺。无论是给Arduino上传代码,还是让树莓派和传感器模块“说上话”,亦或是调试一个自己焊的STM32最小系统…...

Harness 中的请求标识染色:端到端追踪

1. 标题选项(核心关键词:Harness、请求标识染色、端到端追踪、可观测性、CI/CD) 「Harness 可观测性实战:请求标识染色实现全链路端到端追踪」 「从0到1搞定Harness请求染色:让微服务调用链路+变更链路无所遁形」 「告别排查黑洞:Harness请求标识染色的端到端追踪落地指南…...

codex features

这份列表是 OpenAI Codex 内部的功能开关,每个功能都处于特定的开发阶段。下面按稳定程度对这些功能进行了分类说明。 🟢 稳定版 (Stable) - 可以放心使用 这些功能已经过充分测试,适合在日常工作流中启用。功能名称功能说明apps支持 AI 直接…...

AI Agent在科学研究中的辅助作用

AI Agent在科学研究中的辅助作用 关键词:AI Agent, 科学研究辅助, 自主代理架构, 多模态推理, 文献挖掘, 实验设计, 未来展望 摘要:本文将像给小学生讲魔法实验室故事一样,深入浅出地拆解AI Agent这个“超级科研小助手天团”的核心原理、架构…...

模块六-数据合并与连接——32. merge 合并(上)

32. merge 合并(上) 1. 概述 merge 是 Pandas 中最强大的数据合并函数,类似于 SQL 中的 JOIN 操作。它可以根据一个或多个键将两个 DataFrame 的行连接起来。 import pandas as pd import numpy as np# 创建示例数据 # 员工表 employees pd.…...

【2026最新】鸿蒙NEXT数据持久化实战:培训班管理系统数据存储全攻略

鸿蒙开发中数据总是丢失?本地存储和网络请求搞不定?本文用15分钟带你彻底搞懂Preferences、RDB、HTTP三大数据持久化方案,附完整培训班管理系统实战代码和踩坑记录,让你的鸿蒙App数据存储从此安全可靠!一、学员信息本地…...

降AI率软件越便宜越好吗?实测5个主流降AI工具,首选嘎嘎降!

一、前言:2026 年毕业必须通过 aigc 检测 2026 年各高校对学术论文的 AIGC 疑似度的审查全面变严, 均发布了具体 AIGC 检测报告和数值要求,211 和 985 高校规定本科论文 AI 率要低于 20%, 硕士要求 AI 率不高于 15%。普通高校一般要求 AI 率控制在 30% 以内。AIGC 检测率超标的…...

基于Feather RP2040与CircuitPython的CNC旋钮宏键盘DIY指南

1. 项目概述:打造你的专属生产力旋钮如果你经常使用像Cura、Fusion 360或者Adobe系列这类专业软件,一定对频繁切换工具、调整参数时在键盘和鼠标间来回切换的繁琐深有体会。传统的键盘快捷键虽然快,但组合键太多容易忘记,而且缺乏…...

从技巧到工程:构建可维护的Prompt设计体系与实战指南

1. 项目概述:Prompt Engineering 的实战价值最近在 GitHub 上看到一个名为 “imJunaidAfzal/Prompt-Engineering” 的项目,这让我想起了过去一年里,和团队一起从零开始摸索大语言模型应用落地的经历。Prompt Engineering,中文常译…...

ECHO:不止是播放器——一款完整的桌面音乐产品

下载地址:canghaiapp.com 软件介绍:不止是播放器 ECHO 将自己定位为一款“完整的桌面音乐产品”。从技术架构上看,它确实与此相符: 核心技术选型:项目基于 Electron React 技术栈构建。Electron 赋予了它调用原生系…...

BeagleBone Black设备树覆盖层实战:从原理到自定义SPI/UART配置

1. 项目概述:为什么BeagleBone Black开发者必须掌握设备树?如果你正在使用BeagleBone Black(BBB)进行嵌入式开发,并且已经不止一次地困惑于为什么某个外设(比如UART、SPI或者某个GPIO)无法按预期…...

FanControl终极指南:如何突破NVIDIA显卡风扇30%限制实现0 RPM静音控制

FanControl终极指南:如何突破NVIDIA显卡风扇30%限制实现0 RPM静音控制 【免费下载链接】FanControl.Releases This is the release repository for Fan Control, a highly customizable fan controlling software for Windows. 项目地址: https://gitcode.com/Git…...

企业级应用如何通过 Taotoken 统一管理多个团队的模型调用

🚀 告别海外账号与网络限制!稳定直连全球优质大模型,限时半价接入中。 👉 点击领取海量免费额度 企业级应用如何通过 Taotoken 统一管理多个团队的模型调用 在中大型企业的技术实践中,多个项目组或产品线同时接入和使…...

赛车电气系统设计的现代化转型与实践

1. 赛车电气系统设计的现状与挑战当人们谈论赛车技术时,脑海中浮现的往往是碳纤维车身、空气动力学套件或是大马力发动机。但在这光鲜亮丽的表象背后,电气系统才是现代赛车的"神经系统"。有趣的是,这个关键领域的设计方法却呈现出两…...

Topit:macOS窗口置顶的终极解决方案,开源高效的多任务开发利器

Topit:macOS窗口置顶的终极解决方案,开源高效的多任务开发利器 【免费下载链接】Topit Pin any window to the top of your screen / 在Mac上将你的任何窗口强制置顶 项目地址: https://gitcode.com/gh_mirrors/to/Topit Topit是一款专为macOS系统…...

复杂系统交付中的风险治理与经济模型转型

1. 复杂系统交付中的风险本质与治理转型在航空航天、国防军工等复杂系统开发领域,项目失败率长期居高不下。根据IBM对全球500个大型系统的调研,73%的项目存在严重进度延迟,平均超支达到原始预算的189%。这种系统性失效的根源在于传统工程治理…...

Linux驱动开发:原子操作实现LED设备互斥访问

1. 项目概述:用原子操作给LED驱动加把“锁”在嵌入式Linux开发里,驱动开发是绕不开的一环。很多时候,一个硬件设备,比如一个简单的LED灯,可能会被多个用户空间的应用程序同时访问。想象一下,一个APP想开灯&…...

3分钟掌握FanControl:Windows风扇控制终极指南

3分钟掌握FanControl:Windows风扇控制终极指南 【免费下载链接】FanControl.Releases This is the release repository for Fan Control, a highly customizable fan controlling software for Windows. 项目地址: https://gitcode.com/GitHub_Trending/fa/FanCon…...

Linux系统下Vue开发环境搭建:从Node.js到Vite的完整指南

1. 项目概述:为什么要在Linux上搭建Vue环境?对于前端开发者而言,Vue.js 早已不是陌生的名字。它凭借其渐进式的设计理念、灵活的组件化系统和相对平缓的学习曲线,成为了构建现代Web应用的主流框架之一。然而,很多开发者…...

ctfileGet:城通网盘直连地址解析工具的技术原理与实用指南

ctfileGet:城通网盘直连地址解析工具的技术原理与实用指南 【免费下载链接】ctfileGet 获取城通网盘一次性直连地址 项目地址: https://gitcode.com/gh_mirrors/ct/ctfileGet ctfileGet是一个基于Web的开源工具,专门用于解析城通网盘分享链接并获…...

基于RK3568核心板的智能家居控制器:从硬件选型到软件架构实战

1. 项目概述:当智能家居控制器遇上国产高性能核心板最近在做一个智能家居中控的案子,客户对性能、成本和本地化能力要求都比较高。选型阶段,我们团队把市面上主流的几款ARM核心板都摸了一遍,从传统的树莓派CM4到全志、瑞芯微的方案…...

AI智能体工具搜索系统:从MCP协议到语义检索的工程实践

1. 项目概述:从“工具搜索”到“智能体工具箱”的进化 最近在折腾AI智能体(Agent)开发的朋友,估计都绕不开一个核心问题:如何让智能体高效、准确地调用外部工具?无论是让它帮你查天气、发邮件,还…...

基于LLM与RAG构建智能问答系统:架构、实现与优化指南

1. 项目概述:当RAG遇上LLM,构建你的智能知识问答引擎最近在GitHub上看到一个挺有意思的项目,叫“Jenqyang/LLM-Powered-RAG-System”。光看名字,圈内人大概就能猜到个七七八八:这是一个基于大语言模型(LLM&…...

免费开源字体编辑器终极指南:5个核心模块带你从零到专业设计

免费开源字体编辑器终极指南:5个核心模块带你从零到专业设计 【免费下载链接】fontforge Free (libre) font editor for Windows, Mac OS X and GNULinux 项目地址: https://gitcode.com/gh_mirrors/fo/fontforge 想要免费编辑字体却找不到合适的工具&#x…...

Node.js性能预测工具nodestradamus:从监控到预警的实践指南

1. 项目概述与核心价值最近在折腾一些服务器监控和性能预测的活儿,偶然间在GitHub上发现了一个叫nodestradamus的项目,作者是ChristosGrigoras。这个名字挺有意思,结合了“Node.js”和“诺查丹玛斯”(那位著名的预言家&#xff09…...

芯片/半导体/CPO光模块 深度分析报告

芯片/半导体/CPO光模块 深度分析报告报告生成时间:2026年5月16日 分析标的:芯片半导体板块 CPO光模块产业链 龙头标的:中际旭创(300308)、天孚通信(300394)、新易盛(300502)、寒武纪(688256)、海光信息(688041) 数据来源:公开市场…...