之前有个旧同事说他在项目中碰到某些功能需要使用到作业调度,于是找到了这个组件,据说相当好用,叫我有时间的话去了解一下。哈,于是小了解了一下,基本的使用算是明白了,深层次的东西就不了解了,本文简单记录一下最基本的使用!
    关于Quartz.NET,请访问它的网站:http://quartznet.sourceforge.net     下载Quartz.NET,请直接进这里:http://quartznet.sourceforge.net/download.html     首先看下在什么情况下我们会需要使用Quartz.NET来进行作业调度。在业务系统中,我们需要在每天的某一个时间点(例如晚上12点)中做一些统计报表的生成(例如生成当前的一些报表之类~);或者在网站中定时更新静态页面;或者在CRM中会在某些特殊的日子给予提醒。总之这些在根据预定好的时间规则里要去做某些事情的需求,都是可以用这个来解决的,当然了,前提是你开发的东西用的是.NET- -!
配置
    使用Quartz.NET,我们需要在配置文件里面添加点东西,首先是configSections节点中,我们要添加已下内容:
xml代码
1
2
3
4
<section name="quartz" type="System.Configuration.NameValueSectionHandler, System, Version=1.0.5000.0,Culture=neutral, PublicKeyToken=b77a5c561934e089"/>
<sectionGroup name="common">
  <section name="logging" type="Common.Logging.ConfigurationSectionHandler, Common.Logging"/>
</sectionGroup>
    
    从上面添加的节点可以知道,肯定还有一些节点要添加的,在configuration节点中添加如下:
xml代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<common>
  <logging>
    <factoryAdapter type="Common.Logging.Simple.ConsoleOutLoggerFactoryAdapter, Common.Logging">
      <arg key="showLogName" value="true"/>
      <arg key="showDataTime" value="true"/>
      <arg key="level" value="DEBUG"/>
      <arg key="dateTimeFormat" value="HH:mm:ss:fff"/>
    </factoryAdapter>
  </logging>
</common>
<quartz>
  <add key="quartz.scheduler.instanceName" value="ExampleDefaultQuartzScheduler"/>
  <add key="quartz.threadPool.type" value="Quartz.Simpl.SimpleThreadPool, Quartz"/>
  <add key="quartz.threadPool.threadCount" value="10"/>
  <add key="quartz.threadPool.threadPriority" value="2"/>
  <add key="quartz.jobStore.misfireThreshold" value="60000"/>
  <add key="quartz.jobStore.type" value="Quartz.Simpl.RAMJobStore, Quartz"/>
</quartz>
    注:以上的配置我还没有去仔细研究,我是在网上找中文资料抄的 - -!

Quartz.NET中几个重要的东西
 
1、作业调度器接口 IScheduler:这个接口负责启动一个作业和关闭一个作业,在启动和关闭作业之前,它还需要接受两个东西,一个是触发器(Trigger),一个是工作任务(JobDetail),要得到一个这样的接口的实例,我们还需要一个作业调度器工厂(ISchedulerFactory)。

2、作业调度器工厂接口 ISchedulerFactory:从命名来看,很明显是个工厂,专门产出IScheduler接口。

3、工作任务 JobDetail:它定义了你要定时去做什么工作,初始化一个该类的对象,我们要指定工作名(name),工作的组名(group),还有就是一个实现了IJob接口的类的类型(Type),就是说JobD定义的是做什么。
 
4、触发器 Trigger:据说Quartz.NET的触发器有很多,但是一般我们使用的是两种(都是继承了Trigger类的子类),一个是CronTrigger、另外一个是SimpleTrigger,我们要指定触发器名(name),触发器的的职责是定义工作任务的触发规则,也就是说Trigger定义的是啥时候做。
 
5、工作接口 IJob:我们需要把定时要做的事情(业务逻辑,例如去写一下数据库,写一下文件之类的)包装在一个类里面,这个类有一个特点,就是继承IJob这个接口,实现该接口的唯一方法Execute(JobExecutionContext context),在这个方法中将尽情地用代码实现我们的业务,所以可以说这个接口定义的是如何做。     
 
    网上找的资料基本一上来就是代码,没有对几个重要的对象就行讲解,这点比较遗憾!通过上面我们基本可以猜想出写出来的代码会是怎么样:我们先定义个类去实现IJob接口,把要做的“工作”写好,然后从作业调度器工厂(ISchedulerFactory)中拿个作业调度器(IScheduler)出来,定义好工作任务(JobDetail)指定这个工作任务的工作就是我们定义的那个类,定义好触发器(Trigger,也许是用CronTrigger、也许是用SimpleTrigger),把工作任务和触发器交给作业调度器,这样一来,就确定了在什么时候要去做什么,也知道要怎么做了,最后调用作业调度器的一个方法(Statr())来开始任务!

