定时任务调度方案——Xxl-Job

news2024/12/25 8:56:00

定时任务调度方案

随着系统规模的发展,项目的组织结构以及架构越来越复杂,业务覆盖的范围越来越广,定时任务数量日益增多,任务也变得越来越复杂,尤其是为了满足在用户体量日历增大时,系统能够稳定运行,我们往往会扩充服务器做集群,无论是传统垂直项目还是如今主流的分布式。那么对定时任务的要求也逐渐变高,基于现在项目主流架构,定时任务需满足一下要求:

    1. 任务统一管理,提供图形化界面对任务进行配置和调度。
    1. 保证任务调度的幂等性(任务并发控制,同一个任务在同一时间只能允许一个执行)
    1. 任务弹性扩容,可根据繁忙情况动态增减服务器分摊压力,对大任务进行分片处理。
    1. 任务依赖问题,能够处理任务包含子任务的情况,前一个完成后触发子任务执行。
    1. 支持多类型的任务,支持Spring Bean、Shell等。
    1. 任务节点高可用,任务节点异常或者繁忙时能够转移到其他节点执行。
    1. 调度中心高可用,支持集群部署,避免出现单点故障。
    1. 执行状态监控,方便查看任务执行状态,异常情况告警,支持多渠道通知。

#. 发展史:
定时任务随着技术发展,从单线程调度到多线程调度,从单机部署到集群部署,从独立执行到多任务协同执行。请添加图片描述

1. Thread

通过线程休眠实现,由JDK提供

private static int count = 0;

public static void main(String[] args) {
  Runnable runnable = new Runnable() {
    @Override
    public void run() {
      while (count < 8) {
        try {
          Thread.sleep(1000);
          System.out.printf("执行第:%d 次\n", ++count);
        } catch (InterruptedException e) {
          e.printStackTrace();
        }
      }
    }
  };
  new Thread(runnable).start();
}

请添加图片描述

2. 线程池

有延缓执行实现,由JDK提供

private static int count = 0;

public static void main(String[] args) throws InterruptedException {
  for (int i = 0; i < 8; i++) {
    ExecutorService executorService = Executors.newSingleThreadScheduledExecutor();
    Runnable runnable = new Runnable() {
      @Override
      public void run() {
        ++count;
        System.out.printf("执行从 %d 次\n", count);
        if (count > 7) {
          System.exit(0);
        }
      }
    };
    // 也可以使用 ScheduledExecutorService.schedule
    executorService.awaitTermination(1L, TimeUnit.SECONDS);
    executorService.execute(runnable);
  }
}

请添加图片描述

3. Timer TimeTask

由JDK提供,多用于移动端(如安卓开发)

private static int count = 0;

public static void main(String[] args) {
    TimerTask timerTask = new TimerTask() {
        @Override
        public void run() {
            System.out.printf("执行第 %d 次\n", ++count);
            if (count > 7) {
                System.exit(0);
            }
        }
    };
    Timer timer = new Timer();
    timer.schedule(timerTask, 500, 500);
}

请添加图片描述

4. Schedule

由Spring提供, 配合@EnableScheduling 使用

@Component
public class SpringScheduleTest {

    private static int count;

    @Scheduled(cron = "0/1 * * * * ?")
    public void testSchedule() {
        System.out.printf("执行第 %d 次\n", ++count);
    }
}

cron 表达式:秒-分-时-日-月-周-年

5.quartz

有maven独立依赖,在SpringBoot2.0之后自带包

  1. 纯maven项目依赖
    <!-- 核心包 -->
    <dependency>
       <groupId>org.quartz-scheduler</groupId>
       <artifactId>quartz</artifactId>
       <version>2.3.0</version>
    </dependency>
    
    <!-- 工具包 -->
    <dependency>
       <groupId>org.quartz-scheduler</groupId>
       <artifactId>quartz-jobs</artifactId>
       <version>2.3.0</version>
    </dependency>
    
  2. SpringBoot2.0及上依赖
    <dependency>
    	<groupId>org.springframework.boot</groupId>
    	<artifactId>spring-boot-starter-quartz</artifactId>
    </dependency>
    
  3. 测试类 Test.java
    public class Test {
    
        public static void main(String[] args) throws SchedulerException, ParseException {
            Scheduler factory = StdSchedulerFactory.getDefaultScheduler();
            JobDataMap jobDataMap = new JobDataMap();
            jobDataMap.put("tips", "去你大爷的");
            JobDetail jobDetail = JobBuilder.newJob(QuartzJob.class)
                    .withIdentity("testJob", "testJobGroup")
                    .setJobData(jobDataMap).build();
            Trigger trigger = TriggerBuilder.newTrigger()
                    .withIdentity("testTrigger", "testTriggerGroup")
                    .withSchedule(SimpleScheduleBuilder.repeatSecondlyForever(5))
                    .startNow().build();
            factory.scheduleJob(jobDetail, trigger);
            factory.start();
        }
    }
    
  4. 创建任务类 QuartzJob.java
    public class QuartzJob implements Job {
        public static int count;
        @Override
        public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException {
            JobDataMap jobDataMap = jobExecutionContext.getMergedJobDataMap();
            String tips = jobDataMap.getString("tips");
            System.out.printf("获取到参数:%s\n", tips);
            System.out.printf("执行第:%d 次\n", ++count);
        }
    }
    
    请添加图片描述

