Quartz.NET开源作业调度框架系列(一):快速入门step by step

  Quartz.NET是一个被广泛使用的开源作业调度框架 , 由于是用C#语言创建,可方便的用于winform和asp.net应用程序中。Quartz.NET提供了巨大的灵活性但又兼具简单性。开发人员可用它快捷的创建并执行一个自动化作业。Quartz.NET有很多特征,如:数据库支持,集群,插件,支持cron-like表达式等。

1 为什么选择Quartz.NET

      在大部分的应用中,都需要对数据库进行定期备份 , 这个备份任务可以是每天晚上12:00或者每周星期二晚上12:00,或许仅仅每个月的最后一天执行。如果应用程序中需要这样的自动化执行任务 , 那么建议使用Quartz.NET调度器作为框架,为我们提供基础服务。 Quartz.NET允许开发人员根据时间间隔(或天)来调度作业 , 它实现了作业和触发器的多对多关系,还能把多个作业与不同的触发器关联。整合了 Quartz.NET的应用程序可以重用来自不同事件的作业,还可以为一个事件组合多个作业.

2 如何创建一个简单的Quartz.NET

2.1 创建桌面应用程序并添加类库

  用VS2012创建一个QuartzDemo的桌面应用程序 , 并用NuGet程序包管理添加Quartz.NET , 添加好类库后 , 项目文件如下图所示:

   Quartz依赖库为Common.Logging和Common.Logging.Core  ,  二者需要一并导入 . 然后修改Form1的标题为QuartzDemo.

2.2 Form1定制

  在Form1设计视图上 , 为该窗体添加一个TextBox和Panel , 并设置相关属性(背景色,字体等) , 如下图所示:

  为了直观的执行定期任务 , 我们添加一个时序图(用Oxyplot控件) , 这里下载并引用Oxyplot.在Form1.cs文件中  ,首先引入需要的命名空间:

1     using Quartz;
2     using Quartz.Impl;
3     using Quartz.Job;
4     using System.Threading;
5 
6     using OxyPlot;
7     using OxyPlot.Axes;
8     using OxyPlot.Series;

  重载一个带参数的Form1构造方法:

 1         private bool isFirst = true;
 2         public Form1(string msg)
 3         {
 4             InitializeComponent();
 5             _msg = msg;
 6             try
 7             {
 8                 //启动 scheduler
 9                 scheduler.Start();
10                 // 定义一个job并和自定义的HelloJob进行绑定
11                 IJobDetail job = JobBuilder.Create<HelloJob>()
12                     .WithIdentity("HelloJob", "SimpleGroup")
13                     .Build();
14                 #region Cron Expressions
15                // ITrigger trigger2 = TriggerBuilder.Create()
16                //.WithIdentity("trigger3", "group1")
17                //.WithCronSchedule(" 0 0/5 * * * ?", x => x
18                //    .WithMisfireHandlingInstructionFireAndProceed())
19                //.ForJob("job1", "group1")
20                //.Build();
21                 #endregion
22 
23                 //定义一个即时触发的触发器,(每隔1秒进行重复执行)
24                 ITrigger trigger= TriggerBuilder.Create()
25                     .WithIdentity("trigger1", "SimpleGroup")
26                     .StartNow()
27                     .WithSimpleSchedule(x => x
28                         .WithIntervalInSeconds(1)
29                         .RepeatForever())
30                     .Build();
31                 // 将job和trigger进行绑定,并告知 quartz 调度器用trigger去执行job 
32                 scheduler.ScheduleJob(job, trigger);
33             }
34             catch (SchedulerException se)
35             {
36                 Console.WriteLine(se);
37             }
38        
39         }

  在Form1的FormClosed事件,即窗体关闭后,将scheduler关闭:

1         private void Form1_FormClosed(object sender, FormClosedEventArgs e)
2         {
3             //关闭 scheduler
4             scheduler.Shutdown();
5         }

   在窗体加载时,创建一个时序图:

 1         public OxyPlot.WindowsForms.PlotView  Plot;
 2         private LineSeries lineSeries = new LineSeries { Title = "即时监控(1秒)", StrokeThickness =2 };
 3         private void Form1_Load(object sender, EventArgs e)
 4         {
 5             Plot = new OxyPlot.WindowsForms.PlotView();
 6             Plot.Model = new PlotModel();
 7             Plot.Dock = DockStyle.Fill;
 8             this.panel1.Controls.Add(Plot);
 9 
10             Plot.Model.PlotType = PlotType.XY;
11             Plot.Model.Background = OxyColor.FromRgb(255, 255, 255);
12             Plot.Model.TextColor = OxyColor.FromRgb(0, 0, 0);
13             
14             // add Series and Axis to plot model
15             Plot.Model.Series.Add(lineSeries);
16             Plot.Model.Axes.Add(new LinearAxis());
17         }

  定义一个SetMsg方法来更新消息:

 1         private double xInit = 0;
 2         public bool SetMsg(string msg)
 3         {
 4             _msg = msg;
 5             //号称NET4最简单的跨进程更新UI的方法
 6             this.Invoke((MethodInvoker)delegate
 7             {
 8                 // runs on UI thread
 9                 if (isFirst)
10                 {
11                     this.txtLog.AppendText("Hello to Quartz NET ! Created by JackWang 2015");
12                     isFirst = false;
13                 }
14                 this.txtLog.AppendText(string.Format("\r\n$JackWang>> You get {0} message from Quartz MyJob...", _msg));
15                 xInit = xInit + 1;
16                 lineSeries.Points.Add(new DataPoint(xInit,double.Parse(_msg)));
17                 if (lineSeries.Points.Count > 50)
18                 {
19                     //保留最近50个点
20                     lineSeries.Points.RemoveAt(0);
21                 }
22                  //更新图表数据
23                 this.Plot.Model.InvalidatePlot(true);
24                
25             });
26             return true;
27 
28         }
2.3 定义一个HelloJob

  在带参数的Form1构造方法中 , 创建的HelloJob的定义为:

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Linq;
 4 using System.Text;
 5 using System.Windows.Forms;
 6 namespace QuartzDemo
 7 {
 8     using Quartz;
 9     using Quartz.Impl;
10     using Quartz.Job;
11     public  class HelloJob : IJob
12     {
13         private static Form1 __instance = null;
14         public  void Execute(IJobExecutionContext context)
15         {
16           
17             if (isOpen("Form1"))
18             {
19                 //获取当前Form1实例
20                 __instance = (Form1)Application.OpenForms["Form1"];
21                 //随机生成小于100的数
22                 string num = new Random().Next(100).ToString();
23                 //通过方法更新消息
24                 __instance.SetMsg(num);
25             }
26             else
27             {
28                 //__instance = new Form1("0");
29                 //__instance.Show();
30               
31             }
32         
33         }
34         /// <summary>
35         /// 判断窗体是否打开
36         /// </summary>
37         /// <param name="appName"></param>
38         /// <returns></returns>
39         private  bool isOpen(string appName)
40         {
41             FormCollection collection = Application.OpenForms;
42             foreach (Form form in collection)
43             {
44                 if (form.Name == appName)
45                 {
46                     return true;
47                 }
48             }
49             return false;
50         }
51     }
52 }

  注意修改Program.cs中用带参数的Form1构建方法进行实例创建:

1         [STAThread]
2         static void Main()
3         {
4             Application.EnableVisualStyles();
5             Application.SetCompatibleTextRenderingDefault(false);
6             Application.Run(new Form1("0"));
7         }

3 最终效果

   最终效果如下 , 每隔1秒,就是用随机的数来更新文本框和图表中的内容:

  值得注意的是,如果用下面的代码格式,必须保证二者的名称完全一致,否则无法实现任务和触发器的绑定.

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
基于事件驱动的网络框架源码 首先看一下Server类,这个类就是整个框架的核心类,在这个类中只需要 s=new Server(8883,4); s->setUserConnectionCallback(onConnect); s->setUserMessageCallBack(onMessage); s->start(); 就可以启动一个4个线程在8883端口监听的完整的网络程序,其中的两个onXXX就是网络程序需要处理的业务 Server类中包括了有一个重要的类叫做Eventlooper这个类就是对epoll的封装,要用epoll_ctl注册到epoll上的fd又被封装为Channel类,当有数据到来需要操作时,channel中的几个函数指针就指向了需要回调的函数, 这里使用了boost库的function 其中的函数定义为: typedef boost::function EventCallBack; 在epoll返回的时候Eventlooper会遍历可以操作的所有channel,并调用其成员函数handleEvent,该函数会判断events,也就是EPOLLIN EPOLLPRI等,并更具相应的需要去调用处理函数。 这里的Channel并不直接使用而是做基类存在的,更直接的操作在Acceptor和TcpConnection中,其中的 Acceptor 对应了接受连接的socketfd,而TcpConnection则封装了socket的读写操作,在Server的构造函数中会创建一个Acceptor类,真正accept成功以后又会调用到Server的OnConnection函数中,这个函数会根据每个线程的负载将客服端连接的fd注册进相应线程的epoll(当然在单线程的情况下就只有一个epoll)。最后一个作为缓冲区的类charbuff,用链表将单个7KB的缓冲区穿在一起,当数据大于7kb的时候会自动将其释放,避免内存不足,如果数据的块数多余一块那么发送数据就会调用writev。至此整个框架的流程就都已经清楚了,一个简单的实现。 在此还要感谢一下linux多线程网络服务器编程的作者陈硕,CINF的实现部分就是参考其网络框架muduo来的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值