代码示例
    下面用代码来说明问题,我们用代码来完成这样一个需求:在web项目中有一个txt文件(文件名为JobNotePad.txt),每隔十秒钟就把当前时间记录在里面。这里我们使用CronTrigger这种触发器来完成任务。首先我们要实现IJob接口,定义一个类如下:
c#代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
using Quartz;
namespace WebApplication.Common
{
    public class MyJob : IJob
    {
        public void Execute(JobExecutionContext context)
        {
            try
            {           
                var path = @"H:\代码\测试代码\Csharp测试\QuartzTest\WebApplication\WebApplication\JobNotePad.txt";
                var curText = System.IO.File.ReadAllText(path);
                System.IO.File.WriteAllText(path, curText + "\r\n" + DateTime.Now.ToString());
            }
            catch (Exception ex)
            {
                throw;
            }
        }
    }
}
    代码很简单,往txt文件里面写入当前时间。注意这里文件的路径,我直接用了物理路径,因为Quartz.NET的工作是在它自己启的独立进程里面跑的,所以没办法通过当前的上下文对象把虚拟路径转为物理路径,也就是说var path = System.Web.HttpContext.Current.Server.MapPath("~/JobNotePad.txt");这样的代码会报错!这个大家可以去体会一下~~~
    接着我们要在Global.asax文件里写点代码,在网站开启的时候就开始作业,于是在Application_Start中加入如下代码:
c#代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
/// <summary>
/// 作业调度器接口
/// </summary>
IScheduler scheduler;
protected void Application_Start(object sender, EventArgs e)
{
    try
    {
        // 创建一个工作调度器工场
        ISchedulerFactory schedulerFactory = new StdSchedulerFactory();
        // 获取一个任务调度器
        scheduler = schedulerFactory.GetScheduler();
        // 创建一个工作
        JobDetail job = new JobDetail("jobname1", "jobgroup1", typeof(Common.MyJob));
        // 创建一个触发器
        CronTrigger trigger = new CronTrigger();
        trigger.Name = "trigger1";
        trigger.JobName = "jobname1";
        trigger.JobGroup = "jobgroup1";
        trigger.Group = "triggergroup1";
        trigger.CronExpression = new CronExpression("0/10 * * * * ?");
        scheduler.AddJob(job, true);
        DateTime ft = scheduler.ScheduleJob(trigger);
        scheduler.Start();
    }
    catch (Exception ex)
    {
        throw ex;
    }
}
    这里我们发现把IScheduler的实例定义在方法外面了,理由是我们要在Application_End中用到它,如下(关闭作业调度):
c#代码
1
2
3
4
protected void Application_End(object sender, EventArgs e)
{
    scheduler.Shutdown(true);
}
    代码中有个地方比较有趣,就是trigger.CronExpression = new CronExpression("0/10 * * * * ?");这句,表示的是每隔10秒执行一次,从0秒开始,也就是说每个10秒、20秒、30秒...会fire一次工作。关于这个时间,这里有篇文章讲得很好,http://www.cnblogs.com/gooddasenlin/archive/2011/04/06/2007084.html     修改一下需求,现在是要一秒钟内执行一次,执行10次就结束,我们可以使用另外一个触发器(SimpleTrigger)来完成任务,修改Application_Start代码如下:
c#代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
protected void Application_Start(object sender, EventArgs e)
{
    try
    {
        ISchedulerFactory schedulerFactory = new StdSchedulerFactory();
        scheduler = schedulerFactory.GetScheduler();
        JobDetail job = new JobDetail("jobname1", "jobgroup1", typeof(Common.MyJob));
        // 创建一个触发器
        SimpleTrigger trigger = new SimpleTrigger();
        trigger.Name = "trigger1";
        trigger.JobName = "jobname1";
        trigger.JobGroup = "jobgroup1";
        trigger.Group = "triggergroup1";
        trigger.RepeatInterval = new TimeSpan(0, 0, 1);
        trigger.RepeatCount = 10;
        scheduler.AddJob(job, true);
        DateTime ft = scheduler.ScheduleJob(trigger);
        scheduler.Start();
    }
    catch (Exception ex)
    {
        throw ex;
    }
}
    可以看到,我们其实只改了触发器的类型和触发器触发时间的定义。使用这两种触发器基本可以满足我们的需求了!
    测试代码,我们只需要运行网站,然后查看根目录下的JobNotePad.txt文件就知道鸟!