6. Xxl-job

XXL-JOB 是一个轻量级分布式任务调度平台,其核心设计目标是开发迅速、学习简单、轻量级、易扩展。现已开放源代码并接入多家公司线上产品线,开箱即用。
请添加图片描述
详细内容介绍,请查看:文档地址

  1. 创建库,并创建表,sql文件在项目目录下:xxl-job-master/doc/db/tables_xxl_job.sql
    注意:它的数据库名是 xxl_job, 如果不想创建新库,而是导入到现有的库中,需要进入sql文件中改名字。
    请添加图片描述

  2. 运行项目:xxl-job-admin

    1. 项目结构如下:
      请添加图片描述

    2. 修改mysql 连接配置如下:
      请添加图片描述

  3. 运行起来后, 在浏览器访问:http://localhost:8080/xxl-job-admin

    默认账号:admin 密码:123456

    请添加图片描述

  4. 成功登录后进入调度中心
    请添加图片描述

  5. 注册任务执行器,可以单独创建执行器服务,也可集成到现有项目服务中去
    以下为新建SpringBoot 项目为示例

    1. 创建SpringBoot 项目并添加依赖
      <dependency>
        <groupId>com.xuxueli</groupId>
        <artifactId>xxl-job-core</artifactId>
        <version>2.3.1</version>
      </dependency>
      
    2. 在resources目录下创建application.properties文件(yaml文件也一样,如果有就不用创建了)
      # 执行器所在的服务的运行端口, 非执行器端口
      server.port=8081
      
      
      ### xxl-job admin address list, such as "http://address" or "http://address01,http://address02"
      # 注册地址, 这个地址其实就是部署或者运行完xxl-job-admin调度中心的浏览器访问地址
      xxl.job.admin.addresses=http://127.0.0.1:8080/xxl-job-admin
      
      ### xxl-job, access token
      xxl.job.accessToken=default_token
      
      ### xxl-job executor appname
      ### 执行器名称,需要配置到调度中心
      xxl.job.executor.appname=xxl-job-executor
      ### xxl-job executor registry-address: default use address to registry , otherwise use ip:port if address is null
      xxl.job.executor.address=
      ### xxl-job executor server-info
      xxl.job.executor.ip=
      # 这个端口是执行器端口, 不是这个项目server.port 端口,也不能一样,否则会出现端口占用等错误信息
      # 也就是说,执行器项目部署或者运行起来会有两个端口,如果是用docker部署,注意两个端口都要映射到宿主机端口
      xxl.job.executor.port=9999
      ### xxl-job executor log-path
      xxl.job.executor.logpath=/data/applogs/xxl-job/jobhandler
      ### xxl-job executor log-retention-days
      xxl.job.executor.logretentiondays=30
      
    3. 创建配置类 XxlJobConfig.java
      @Configuration
      public class XxlJobConfig {
      
          @Value("${xxl.job.admin.addresses}")
          private String adminAddresses;
      
          @Value("${xxl.job.accessToken}")
          private String accessToken;
      
          @Value("${xxl.job.executor.appname}")
          private String appname;
      
          @Value("${xxl.job.executor.address}")
          private String address;
      
          @Value("${xxl.job.executor.ip}")
          private String ip;
      
          @Value("${xxl.job.executor.port}")
          private int port;
      
          @Value("${xxl.job.executor.logpath}")
          private String logPath;
      
          @Value("${xxl.job.executor.logretentiondays}")
          private int logRetentionDays;
      
          @Bean
          public XxlJobSpringExecutor xxlJobExecutor() {
              XxlJobSpringExecutor xxlJobSpringExecutor = new XxlJobSpringExecutor();
              xxlJobSpringExecutor.setAdminAddresses(adminAddresses);
              xxlJobSpringExecutor.setAppname(appname);
              xxlJobSpringExecutor.setAddress(address);
              xxlJobSpringExecutor.setIp(ip);
              xxlJobSpringExecutor.setPort(port);
              xxlJobSpringExecutor.setAccessToken(accessToken);
              xxlJobSpringExecutor.setLogPath(logPath);
              xxlJobSpringExecutor.setLogRetentionDays(logRetentionDays);
              return xxlJobSpringExecutor;
          }
      }
      
    4. 创建任务 (Job),注意不同版本写法会有些不一样
      @Component
      public class SampleXxlJob {
      
          /**
           * 1、简单任务示例(Bean模式)
           */
          @XxlJob("demoJobHandler")
          public void demoJobHandler() throws Exception {
              XxlJobHelper.log("XXL-JOB, Hello World.");
              for (int i = 0; i < 5; i++) {
                  XxlJobHelper.log("beat at:" + i);
                  TimeUnit.SECONDS.sleep(2);
              }
              XxlJobHelper.handleSuccess("XXL-JOB:执行成功");
          }
      }
      
    5. 配置执行器
      1. 配置
        请添加图片描述
      2. 成功
        请添加图片描述
    6. 配置任务并执行
      1. 配置任务
        请添加图片描述

      2. 执行一次
        请添加图片描述
        请添加图片描述

      3. 查看日志
        请添加图片描述
        请添加图片描述
        请添加图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/385765.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

