java目前常用的几种定时任务
java目前常用的几种定时任务
一、JDK自带的Timer
Timer是jdk中提供的一个定时器工具,使用的时候会在主线程之外起一个单独的线程执行指定的计划任务,可以指定执行一次或者反复执行多次。
TimerTask是一个实现了Runnable接口的抽象类,代表一个可以被Timer执行的任务。
- 1 import org.junit.Test;
- 2 import java.util.Timer;
- 3 import java.util.TimerTask;
- 4
- 5 public class TimerTest {
- 6 @Test
- 7 public void test(){
- 8 Timer timer = new Timer();
- 9 TimerTask timerTask = new TimerTask() {
- 10 @Override
- 11 public void run() {
- 12 System.out.println("当前时间");//具体的任务
- 13 }
- 14 };
- 15 timer.schedule(timerTask1,1000,1000);//任务,开始时间ms,间隔ms
- 16 System.out.println("start");
- 17 try {
- 18 Thread.sleep(2000);//不停两秒,显示不出来,因为test情况下当前用户线程结束,可是定时任务1s还没开始,就被终止
- 19 }catch (Exception e) {
- 20
- 21 }
- 22 System.out.println("end");
- 23 }
- 24 }
终止timer的方式
- 调用timer的cancle方法,
- 把timer线程设置成daemon线程,(new Timer(true)创建daemon线程),在jvm里,如果所有用户线程结束,那么守护线程也会被终止,不过这种方法一般不用。
- 当所有任务执行结束后,删除对应timer对象的引用,线程也会被终止。
- 调用System.exit方法终止程序
注意点
- 每一个Timer仅对应唯一一个线程。
- Timer不保证任务执行的十分精确, schedule如果某一次调度时间比较长,那么后面的时间会顺延和scheduleAtFixedRate(严格按照调度时间来的,如果某次调度时间太长了,那么会通过缩短间隔的方式保证下一次调度在预定时间执行)会有不同的时间差。
- Timer类的线程安全的
- jdk1.5之后,ScheduledExecutorService代替了Timer来实现任务调度,加入了线程池等特性。
二、spring的Task
Task底层的实现,使用的ScheduledExecutorService。
2.1、注解的形式
使用@EnableScheduling启动定时任务注解解析,之后@Schedule写在执行的任务上即可
[cron表达式详解](https://www.jianshu.com/p/1defb0f22ed1)
2.2、直接代码的形式
- 1 @EnableScheduling
- 2 public class Task {
- 3
- 4 @Scheduled(zone = "Asia/Beijing",cron = "0/10 * * * * * *")//zone表示时区
- 5 public void schedule4(){
- 6 }
- 7 /*
- 8 fixedDelay对应的fixedDelayString支持字符串形式、占位符${}|#{}
- 9 */
- 10 @Scheduled(fixedDelay = 5000)//上一次执行完毕时间点之后多长时间再执行
- 11 public void schedule1(){
- 12 }
- 13 @Scheduled(fixedRate = 5000)//上一次开始执行时间点之后多长时间再执行
- 14 public void schedule2(){
- 15 }
- 16 @Scheduled(initialDelay = 5000)//第一次延迟多长时间后再执行
- 17 public void schedule3(){
- 18 }
- 19
- 20 }
TaskScheduler的子类
ConcurrentTaskScheduler:以当前线程执行任务。如果任务简单,可以直接使用这个类来执行。快捷方便。
DefaultManagedTaskScheduler:以当前线程执行任务,这是ConcurrentTaskScheduler的子类,添加了JNDI的支持。和ConcurrentTaskScheduler一样的用法,需要使用JNDI可以单独设置
ThreadPoolTaskScheduler:TaskScheduler接口的默认实现类,多线程定时任务执行。<font color=#0075EA>可以设置执行线程池数(默认一个线程),使用前必须得先调用initialize()【初始化方法】,有shutDown()方法,执行完后可以关闭线程</font>。
TimerManagerTaskScheduler:用于包装CommonJ中的TimerManager接口。在使用CommonJ进行调度时使用。(没有使用过)
三、Quartz
四、elastic-job分布式定时任务
基于Zookepper和Quartz开发,并且开源的Java分布式定时任务,解决Quartz不支持分布式的弊端,Elastic-Job采用分片的方式,是分布式调度解决方案。适用场景是:相对于流程比较简单,但是任务可以拆分到多个线程去执行。 每个任务都使用独立的线程池。
- 1 //注册zookeeper
- 2 @Bean(initMethod = "init")
- 3 public ZookeeperRegistryCenter regCenter( final String serverList,final String namespace) {
- 4 return new ZookeeperRegistryCenter(new ZookeeperConfiguration(serverList, namespace));
- 5 }
- 1 //配置任务config
- 2 @Configuration
- 3 public class LoginOverdueJobConfig {
- 4 @Resource
- 5 private ZookeeperRegistryCenter regCenter;
- 6
- 7 @Resource
- 8 private JobEventConfiguration jobEventConfiguration;
- 9
- 10 private LiteJobConfiguration getLiteJobConfiguration(final Class<? extends SimpleJob> jobClass, final String cron, final int shardingTotalCount, final String shardingItemParameters) {
- 11 return LiteJobConfiguration.newBuilder(new SimpleJobConfiguration(JobCoreConfiguration.newBuilder(
- 12 jobClass.getName(), cron, shardingTotalCount).shardingItemParameters(shardingItemParameters).build(), jobClass.getCanonicalName())).overwrite(true).build();
- 13 }
- 14
- 15 @Bean(initMethod = "init")
- 16 public JobScheduler simpleCloudStorageJobScheduler(final LoginOverdueJob loginOverdueJob, @Value("${simpleJob.cron}") final String cron, @Value("${simpleJob.shardingTotalCount}") final int shardingTotalCount,
- 17 @Value("${simpleJob.shardingItemParameters}") final String shardingItemParameters) {
- 18 return new SpringJobScheduler(loginOverdueJob, regCenter, getLiteJobConfiguration(loginOverdueJob.getClass(), cron, shardingTotalCount, shardingItemParameters), jobEventConfiguration);
- 19 }
- 20 }
- 1 //任务执行函数
- 2 @Component
- 3 public class LoginOverdueJob implements SimpleJob {
- 4 public void execute(ShardingContext shardingContext) {
- 5
- 6 }
- 7 }
java目前常用的几种定时任务的更多相关文章
- Java中常用的四种线程池
在Java中使用线程池,可以用ThreadPoolExecutor的构造函数直接创建出线程池实例,如何使用参见之前的文章Java线程池构造参数详解.不过,在Executors类中,为我们提供了常用线程 ...
- java中常用的几种缓存类型介绍
在平时的开发中会经常用到缓存,比如locache.redis等,但一直没有对缓存有过比较全面的总结.下面从什么是缓存.为什么使用缓存.缓存的分类以及对每种缓存的使用分别进行分析,从而对缓存有更深入的了 ...
- java最常用的几种加密算法
1. BASE64 Base64是网络上最常见的用于传输8Bit字节代码的编码方式之一,大家可以查看RFC2045-RFC2049,上面有MIME的详细规范.Base64编码可用于在HTTP环境下传递 ...
- java map常用的4种遍历方法
public static void main(String[] args) { Map<String, String> map = new HashMap<String, Stri ...
- Java中常用的6种排序算法详细分解
排序算法很多地方都会用到,近期又重新看了一遍算法,并自己简单地实现了一遍,特此记录下来,为以后复习留点材料. 废话不多说,下面逐一看看经典的排序算法: 1. 选择排序 选择排序的基本思想是遍历数组的过 ...
- java常用的几种线程池比较
1. 为什么使用线程池 诸如 Web 服务器.数据库服务器.文件服务器或邮件服务器之类的许多服务器应用程序都面向处理来自某些远程来源的大量短小的任务.请求以某种方式到达服务器,这种方式可能是通过网络协 ...
- java线程池与五种常用线程池策略使用与解析
背景:面试中会要求对5中线程池作分析.所以要熟知线程池的运行细节,如CachedThreadPool会引发oom吗? java线程池与五种常用线程池策略使用与解析 可选择的阻塞队列BlockingQu ...
- java算法03 - 常用的8种排序算法
Java常用的八种排序算法: 插入排序 - 直接插入排序 每次将待排序的记录按照关键字的大小,插入到前面已经排好序的记录的适当位置.直到全部记录插入完成. 代码实现 /** * 直接插入排序 O(n^ ...
- java 获取键盘输入常用的两种方法
java 获取键盘输入常用的两种方法 方法1: 通过 Scanner Scanner input = new Scanner(System.in); String s = input.nextLine ...
随机推荐
- Android 对接硬件串口篇
简介:硬件设备有IC卡片,指压测试仪(测试脉搏信号.心率.血压),经过串口获取硬件设备发送的数据. 正文:第一步:获得硬件设备,应用市场下载串口调适软件,测试一下在Android环境下数据是否能正常获 ...
- 【C#】根据开始时间和结束时间筛选存在的信息
背景 业务需求中,需要根绝开始时间和结束时间筛选一段时间内的任务存在个数. 示例图片 根据开始时间 9:00到 结束时间11:00 筛选信息 总共有这么四种情况可能出现 插入测试数据 CREATE T ...
- 并发编程AQS----共享锁
Semaphore Semaphore 字面意思是信号量的意思,它的作用是控制访问特定资源的线程数目.应用场景:资源访问,服务限流. Semaphore 实现AbstractQueuedSynchro ...
- EOJ Monthly 2019.11 A(进制转换)
"欢迎您乘坐东方航空公司航班 MU5692 由银川前往上海......" "我们的飞机很快就要起飞了,请收起小桌板,摘下耳机......" 收起了小桌板,摘下了 ...
- 基于Scrapy的B站爬虫
基于Scrapy的B站爬虫 最近又被叫去做爬虫了,不得不拾起两年前搞的东西. 说起来那时也是突发奇想,想到做一个B站的爬虫,然后用的都是最基本的Python的各种库. 不过确实,实现起来还是有点麻烦的 ...
- 云原生时代高性能Java框架—Quarkus(二)
--- *构建Quarkus本地镜像.容器化部署Quarkus项目* Quarkus系列博文 Quarkus&GraalVM介绍.创建并启动第一个项目 构建Quarkus本地镜像.容器化部署Q ...
- 解决element上传功能清除单个文件的问题
今天,在使用 element 实现一个上传文件的功能. 接下来,要对上传的文件列表,实现删除单文件的功能. 看了 element 开发文档,发现 on-remove 没有特别的详细的说明,刚开始使用 ...
- python的常用模块
一.random随机数模块 使用随机数模块需要导入随机数模块import random 1.random.random() 生成[0,1)之间的随机小数 2.random.randint(a,b) 生 ...
- Android中Fragment生命周期和基本用法
1.基本概念 1. Fragment是什么? Fragment是可以让你的app纵享丝滑的设计,如果你的app想在现在基础上性能大幅度提高,并且占用内存降低,同样的界面Activity占用内存比Fra ...
- Kaggle 入门题-泰坦尼克号灾难存活预测
这个题目的背景概况来讲就是基于泰坦尼克号这个事件,然后大量的人员不幸淹没在这个海难中,也有少部分人员在这次事件之中存活,然后这个问题提供了一些人员的信息如姓名.年龄.性别.票价,所在客舱等等一些信息, ...