posted @ 2012-07-13 14:39 serafin 阅读(548) 评论(0) 推荐(0) 编辑
摘要: 这又是一个很不错的VS插件,相信很多人都非常喜欢VS自带的代码折叠功能,可以把一个方法或者一个类折叠成一行,但是很遗憾的是这个功能没法比应用到js和css上面,以为ajax的兴起,js在开发中的地位越来越高,但是随着而来的是冗长而不可折叠的代码出现。JSEnhancements很好地补缺了这一块功能,它使得我们可以有序地折叠js和css代码,就像我们对付C#代码一样! 阅读全文
posted @ 2012-07-13 11:57 serafin 阅读(579) 评论(0) 推荐(0) 编辑
摘要: 之前在补习基础知识的时候写了两篇文章,分别简要记录了asp.net配置文件(web.config)中httpHandlers和configSections的作用,既然是补习,那就尽量写成一个系列那样子,这篇文章主要讲述另外一个也是比较底层的配置,是和httpHandlers在web.config中处于同一级的httpModules。 阅读全文
posted @ 2012-07-13 11:56 serafin 阅读(1032) 评论(0) 推荐(1) 编辑
摘要: 之前的一篇文章“使用URLRewriter让你的站点牛逼起来- -!”中在讲述使用URLRewriter.dll的时候对web.config文件的配置中,第一步就提到对“configSections”节点的配置,在第二步中又根据第一步的配置来进行了进一步的节点配置——自定义节点的配置(“RewriterConfig”节点)。本文将通过实例简单描述一下configSections加点的使用方法^^ 阅读全文
posted @ 2012-07-13 11:49 serafin 阅读(1451) 评论(1) 推荐(1) 编辑
摘要: 写了个文章尽量用自己的话讲述了URLRewriter.dll组件的基本使用,组件本身使用起来比较方便,通过简单的配置边可以进行URL重写并且实现用于装逼或者加强SEO的伪静态。但是为了了解配置文件各个节点的作用还是小费了一番功夫的,归根结底还是基础不扎实惹的祸。这篇文章尽量用自己的语言简单讲述一下httpHandlers节点的使用^_^ 阅读全文
posted @ 2012-07-13 11:33 serafin 阅读(945) 评论(0) 推荐(0) 编辑
摘要: 之前有个网页提醒我用一下URLRewriter.dll给自己的博客做一下伪静态化,据说是可以提高SEO,今天试了一下这个组件,果然好用,至于对SEO有没有帮助就不得而知了(我一直觉得最好的SEO是¥......) URLRewriter使用起来很简单,只需要通过简单的配置就ok了,大致的步骤如下: 1、把组件拿过来然后引用到项目里面去,组件的下载地址在这里:传送门......传过去会看到一个超链接可以点击,点下去后下载一个msi文件,安装吧少年!!! 2、把URLRewriter.dll应用到项目中,然后对Web.config进行必要的配置: (1)找到configuration节点中的co. 阅读全文
posted @ 2012-07-13 11:25 serafin 阅读(569) 评论(0) 推荐(0) 编辑
摘要: 之前一直苦恼于在页面上找JS方法,后来发现了这个VS插件,她使得开发者可以很快地定位到文件中的某个JS方法或者变量,相当好用,在很大程度上提高了开发体验。 阅读全文
posted @ 2012-07-13 11:17 serafin 阅读(982) 评论(0) 推荐(0) 编辑
摘要: 之前在博客园有幸从网友那里得知一个C#的模型验证组件(哈 不知道这样表述正确不),组件的功能比较简单,主要是实现了对Model的验证,例如验证用户名是否为空,密码长度是不是多余6个字符,当然还有其他更加灵活的的验证方法。这里介绍给大家知道一下,昨天我试用了一下感觉很不错^_^ 阅读全文
posted @ 2012-07-13 11:15 serafin 阅读(449) 评论(0) 推荐(0) 编辑
摘要: 在编码的过程中经常会遇到正则表达式的使用,一般来说我都是直接google找资料的 - -!,不过觉得最好还是要了解一下正则表达式是怎么写的。刚看到的这篇文章不错,转了大部分内容过来,作为笔记用,方便以后需要的时候可以查看^_^ 阅读全文
posted @ 2012-07-13 11:12 serafin 阅读(343) 评论(0) 推荐(0) 编辑
摘要: "只要 new 表达式之后的 constructor 返回(return)一个引用对象(数组,对象,函数等),都将覆盖new创建的匿名对象,如果返回(return)一个原始类型(无 return 时其实为 return 原始类型 undefined),那么就返回 new 创建的匿名对象" 阅读全文
posted @ 2012-07-13 11:01 serafin 阅读(541) 评论(0) 推荐(0) 编辑
点击右上角即可分享
微信分享提示