汽车改色避坑指南

关于汽车改色&#xff0c;真的不是只看颜色看价格那么简单。 如何选择一家靠谱的改色店&#xff1f;要看这家店是否正规&#xff0c;是否有完备的售后和质保流程&#xff0c;后续剐蹭、磕碰、划痕是否可以修复&#xff1f;是否提前告知注意事项&#xff0c;以及改色后备案流程&…

浅谈车载测试之智能座舱人机交互

1、引言 目前&#xff0c;汽车不仅在动力源、驱动方式和驾驶体验上发生了变化&#xff0c;驾驶舱也告别了传统的枯燥机械和电子空间&#xff0c;智能化水平飙升&#xff0c;成为继家庭和办公室以外人们生活之后的“第三空间”。通过人脸&#xff1b;指纹识别、语音 / 手势交互…

【线程-J.U.C】

Lock J.U.C最核心组件&#xff0c;Lock接口出现之前&#xff0c;多线程的并发安全只能由synchronized处理&#xff0c;但java5之后&#xff0c;Lock的出现可以解决synchronized的短板&#xff0c;更加灵活。 Lock本质上是一个接口&#xff0c;定义了释放锁&#xff08;unlock&…

解决:centos7如何解决网络不可达和wget: 无法解析主机地址 “downloads.mysql.com”

遇到此类问题可能会有多重解决方法&#xff0c;需要一个一个的去排除。 1、查看自己的网络设置是不是设置的NAT模式&#xff0c;设置完成后再去ping一下网络地址。 2、ping一下百度看能不能拼成功&#xff0c;下图是ping成功的样式。&#xff08;如果不行继续往下走&#xff0…

牛客sql题目练习

Sql3描述 题目&#xff1a;现在运营需要查看用户来自于哪些学校&#xff0c;请从用户信息表中取出学校的去重数据。 示例:user_profile iddevice_idgenderageuniversityprovince12138male21北京大学Beijing23214male复旦大学Shanghai36543female20北京大学Beijing42315femal…

PowerShell攻击指南

文章目录一&#xff1a;PowerShell简介1.1&#xff1a;基本概念1.2&#xff1a;执行策略与绕过1.3&#xff1a;常用命令二&#xff1a;PowerSploit2.1&#xff1a;PowerSploit安装2.2&#xff1a;PowerSploit攻击实战2.2.1&#xff1a;直接shellcode反弹meterpreter shell2.2.2…

【C/C++每日一练】总目录(不断更新中...)

C/C 2023.03 20230303 1. 字符串相乘 ★★ 2. 单词拆分 II ★★★ 3. 串联所有单词的子串 ★★★ 20230302 1. 个位数是6&#xff0c;且能被3整除的五位数共有多少个&#xff1f; ☆ 2. 不同方式求n的阶乘 ★ 3. 报数游戏 ★☆ 20230301 1. 冒泡排序法排序 ★ …

视频号频出10w+,近期爆红的账号有哪些?

回顾2月&#xff0c;视频号持续放出大动作&#xff0c;不仅进行了16小时不间断的NBA全明星直播&#xff0c;还邀请国际奥委会入驻&#xff0c;分享奥运的最新资讯。视频号成为越来越多官方机构宣传推广的有效渠道。官方积极入驻&#xff0c;内容创作生态也在同步繁荣发展&#…

中村成洋《垃圾回收的算法与实现》PDF 读书笔记

观前提醒 为了能够锻炼自己&#xff0c;我会查阅大量外文不停的修改内容&#xff0c;少部分会提示成中文。 可能有误&#xff0c;请见谅 提示&#xff1a;若是觉得阅读困难&#xff0c;可以看如下内容 脚本之家可获取&#xff0c;若失效可私信浏览器的沙拉查词扩展&#xf…

Shell脚本学习指南 - 第二章入门篇

