使用mongodb作为Quartz.Net下的JobStore实现底层的持久化机制
我们都知道默认的Quartz底层采用的是RAMJobStore,所有的Job,Trigger,Calendar都是用Dictionary,SortSet等等这样的数据结构进行储存,相对来说性
能肯定快的没法说,但是面对灾难重启的时候还是很拿不出手的,而且都是全内存的,也没法实现多机器搭建Quartz集群,这一点还是很讨厌,虽然官方已经
提供了一些关系性持久化存储方案,但面对如今这么火的nosql,不进行官方支持还是有点可惜,不过基于Quartz本身的插拔式设计,一切都不是问题。
一:IJobStore
从github上下载源码:https://github.com/quartznet/quartznet,从源码你会发现IJobStore几乎实现了所有对Trigger,Job和Scheduler所有的容器管理操作。

然后你可以看到它的几个实现子类,全内存的RAMJobStore。
public class RAMJobStore: IJobStore
{
....
}
以及JobStoreSupport下的带锁JobStoreTX和不带锁的JobStoreCMT。
public class JobStoreSupport: IJobStore
{
....
} //带锁机制
public class JobStoreTX: JobStoreSupport
{
....
} //不带锁
public class JobStoreCMT: JobStoreSupport
{
....
}
所以你应该明白,本节课跟大家讲到的Redis和Mongodb的JobStore存储,必然也是实现了IJobStore接口,对吧。
二:MongoDB的JobStore
1. 安装mongodb
既然要使用mongodb,你必然要有mongodb的安装程序,可以去官网: wget https://fastdl.mongodb.org/linux/mongodb-linux-x86_64-3.4.5.tgz 一下,
这里我采用linux平台的centos。

2. nuget上pull些dll
大家可以在nuget控制台执行Install-Package Quartz.Spi.MongoDbJobStore,如下所示:
PM> Install-Package Quartz.Spi.MongoDbJobStore
正在尝试收集与目标为“.NETFramework,Version=v4.5.2”的项目“ConsoleApplication1”有关的包“Quartz.Spi.MongoDbJobStore.2.0.”的依赖项信息
正在尝试解析程序包“Quartz.Spi.MongoDbJobStore.2.0.”的依赖项,DependencyBehavior 为“Lowest”
正在解析操作以安装程序包“Quartz.Spi.MongoDbJobStore.2.0.”
已解析操作以安装程序包“Quartz.Spi.MongoDbJobStore.2.0.”
正在将程序包“Common.Logging.Core.3.3.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“Common.Logging.Core.3.3.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“Common.Logging.Core.3.3.”添加到“packages.config”
已将“Common.Logging.Core 3.3.”成功安装到 ConsoleApplication1
正在将程序包“Common.Logging.3.3.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“Common.Logging.3.3.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“Common.Logging.3.3.”添加到“packages.config”
已将“Common.Logging 3.3.”成功安装到 ConsoleApplication1
正在将程序包“MongoDB.Bson.2.4.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“MongoDB.Bson.2.4.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“MongoDB.Bson.2.4.”添加到“packages.config”
已将“MongoDB.Bson 2.4.”成功安装到 ConsoleApplication1
正在将程序包“Quartz.2.4.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“Quartz.2.4.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“Quartz.2.4.”添加到“packages.config”
已将“Quartz 2.4.”成功安装到 ConsoleApplication1
正在将程序包“System.Runtime.InteropServices.RuntimeInformation.4.3.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“System.Runtime.InteropServices.RuntimeInformation.4.3.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“System.Runtime.InteropServices.RuntimeInformation.4.3.”添加到“packages.config”
已将“System.Runtime.InteropServices.RuntimeInformation 4.3.”成功安装到 ConsoleApplication1
正在将程序包“MongoDB.Driver.Core.2.4.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“MongoDB.Driver.Core.2.4.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“MongoDB.Driver.Core.2.4.”添加到“packages.config”
已将“MongoDB.Driver.Core 2.4.”成功安装到 ConsoleApplication1
正在将程序包“MongoDB.Driver.2.4.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“MongoDB.Driver.2.4.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“MongoDB.Driver.2.4.”添加到“packages.config”
已将“MongoDB.Driver 2.4.”成功安装到 ConsoleApplication1
正在将程序包“Quartz.Spi.MongoDbJobStore.2.0.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“Quartz.Spi.MongoDbJobStore.2.0.”添加到文件夹“C:\\ConsoleApplication1\packages”
已将程序包“Quartz.Spi.MongoDbJobStore.2.0.”添加到“packages.config”
已将“Quartz.Spi.MongoDbJobStore 2.0.”成功安装到 ConsoleApplication1
也可以到github中下载源码:https://github.com/chrisdrobison/mongodb-quartz-net

