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的方式

  1. 调用timer的cancle方法,
  2. 把timer线程设置成daemon线程,(new Timer(true)创建daemon线程),在jvm里,如果所有用户线程结束,那么守护线程也会被终止,不过这种方法一般不用。
  3. 当所有任务执行结束后,删除对应timer对象的引用,线程也会被终止。
  4. 调用System.exit方法终止程序

注意点

  1. 每一个Timer仅对应唯一一个线程。
  2. Timer不保证任务执行的十分精确, schedule如果某一次调度时间比较长,那么后面的时间会顺延和scheduleAtFixedRate(严格按照调度时间来的,如果某次调度时间太长了,那么会通过缩短间隔的方式保证下一次调度在预定时间执行)会有不同的时间差。
  3. Timer类的线程安全的
  4. 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的子类

  1. ConcurrentTaskScheduler:以当前线程执行任务。如果任务简单,可以直接使用这个类来执行。快捷方便。

  2. DefaultManagedTaskScheduler:以当前线程执行任务,这是ConcurrentTaskScheduler的子类,添加了JNDI的支持。和ConcurrentTaskScheduler一样的用法,需要使用JNDI可以单独设置

  3. ThreadPoolTaskScheduler:TaskScheduler接口的默认实现类,多线程定时任务执行。<font color=#0075EA>可以设置执行线程池数(默认一个线程),使用前必须得先调用initialize()【初始化方法】,有shutDown()方法,执行完后可以关闭线程</font>。

  4. 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目前常用的几种定时任务的更多相关文章

  1. Java中常用的四种线程池

    在Java中使用线程池,可以用ThreadPoolExecutor的构造函数直接创建出线程池实例,如何使用参见之前的文章Java线程池构造参数详解.不过,在Executors类中,为我们提供了常用线程 ...

  2. java中常用的几种缓存类型介绍

    在平时的开发中会经常用到缓存,比如locache.redis等,但一直没有对缓存有过比较全面的总结.下面从什么是缓存.为什么使用缓存.缓存的分类以及对每种缓存的使用分别进行分析,从而对缓存有更深入的了 ...

  3. java最常用的几种加密算法

    1. BASE64 Base64是网络上最常见的用于传输8Bit字节代码的编码方式之一,大家可以查看RFC2045-RFC2049,上面有MIME的详细规范.Base64编码可用于在HTTP环境下传递 ...

  4. java map常用的4种遍历方法

    public static void main(String[] args) { Map<String, String> map = new HashMap<String, Stri ...

  5. Java中常用的6种排序算法详细分解

    排序算法很多地方都会用到,近期又重新看了一遍算法,并自己简单地实现了一遍,特此记录下来,为以后复习留点材料. 废话不多说,下面逐一看看经典的排序算法: 1. 选择排序 选择排序的基本思想是遍历数组的过 ...

  6. java常用的几种线程池比较

    1. 为什么使用线程池 诸如 Web 服务器.数据库服务器.文件服务器或邮件服务器之类的许多服务器应用程序都面向处理来自某些远程来源的大量短小的任务.请求以某种方式到达服务器,这种方式可能是通过网络协 ...

  7. java线程池与五种常用线程池策略使用与解析

    背景:面试中会要求对5中线程池作分析.所以要熟知线程池的运行细节,如CachedThreadPool会引发oom吗? java线程池与五种常用线程池策略使用与解析 可选择的阻塞队列BlockingQu ...

  8. java算法03 - 常用的8种排序算法

    Java常用的八种排序算法: 插入排序 - 直接插入排序 每次将待排序的记录按照关键字的大小,插入到前面已经排好序的记录的适当位置.直到全部记录插入完成. 代码实现 /** * 直接插入排序 O(n^ ...

  9. java 获取键盘输入常用的两种方法

    java 获取键盘输入常用的两种方法 方法1: 通过 Scanner Scanner input = new Scanner(System.in); String s = input.nextLine ...

随机推荐

  1. 前端03 /css简绍/css选择器

    前端03 /css简绍/css选择器 目录 前端03 /css简绍/css选择器 昨日内容回顾 html标签 常用标签 table标签:表格标签 input标签 select下拉框 textarea多 ...

  2. web CSS3 实现3D旋转木马

    3D 旋转木马是CSS中常见的特效之一,旋转木马可以有多种方法实现,这里我使用纯CSS实现这种动画的效果. 简要介绍一下重点 transform: rotateY(60deg) translateZ( ...

  3. 02-Python运算符

    一.简介 以10 - 5为例,‘10 - 5’叫做表达式,表达式可以分解成运算符和操作数.整数10和5被称为操作数.‘-’称为运算符. 二.算术运算符 运算符 描述 示例 结果 + 加 - 两个对象相 ...

  4. DEX文件解析--7、类及其类数据解析(完结篇)

    一.前言    前置技能链接:       DEX文件解析---1.dex文件头解析       DEX文件解析---2.Dex文件checksum(校验和)解析       DEX文件解析--3.d ...

  5. xenomai内核解析--双核系统调用(三)--如何为xenomai添加一个系统调用

    版权声明:本文为本文为博主原创文章,转载请注明出处.如有错误,欢迎指正. @ 目录 一.添加系统调用 二.Cobalt库添加接口 三.应用使用 一.添加系统调用 下面给xenomai添加一个系统调用g ...

  6. layui 魔改:上传时的真实进度条

    这个问题本身不复杂,难点在于需要改 layui 的源码. HTML略. 网页的JS域: layui.use(['upload','element','layer'], function(){ var ...

  7. pandas | 如何在DataFrame中通过索引高效获取数据?

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是pandas数据处理专题的第四篇文章,我们一起来聊聊DataFrame中的索引. 上一篇文章当中我们介绍了DataFrame数据结构当 ...

  8. C++语法小记---运算符重载

    运算符重载 运算符重载的本质也是对已有功能的扩展 运算符重载的本质就是函数重载,只是函数变成了 operator + 运算符 当成员函数和全局函数对运算符进行重载时,优先调用成员函数 运算符重载为成员 ...

  9. three.js 数学方法之Plane

    今天郭先生就来继续说一说three.js数学方法中的plane(平面).在三维空间中无限延伸的二维平面,平面方程用单位长度的法向量和常数表示.构造器为Plane( normal : Vector3, ...

  10. Teambition如何使用二次验证码/虚拟MFA/两步验证/谷歌验证器?

    一般点账户名——设置——安全设置中开通虚拟MFA两步验证 具体步骤见链接  Teambition如何使用二次验证码/虚拟MFA/两步验证/谷歌验证器? 二次验证码小程序于谷歌身份验证器APP的优势 1 ...