shell脚本的第一行#! #! /bin/awk -f 内核会扫描文件开头的#!后面内容&#xff0c;跳过所有空白符号&#xff0c;寻求可以用来执行程序的解释器的full path和option&#xff08;option后面的空格会识别&#xff09; ; shell用分号隔开多条语句 & 后台执行该命令&#xff…

ChatGPT解答:JavaScript保存当前网页页面图片为pdf文件或者word文件,前端用vue2,给出详细的方案和代码

ChatGPT解答&#xff1a;JavaScript保存当前网页页面图片为pdf文件或者word文件&#xff0c;前端用vue2&#xff0c;给出详细的方案和代码 ChatGPTDemo Based on OpenAI API (gpt-3.5-turbo). JavaScript保存当前网页页面图片为pdf文件或者word文件&#xff0c;前端用vue2&am…

Python 操作Redis

在 Python中我们使用 redis库来操作 Redis数据库。Redis数据库的使用命令这里就不介绍了。 需要安装 redis库。检查是否安装redis&#xff1a; pip redis 如果未安装&#xff0c;使用 pip命令安装 redis。 pip install redis #安装最新版本 一、Redis连接 Redis提供两个类 Re…

CEC2021:鱼鹰优化算法(Osprey optimization algorithm,OOA)求解CEC2021(提供MATLAB代码

一、鱼鹰优化算法简介 鱼鹰优化算法&#xff08;Osprey optimization algorithm&#xff0c;OOA&#xff09;由Mohammad Dehghani 和 Pavel Trojovsk于2023年提出&#xff0c;其模拟鱼鹰的捕食行为。 鱼鹰是鹰形目、鹗科、鹗属的仅有的一种中型猛禽。雌雄相似。体长51-64厘米…

2023年“楚怡杯“湖南省职业院校技能竞赛“网络安全”竞赛任务书

2023年“楚怡杯“湖南省职业院校技能竞赛“网络安全”竞赛任务书 一、竞赛时间 总计&#xff1a;360分钟 竞赛阶段竞赛阶段 任务阶段 竞赛任务 竞赛时间 分值 A模块 A-1 登录安全加固 180分钟 200分 A-2 本地安全策略配置 A-3 流量完整性保护 A-4 事件监控 …

技术官方文档中的代码是用什么展示的?代码高亮插件总结

****内容预警****菜鸟教程***大佬绕道我们经常看到各种技术官方文档&#xff0c;有很多代码展示的区域&#xff0c;用于我们复制粘贴代码&#xff0c;比如vue 的官网当我们需要自己实现这么一个网站的时候&#xff0c;我就开始手忙脚乱&#xff0c;这到底是咋实现的&#xff1f…

如何使用ChatGPT快速构建一个网站模板?

欢迎来到令人兴奋的自然语言处理和机器学习世界&#xff01;今天&#xff0c;我们将探索 ChatGPT 的功能&#xff0c;它是由 OpenAI 公司开发的目前最先进的人工智能工具。当然&#xff0c;你也可以将其看作是一个智能机器人。ChatGPT 最令人印象深刻的功能之一是它能够根据简单…

如何获取或设置CANoe以太网网卡信息(SET篇)

CAPL提供了一系列函数用来操作CANoe网卡。但是,但是,首先需要明确一点,不管是获取网卡信息,还是设置网卡信息,只能访问CAPL程序所在的节点下的网卡,而不是节点所在的以太网通道下的所有网卡 关于第一张图中,Class节点下,有三个网卡:Ethernet1、VLAN 1.100、VLAN 1.200…

我的 System Verilog 学习记录(8)

引言 本文简单介绍 SystemVerilog 的接口。 前文链接&#xff1a; 我的 System Verilog 学习记录&#xff08;1&#xff09; 我的 System Verilog 学习记录&#xff08;2&#xff09; 我的 System Verilog 学习记录&#xff08;3&#xff09; 我的 System Verilog 学习记…

哪个牌子的蓝牙耳机音质好?公认音质最好的真无线耳机推荐

现如今&#xff0c;使用蓝牙耳机的人越来越多&#xff0c;更多的蓝牙耳机品牌出现在大众视野。哪个牌子的蓝牙耳机音质好&#xff1f;最近看到很多人问音质。都说蓝牙耳机的音质比不上有线耳机的音质&#xff0c;但经过那么多年的技术进步&#xff0c;蓝牙耳机在音质上也有着不…

CSS - 扫盲

文章目录1. 前言2. CSS2.1 css 的引入方式2.2 选择器2.3 CSS 常用属性2.3.1 字体属性2.3.2 文本属性2.3.3 背景属性2.4 圆角矩形2.5 元素的显示模式2.6 盒子模型2.7 弹性布局1. 前言 上文我们简单 将 HTML 过了一遍 &#xff0c; 知道了 HTML 知识表示页面的结构和内容 &#x…