3. 启动运行
然后可以看一下此页面上的Basic Usage##上的默认配置:
var properties = new NameValueCollection();
properties[StdSchedulerFactory.PropertySchedulerInstanceName] = instanceName;
properties[StdSchedulerFactory.PropertySchedulerInstanceId] = $"{Environment.MachineName}-{Guid.NewGuid()}";
properties[StdSchedulerFactory.PropertyJobStoreType] = typeof (MongoDbJobStore).AssemblyQualifiedName;
// I treat the database in the connection string as the one you want to connect to
properties[$"{StdSchedulerFactory.PropertyJobStorePrefix}.{StdSchedulerFactory.PropertyDataSourceConnectionString}"] = "mongodb://localhost/quartz";
// The prefix is optional
properties[$"{StdSchedulerFactory.PropertyJobStorePrefix}.collectionPrefix"] = "prefix"; var scheduler = new StdSchedulerFactory(properties);
return scheduler.GetScheduler();
<1> PropertySchedulerInstanceName: 就是对Scheduler的Name进行的配置,大家可以根据情况定义一个简明释义的名字。
<2> PropertySchedulerInstanceId: 可以看到这个项采用的是machineName+NewGuid来保证Scheduler容器的SchedulerID唯一,唯一性特别重要,因为在
Cluster 中就是用它来保证唯一性的,不过上面的代码有点累赘,其实只要写上“AUTO”就可以了,由底层的
SimpleInstanceIdGenerator来保证uniqueID的生成,如StdSchedulerFactory.Instantiate方法源码所示:
if (schedInstId.Equals(AutoGenerateInstanceId))
{
autoId = true;
instanceIdGeneratorType = LoadType(cfg.GetStringProperty(PropertySchedulerInstanceIdGeneratorType)) ?? typeof(SimpleInstanceIdGenerator);
}
else if (schedInstId.Equals(SystemPropertyAsInstanceId))
{
autoId = true;
instanceIdGeneratorType = typeof(SystemPropertyInstanceIdGenerator);
}
<3> PropertyJobStoreType:这个属性将MongoDbJobStore作为底层的IJobStore实现者。
<4> PropertyDataSourceConnectionString,collectionPrefix: 这两个没什么好说的,一个是mongodb的connectionstring,一个是collection的前缀。
好了,下面就是我的完整代码:
static void Main(string[] args)
{ LogManager.Adapter = new Common.Logging.Simple.TraceLoggerFactoryAdapter()
{
Level = LogLevel.All
}; var properties = new NameValueCollection();
properties[StdSchedulerFactory.PropertySchedulerInstanceId] = "AUTO";
properties[StdSchedulerFactory.PropertyJobStoreType] = typeof(MongoDbJobStore).AssemblyQualifiedName; // I treat the database in the connection string as the one you want to connect to
properties[$"{StdSchedulerFactory.PropertyJobStorePrefix}.{StdSchedulerFactory.PropertyDataSourceConnectionString}"] = "mongodb://192.168.23.163/quartz"; // The prefix is optional
properties[$"{StdSchedulerFactory.PropertyJobStorePrefix}.collectionPrefix"] = "prefix"; var factory = new StdSchedulerFactory(properties); //scheduler
IScheduler scheduler = factory.GetScheduler(); scheduler.Start(); var job = JobBuilder.Create<HelloJob>().WithIdentity("test", "datamip").Build(); var trigger = TriggerBuilder.Create().WithCronSchedule("* * * * * ?").Build(); if (!scheduler.CheckExists(job.Key))
{
scheduler.ScheduleJob(job, trigger);
} Console.Read();
}
这个我自定义的HelloJob中,我特意记录一下scheduler的调度时间schedulertime和Trigger应该触发的时间nextFireTime。
class HelloJob : IJob
{
static int index = ; public void Execute(IJobExecutionContext context)
{
Console.WriteLine("{4} index={0},current={1}, scheuler={2},nexttime={3}",
index++, DateTime.Now,
context.ScheduledFireTimeUtc?.LocalDateTime,
context.NextFireTimeUtc?.LocalDateTime,
context.JobDetail.JobDataMap["key"]);
}
}
接下来执行一下:

然后通过robomongo到数据库看一下,有5个collection,里面都有数据,没毛病。

