Quartz集群增强版_00.How to use?(如何使用)

news/2024/11/14 18:17:06/文章来源:https://www.cnblogs.com/funnyzpc/p/18540378

Quartz集群增强版_00.How to use?(如何使用)

转载请著名出处 https://www.cnblogs.com/funnyzpc/p/18540378

开源地址 https://github.com/funnyzpc/quartz

表的基本结构

    总的来说任务的配置及开发基本遵从上图的表的基本关系,除 app 以及 node 之外均需要手动手动配置,appnode 在执行端启动的时候会自动生成对应 app 以及 node 的数据 ~

后管配置

先看一下后管的基本页面~
因为 appnode 是一对多的关系,这里就放到一个page下:

  • 这里需要说明的是appnode一般无需新增,如果特殊情况下请参照下图:

app新增

node新增

因为node必须关联已有的app才可新增,新增入口在app列表

另外,需要说明的是:

  • 如果执行端获取不到宿主机IP以及主机名称会随机生成一个同名的 主机IP以及主机名称,此时在管理端手动新增就毫无意义了

    删除

  • 删除应用必须先删除应用关联的节点(node),节点被删除则节点对应的执行端无法执行其任务,删除应用也是

  • 删除应用或节点不会变更任务及执行项的状态,也不会删除任务及执行项,没有节点的执行项不会执行也会定期被清理

    启用/关闭

启用与关闭只操作节点或应用,关闭节点则节点下的所有任务均不会执行,关闭应用则应用关联的所有结点都不会执行任务,同时这个操作也不会变更任务或执行项~

再看看节点任务及执行配置:

任务/执行配置是管理端主要任务,执行配置使用关联任务配置(PID)关联相应的任务(job),执行项(execute)是不可独立存在的!

新增任务配置

  • 应用名称/调度名称就是自动或手动配置的应用信息
    任务状态在配置时仅可有 初始化(INIT)/正常执行(EXECUTING) 这两种状态,如果只是配置不想立即执行就选 初始化(INIT)

    新增执行配置-CRON时间任务

  • 任务类型仅可为简单任务(SIMPLE)或表达式(CRON)的时间项的任务,两种类型的执行配置(execute)填写的字段会有区别
    CRON任务CRON表达式是必填项,时区现阶段默认是Asia/Shanghai ,后续会改成从系统获取默认
    开始时间一般不填则默认就是-1,新增提交后是按当前时间补充
    结束时间也是非必填的,结束时间默认也是-1,结束时间如果是-1则在执行完最后一次任务之后会补充为最后一次执行时间

    新增执行配置-SIMPLE时间任务

  • 图中圈出的为必填项,需要说明的是:如果执行结束时间执行次数均设置,具体任务执行时会依限制范围最小的为实际执行,比如设置的结束时间较长但是执行次数只有几次,那最终大概率只会以执行次数为限制执行

    另外,对于执行配置,当执行完成后,对应的执行配置仅可删除不可 修改或启停,已经完成的对此类操作是没有意义的,不如新增一个执行配置

管理端开发配置及集成

这里仅以springboot为例:

  • 添加依赖,如果有maven私服建议放到私服
    <dependency><groupId>org.quartz-scheduler.internal</groupId><artifactId>quartz-client</artifactId><version>2.3.2</version><!-- 这是本地引入,建议放到私服--><scope>system</scope><systemPath>${pom.basedir}/src/main/resources/lib/quartz-client-2.3.2.jar</systemPath></dependency>
  • 启动类需要排除自动装配
// 这一行是重点!
@SpringBootApplication(exclude = {QuartzAutoConfiguration.class})
public class MeeAdminApplication {/*** 日志*/private static final Logger LOG= LoggerFactory.getLogger(MeeAdminApplication.class);public static void main(String[] args)throws Exception {ConfigurableApplicationContext application = SpringApplication.run(MeeAdminApplication.class, args);Environment env = application.getEnvironment();String ip = InetAddress.getLocalHost().getHostAddress();String port = env.getProperty("server.port");String path = env.getProperty("server.servlet.context-path");LOG.info("\n\t----------------------------------------------------------\n\t" +"Application MeeAdminApplication is running!\n\t" +"Local: \t\thttp://localhost:" + port + path + "/\n\t" +"External: \thttp://" + ip + ":" + port + path + "/\n\t" +"----------------------------------------------------------");}}
  • 需要配置一个实例以使用
@Service
public final class QrtzJobServiceImpl implements QrtzJobService {/***   日志*/private static final Logger LOG = LoggerFactory.getLogger(QrtzJobServiceImpl.class);/*** quartz定时任务api*/private final Scheduler scheduler;public QrtzJobServiceImpl(DataSource dataSource) {this.scheduler = new StdScheduler(dataSource);}
}
  • 调用sdk
    @Overridepublic MeeResult<Integer> updateJobState(String job_id,String state) {Object[] result = scheduler.updateJobStateInAll(job_id,state);int updateCount = (int)result[0];if(updateCount>0){return ResultBuild.build(updateCount);}else{return ResultBuild.fail((String)result[1]);}}