好了,本篇就说到这里了,当然还有基于redis的JobStore,有兴趣大家可以自己尝试一下。
使用mongodb作为Quartz.Net下的JobStore实现底层的持久化机制的更多相关文章
- Quartz.net 3.x使用总结(二)——Db持久化和集群
上一篇简单介绍了Quartz.net的概念和基本用法,这一篇记录一下Quartz.net通过数据库持久化Trigger和Jobs等数据,并简单配置Quartz.net的集群. 1.JobStore介绍 ...
- 痞子衡嵌入式:MCUXpresso IDE下SDK工程导入与workspace管理机制
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是MCUXpresso IDE下SDK工程导入与workspace管理机制. MCUXpresso IDE是恩智浦软件团队倾注很大心血研发 ...
- MongoDB学习笔记—Linux下搭建MongoDB环境
1.MongoDB简单说明 a MongoDB是由C++语言编写的一个基于分布式文件存储的开源数据库系统,它的目的在于为WEB应用提供可扩展的高性能数据存储解决方案. b MongoDB是一个介于关系 ...
- Mongodb在Windows 7下的安装及配置
第一步 下载MongoDB: 下载mongodb的windows版本,有32位和64位版本,根据操作系统情况下载,下载地址:http://www.mongodb.org/downloads 解压缩至指 ...
- MongoDB介绍与windows下安装
MongoDB是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能最丰富,最像关系数据库的.他支持的数据结构非常松散,是类 似json的bjson格式,因此可以存储比较复杂的数据类型. ...
- MongoDb的bin目录下文件mongod,mongo,mongostat命令的说明及使用
MongoDB的下载地址:http://www.mongodb.org/downloads. 下载好直接解压安装包,即可使用. bin目录下的几个文件说明: mongo 客户端程序,连接MongoDB ...
- MongoDB 介绍及Windows下安装
一.MongoDB简介 MongoDB是一个高性能,开源,无模式的文档型数据库,是当前NoSql数据库中比较热门的一种.它在许多场景下可用于替代传统的关系型数据库或键/值存储方式.Mongo使用C++ ...
- MongoDB深圳用户组线下活动召集
MongoDB线下用户组是由全世界MongoDB爱好者发起的不定期线下交流活动.目前全球有100多个MongoDB用户组,3万5千多爱好者参与.用户组活动的形式通常会有一到两个MongoDB相关的技术 ...
- MongoDB在Linux系统下的安装与启动
Mongodb介绍 MongoDB是一个开源文档数据库,提供高性能,高可用性和自动扩展,官方文档:https://docs.mongodb.com/manual/introduction/ Mongo ...
随机推荐
- C#基础知识-基本的流程控制语句(三)
所谓的流程控制就是在程序运行中控制程序的走向,可以通过各种的条件判断执行代码的顺序,有if... if...else.. else...if |switch case...|while... Do.. ...
- 堆和栈(java内存)
栈内存: 在函数中定义的一些基本类型的变量和对象的引用变量都是在函数的栈内存中分配.当在一段代码块中定义一个变量时,java就在栈中为这个变量分配内存空间,当超过变量的作用域后,java会自动释放掉为 ...
- hdu1198 Farm Irrigation 并查集
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1198 简单并查集 分别合并竖直方向和水平方向即可 代码: #include<iostream&g ...
- InputStream类详解
InputStream这个抽象类是所有基于字节的输入流的超类,抽象了Java的字节输入模型.在这个类中定义了一些基本的方法.看一下类的定义: public abstract class InputSt ...
- Apache solr(二)
上一篇试着进行了solr的安装和配置,以及如何solr的检索,今天试着简单的将solr连接MySQL数据库(才尝试了单表.一对多和多对多的还有待研究) 1.MySQL的目录结构 2.新建一个democ ...
- CentOS 7.2下安装Mono 5.0
微软Build2017大会期间.NET领域的.NET core之外,就是Visual Studio For Mac,大家都知道Visual Studio For Mac 是基于Mono运行的,Mono ...
- 019 关联映射文件中集合标签中的lazy(懒加载)属性
<set>.<list>集合上,可以取值:true/false/extra,(默认值为:true) 实例一:(集合上的lazy=true(默认))class默认lazy=tru ...
- FPGA两种寄存器的使能
在FPGA中,寄存器的使能设计一般有两种方式: 1.直接使用寄存器的使能端口. 2.使用一个数据选择器连接寄存器的D端口,通过数据选择器的sel端口做使能.如下图 这个方式与直接使用寄存器的CE端口有 ...
- Java ClassLoader 原理分析
一.ClassLoader(类加载器)的作用 如果一个程序包含不止一个class文件,那么当程序启动时,带有main方法的类的class文件作为程序入口先被JVM加载,然后根据程序调用的需要,再逐步进 ...
- 关于微信小程序的的总结
微信小程序学完了,给大家分享一些自己学小程序的心得,希望能帮到大家. 首先,我谈谈小程序数据绑定的那一块,所有从本地或者远程服务器的API传过来,都必须绑定到data: {}, 绑定格式是一个一个的键 ...