Scheduler 提供了多种多样的api,注意部分接口的区别:

    如果管理端执行端一体 则无需引入client依赖(quartz-client),也无需在启动类中排除自动装配(QuartzAutoConfiguration),使用sdk也无需使用构造方式传入database,仅此即可:

    @Autowiredprivate Scheduler scheduler;

执行端开发配置及集成

  • 引入依赖同时排除原生Quartz
        <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-quartz</artifactId><version>${spring-boot-current.version}</version><exclusions><exclusion><groupId>org.quartz-scheduler</groupId><artifactId>quartz</artifactId></exclusion></exclusions></dependency><dependency><groupId>org.quartz-scheduler.internal</groupId><artifactId>quartz-core</artifactId><version>2.3.2</version><!-- 这是本地引入,建议放到私服--><scope>system</scope><systemPath>${pom.basedir}/src/main/resources/lib/quartz-core-2.3.2.jar</systemPath></dependency>
  • 添加依赖配置项
### ----------- quartz ------------------
spring.quartz.job-store-type=jdbc
spring.quartz.properties.org.quartz.jobStore.class=org.springframework.scheduling.quartz.LocalDataSourceJobStore
spring.quartz.properties.org.quartz.jobStore.clusterCheckinInterval=6000
spring.quartz.properties.org.quartz.jobStore.driverDelegateClass=org.jdbcjobstore.impl.org.quartz.StdJDBCDelegate
spring.quartz.properties.org.quartz.jobStore.isClustered=true
# 表名前缀
spring.quartz.properties.org.quartz.jobStore.tablePrefix=QRTZ_
spring.quartz.properties.org.quartz.scheduler.instanceName=${spring.application.name}
#spring.quartz.properties.org.quartz.threadPool.class=org.quartz.simpl.SimpleThreadPool
spring.quartz.properties.org.quartz.threadPool.class=org.quartz.impl.MeeThreadPool
# 线程数配置
spring.quartz.properties.org.quartz.threadPool.threadCount=10
spring.quartz.properties.org.quartz.threadPool.threadPriority=5
# 綫程继承初始化线程的上下文类加载器
spring.quartz.properties.org.quartz.threadPool.threadsInheritContextClassLoaderOfInitializingThread=true
#Whether to enable pessimistic lock to control trigger concurrency in the cluster 是否启用悲观锁来控制集群中的触发并发
spring.quartz.properties.org.quartz.jobStore.acquireTriggersWithinLock=true

    配置项里面 要注意线程数的配置,如果使用的 MeeThreadPoolthreadCount为最大线程数,核心线程数 threadCount-2 ,最少为2,具体多少按实际CPU核心个数以及是否是IO密集型还是CPU密集型来配置即可~
其次要注意 tablePrefix 如果表名有变更则按照变更后的表名前缀配置即可

  • 定义一个任务
    • 如果使用的是spring提供的QuartzJobBean来开发:
      import com.mee.quartz.util.DateUtil;import org.quartz.JobExecutionContext;import org.quartz.JobExecutionException;import org.quartz.impl.QrtzExecute;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.scheduling.quartz.QuartzJobBean;import javax.sql.DataSource;public class ATestJob extends QuartzJobBean {private static final Logger log = LoggerFactory.getLogger(ATestJob.class);@Overrideprotected void executeInternal(JobExecutionContext context) throws JobExecutionException {try {log.info("===>ATestJob::executeInternal {}-{} : {}-{}<===" ,context.getJobId(),context.getExecuteId(),context.getJobType(),context.getJobClassName());} catch (Exception e) {throw new JobExecutionException(e);}}}
    
    • 如果使用的是Quartz提供的Job接口来开发,也可:
      import org.quartz.Job;import org.quartz.JobExecutionContext;import org.quartz.JobExecutionException;import org.quartz.impl.QrtzExecute;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import java.util.concurrent.TimeUnit;public class Job01TestService  implements Job {private static final Logger LOGGER = LoggerFactory.getLogger(Job01TestService.class);@Overridepublic void execute(JobExecutionContext context) throws JobExecutionException {LOGGER.info("=>>{}-{}.{}-{}",context.getJobId(),context.getExecuteId(),context.getJobType(),context.getJobClassName());}
    }

以上两种方式皆可,需要注意的是,不管是继承 QuartzJobBean 还是实现的 ``Job,均无需将类著名为spring bean类(@Service or @Component),Quartz内部自会创建任务类为spring bean ~

开发注意事项

  • 使用 quartz-client 添加的任务一般最晚会在 5秒 之后执行,因为任务轮询是 5秒 一轮询
  • 执行端执行异常(Quartz内的非业务的)的任务最晚在15S之后恢复任务执行,因为集群/缺火处理是 15秒 一轮询
  • 添加的任务如果不执行首先则要注意 spring.quartz.properties.org.quartz.scheduler.instanceName 配置项是否有配置,这个配置项对应 app 表中的 application 字段
  • 实际任务如有日志出现 任务延迟,建议排查宿主机资源是否占满,或者线程数配置是否合理

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

开源三代示波器的高速波形刷新方案开源,支持VNC远程桌面,手机,Pad,电脑均可访问(2024-11-11)

说明: 1、本来这段时间是一年一度Hackaday硬件设计开源盛宴,但hackaday电子大赛在去年终结了。所以我开源个我的吧。 2、三代示波器的高速波形刷新方案,前两年就做好了,这两年忙H7-TOOL的更新比较多,三代示波器的更新就搁置了。但刷新方案是没问题的,开源分享给大家。 3、…

PSQL 环境安装配置

准备工作:安装包 plsqldev1400x64.msi 注册码 汉化包 chinese.exe 轻量级数据 instantclient_11_2 安装【PSQL】第一步大法操作! 默认的安装路径:C:\Program Files\PLSQL Developer 14安装【轻量级 instantclient_11_2】 复制或解压到 C:\Program Files\PLSQL Developer…

零声QT学习 一

int main(int argc, char *argv[]) {QApplication a(argc, argv);//QApplication a(argc, argv),针对QWidget应用程序,管理和设置Qt程序的运行//QGuiApplication a(argc, argv),针对非QWidget应用程序,如QQuick//QcoreApplication a(argc, argv),针对无界面的应用程序MainWindo…

【教程】第四章:任务与评论插件 —— 如虎添翼,顺利掌握

一起在 NocoBase 中创造精彩应用!这些教程将通过手把手的操作,帮助你全面掌握核心功能,激发灵感,打造并分享满足多样需求的应用。回顾上一节 小伙伴们还记得上一节的挑战任务吗?我们要为任务表配置 状态 和 附件 字段,并在任务列表里展示它们。别急,咱们先揭晓答案!状态…

wsl2踩坑日记(配置代理/zsh+p10k/Neovim)

1. proxy wsl --install Ubuntu-24.04 安装好 wsl 之后,测试了一下 v2rayN 的代理能不能正常使用(用 vultr 服务器搭建的校园网 ipv6 免流),发现可以 curl www.google.com,但是 sudo apt-get update 报错 Clearsigned file isnt valid, got NOSPLIT (does the network req…

基于MIMO系统的SDR-AltMin混合预编码算法matlab性能仿真

1.算法仿真效果 matlab2022a仿真结果如下(完整代码运行后无水印): 仿真操作步骤可参考程序配套的操作视频。2.算法涉及理论知识概要基于MIMO(Multiple-Input Multiple-Output)系统的SDR-AltMin混合预编码算法是一种先进的无线通信技术,它结合了凸优化和交替最小化技术来…

前端技术对html中块级元素的学习

块级元素目录块级元素列表元素有序列表无序列表自定义列表 列表元素 有序列表 在 HTML 中, 标签用来表示有序列表。有序列表之间的内容有先后顺序之分,例如菜谱中的一系列步骤,这些步骤需要按顺序完成,这时就可以使用有序列表。 我们来看一个简单的实例: <!DOCTYPE ht…

IDEA-idea激活

通过百度网盘分享的文件:IDEA激活工具 链接:https://pan.baidu.com/s/18QIqrMVE4ScNhBjhwde_7Q 提取码:sky1二、重启电脑

说明与笔记导航

对使用这些笔记的同学想说的话,以及更新进度。为什么写这么多B东西? 其一呢是帮助我自己,边写笔记边梳理知识;其二呢是帮助各位义父义母考试成功。 更新进度与内容说明 11.11:本周工作日需突击学习python,计划今晚更新有限体积法剩余部分。 目前进度:3009 建模:数值方法…

鸿蒙NEXT开发案例:指尖轮盘

【1】引言 “指尖轮盘”是一个简单而有趣的互动游戏(类似抓阄),这个应用通过触摸屏幕的方式,让玩家参与一个激动人心的游戏,最终选出幸运的赢家。未来可以进一步扩展功能,如增加游戏模式、优化动画效果、增加音效等,提升用户体验。 【2】环境准备 电脑系统:windows 10 …

antD——Warning: `callback` is deprecated. Please return a promise instead.

参考: 1. https://blog.csdn.net/huochai770880/article/details/125925665我的情况 antD表单校验,代码未报错,但提交时控制台报错:Warning: `callback` is deprecated. Please return a promise instead.原报错代码:const validateParams = useCallback((_: RuleObject, …

Qml 中的那些坑(七)---ComboBox嵌入Popup时,滚动内容超过其可见区域不会关闭ComboBox弹窗

最近在写信息提交 ( 表单 ) 的窗口时发现一个奇怪的 BUG: 可以看到,当 `ComboBox` 嵌入 `Popup` 时,点开 `ComboBox`,然后滚动内容超过其可见区域并不会关闭 `ComboBox` 弹窗,并且会超出其 `父 Popup` 范围。【写在前面】 最近在写信息提交 ( 表单 ) 的窗口时发现一个奇怪…