Quartz下CronExpression表达式生成

 /// <summary>
    /// CronTrigger表达式配置
    /// </summary>
    /// <remarks>
    /// CronTrigger配置格式:
    /// 格式: [秒] [分] [小时] [日] [月] [周] [年]
    /// 序号 说明 必填	允许填写的值	    允许的通配符 
    /// 1    秒   是   0-59                , - * /
    /// 2    分   是   0-59                , - * /
    /// 3    小时 是   0-23                , - * /
    /// 4    日   是   1-31                , - * ? / L W
    /// 5    月   是   1-12 or JAN-DEC     , - * /
    /// 6    周   是   1-7 or SUN-SAT      , - * ? / L #
    /// 7    年   否   empty 或 1970-2099  , - * /
    /// 通配符说明:
    /// * 表示所有值. 例如:在分的字段上设置 "*",表示每一分钟都会触发。
    /// ? 表示不指定值。使用的场景为不需要关心当前设置这个字段的值。例如:要在每月的10号触发一个操作,但不关心是周几,所以需要周位置的那个字段设置为"?" 具体设置为 0 0 0 10 * ?
    /// - 表示区间。例如 在小时上设置 "10-12",表示 10,11,12点都会触发。
    /// , 表示指定多个值,例如在周字段上设置 "MON,WED,FRI" 表示周一,周三和周五触发
    /// / 用于递增触发。如在秒上面设置"5/15" 表示从5秒开始,每增15秒触发(5,20,35,50)。 在月字段上设置'1/3'所示每月1号开始,每隔三天触发一次。
    /// L 表示最后的意思。在日字段设置上,表示当月的最后一天(依据当前月份,如果是二月还会依据是否是润年[leap]), 在周字段上表示星期六,相当于"7"或"SAT"。如果在"L"前加上数字,则表示该数据的最后一个。例如在周字段上设置"6L"这样的格式,则表示“本月最后一个星期五" 
    /// W 表示离指定日期的最近那个工作日(周一至周五). 例如在日字段上设置"15W",表示离每月15号最近的那个工作日触发。如果15号正好是周六,则找最近的周五(14号)触发, 如果15号是周未,则找最近的下周一(16号)触发.如果15号正好在工作日(周一至周五),则就在该天触发。如果指定格式为 "1W",它则表示每月1号往后最近的工作日触发。如果1号正是周六,则将在3号下周一触发。(注,"W"前只能设置具体的数字,不允许区间"-").
    /// 小提示 'L'和 'W'可以一组合使用。如果在日字段上设置"LW",则表示在本月的最后一个工作日触发(一般指发工资 ) 
    /// # 序号(表示每月的第几个周几),例如在周字段上设置"6#3"表示在每月的第三个周六.注意如果指定"#5",正好第五周没有周六,则不会触发该配置(用在母亲节和父亲节再合适不过了)
    /// 小提示:周字段的设置,若使用英文字母是不区分大小写的 MON 与mon相同.
    /// </remarks>
    public class CronTriggerBuilder : IEnumerable<BaseCronCake>
    {

        /// <summary>
        /// 列表存储
        /// </summary>
        private Dictionary<CronCakeCycle, BaseCronCake> dicCronCake = new Dictionary<CronCakeCycle, BaseCronCake>();

        /// <summary>
        /// 构造函数
        /// </summary>
        protected CronTriggerBuilder()
        {

        }

        /// <summary>
        /// 创建表达式配置实例
        /// </summary>
        /// <returns></returns>
        public static CronTriggerBuilder Create()
        {
            return new CronTriggerBuilder();
        }

        /// <summary>
        /// 获取表达式块
        /// </summary>
        /// <param name="cronCakeCycle">计划周期类型</param>
        /// <returns>返回表达式块</returns>
        public BaseCronCake this[CronCakeCycle cronCakeCycle]
        {
            get
            {
                return this.dicCronCake[cronCakeCycle];
            }
        }

        /// <summary>
        /// 新增表达式块
        /// </summary>
        /// <param name="cronCake"></param>
        /// <returns></returns>
        public CronTriggerBuilder WithCronCake(BaseCronCake cronCake)
        {
            if (dicCronCake.ContainsKey(cronCake.Key))
                dicCronCake[cronCake.Key] = cronCake;
            dicCronCake.Add(cronCake.Key, cronCake);
            return this;
        }

        /// <summary>
        /// 生成表达式
        /// </summary>
        /// <returns></returns>
        internal string Build()
        {
            if (dicCronCake.Count <= 0)
                return string.Empty;
            IList<string> list = new List<string>();
            BaseCronCake baseCronCake = dicCronCake.Values.First(p => p.Key == CronCakeCycle.Second);
            if (baseCronCake != null)
                list.Add(baseCronCake.Build());
            baseCronCake = dicCronCake.Values.Single(p => p.Key == CronCakeCycle.Minute);
            if (baseCronCake != null)
                list.Add(baseCronCake.Build());
            baseCronCake = dicCronCake.Values.Single(p => p.Key == CronCakeCycle.Hour);
            if (baseCronCake != null)
                list.Add(baseCronCake.Build());
            baseCronCake = dicCronCake.Values.Single(p => p.Key == CronCakeCycle.Day);
            if (baseCronCake != null)
                list.Add(baseCronCake.Build());
            baseCronCake = dicCronCake.Values.Single(p => p.Key == CronCakeCycle.Month);
            if (baseCronCake != null)
                list.Add(baseCronCake.Build());
            baseCronCake = dicCronCake.Values.Single(p => p.Key == CronCakeCycle.Week);
            if (baseCronCake != null)
                list.Add(baseCronCake.Build());
            baseCronCake = dicCronCake.Values.Single(p => p.Key == CronCakeCycle.Year);
            if (baseCronCake != null)
                list.Add(baseCronCake.Build());
            return string.Join<string>(" ", list);
        }

        #region IEnumerable<BaseCronCake> 成员

        public IEnumerator<BaseCronCake> GetEnumerator()
        {
            return dicCronCake.Values.GetEnumerator();
        }

        #endregion

        #region IEnumerable 成员

        System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
        {
            return dicCronCake.Values.GetEnumerator();
        }

        #endregion
    }

 /// <summary>
    /// 基本的Cron表达式块
    /// </summary>
    public abstract class BaseCronCake
    {
        /// <summary>
        /// 逗号
        /// </summary>
        public const char Comma = ',';
        /// <summary>
        /// 星号
        /// </summary>
        public const char Asterisk = '*';
        /// <summary>
        /// 折线
        /// </summary>
        public const char Broken = '-';
        /// <summary>
        /// 反斜杠
        /// </summary>
        public const char Backslash = '/';
        /// <summary>
        /// 问号
        /// </summary>
        public const char Question = '?';
        /// <summary>
        /// 要输出的字符串
        /// </summary>
        protected string printString = string.Empty;

        /// <summary>
        /// 构造函数
        /// </summary>
        protected BaseCronCake()
        {

        }

        /// <summary>
        /// 默认的通配符返回
        /// </summary>
        protected virtual string defaultWildcard
        {
            get
            {
                return Asterisk.ToString();
            }
        }

        /// <summary>
        /// 使用周期的方式
        /// </summary>
        /// <param name="startCycle">周期开始</param>
        /// <param name="endCycle">周期结束</param>
        public virtual void WithCycle(int startCycle, int endCycle)
        {
            checkCycle(startCycle, endCycle);
            //使用周期的方式来处理表达式
            printString = string.Concat(startCycle, Broken, endCycle);
        }

        /// <summary>
        /// 校验循环周期的值
        /// </summary>
        /// <param name="startCycle">周期开始</param>
        /// <param name="endCycle">周期结束</param>
        protected virtual void checkCycle(int startCycle, int endCycle)
        {
            if (startCycle < 0 || endCycle < 0)
                throw new ArgumentException("Parameter error ", "startCycle");
            if (startCycle >= endCycle)
                throw new ArgumentException("Parameter error ", "startCycle");
        }

        /// <summary>
        /// 使用递增的方式
        /// </summary>
        /// <param name="start">开始</param>
        /// <param name="interval">递增间隔</param>
        public virtual void WithIncrease(int start, int interval)
        {
            printString = string.Concat(start, Backslash, interval);
        }

        /// <summary>
        /// 校验递增的值
        /// </summary>
        /// <param name="start">开始</param>
        /// <param name="interval">递增间隔</param>
        protected virtual void checkIncrease(int start, int interval)
        {
            if (start < 0 || interval <= 0)
                throw new ArgumentException("Parameter error ", "start");
        }

        /// <summary>
        /// 使用指定的方式
        /// </summary>
        public virtual void WithSpecified(IEnumerable<int> ranage)
        {
            Array.Sort<int>(ranage.ToArray());
            checkSpecified(ranage);
            if (ranage.Count() == 1)
                printString = ranage.First().ToString();
            else
                printString = string.Join<int>(Comma.ToString(), ranage);
        }

        /// <summary>
        /// 校验指定的值
        /// </summary>
        /// <param name="ranage">要检验的列表</param>
        protected virtual void checkSpecified(IEnumerable<int> ranage)
        {
            if (ranage.Count() <= 0)
                throw new ArgumentException("Parameter error ", "ranage");
            if (ranage.First() < 0)
                throw new ArgumentException("Parameter error ", "ranage");
        }

        /// <summary>
        /// 输出表达式
        /// </summary>
        /// <returns></returns>
        public virtual string Build()
        {
            if (string.IsNullOrEmpty(printString))
                printString = defaultWildcard;
            return printString;
        }

        /// <summary>
        /// 类型
        /// </summary>
        internal abstract CronCakeCycle Key { get; }
    }

/// <summary>
    /// 计划周期
    /// </summary>
    internal enum CronCakeCycle
    {
        /// <summary>
        /// 秒
        /// </summary>
        Second = 1,
        /// <summary>
        /// 分钟
        /// </summary>
        Minute = 2,
        /// <summary>
        /// 小时
        /// </summary>
        Hour = 3,
        /// <summary>
        /// 天
        /// </summary>
        Day = 4,
        /// <summary>
        /// 周
        /// </summary>
        Week = 5,
        /// <summary>
        /// 月
        /// </summary>
        Month = 6,
        /// <summary>
        /// 年
        /// </summary>
        Year = 7,
    }

/// <summary>
    /// 秒
    /// </summary>
    public class SecondsCronCake : BaseCronCake
    {
        /// <summary>
        /// 校验指定的值
        /// </summary>
        /// <param name="ranage">要检验的列表</param>
        protected override void checkSpecified(IEnumerable<int> ranage)
        {
            base.checkSpecified(ranage);
            int max = ranage.Last();
            if (max > 60)
                throw new ArgumentException("Parameter error ", "ranage");
        }

        /// <summary>
        /// 类型
        /// </summary>
        internal override CronCakeCycle Key
        {
            get { return  CronCakeCycle.Second; }
        }
    }

    /// <summary>
    /// 分钟
    /// </summary>
    public class MinuteCronCake : BaseCronCake
    {
        /// <summary>
        /// 校验指定的值
        /// </summary>
        /// <param name="ranage">要检验的列表</param>
        protected override void checkSpecified(IEnumerable<int> ranage)
        {
            base.checkSpecified(ranage);
            int max = ranage.Last();
            if (max > 60)
                throw new ArgumentException("Parameter error ", "ranage");
        }

        /// <summary>
        /// 类型
        /// </summary>
        internal override CronCakeCycle Key
        {
            get { return  CronCakeCycle.Minute; }
        }
    }

    /// <summary>
    /// 小时
    /// </summary>
    public class HourCronCake : BaseCronCake
    {
        /// <summary>
        /// 校验指定的值
        /// </summary>
        /// <param name="ranage">要检验的列表</param>
        protected override void checkSpecified(IEnumerable<int> ranage)
        {
            base.checkSpecified(ranage);
            int max = ranage.Last();
            if (max > 24)
                throw new ArgumentException("Parameter error ", "ranage");
        }

        /// <summary>
        /// 类型
        /// </summary>
        internal override CronCakeCycle Key
        {
            get { return CronCakeCycle.Hour; }
        }
    }

    /// <summary>
    /// 天
    /// </summary>
    public class DayCronCake : BaseCronCake
    {
        /// <summary>
        /// 校验指定的值
        /// </summary>
        /// <param name="ranage">要检验的列表</param>
        protected override void checkSpecified(IEnumerable<int> ranage)
        {
            base.checkSpecified(ranage);
            int max = ranage.Last();
            if (max > 31)
                throw new ArgumentException("Parameter error ", "ranage");
        }

        /// <summary>
        /// 不指定
        /// </summary>
        public void WithNoSpecified()
        {
            printString = Question.ToString();
        }

        /// <summary>
        /// 每月start号最近的工作日
        /// </summary>
        /// <param name="start">最近的天</param>
        public void WithNearByWorkDay(int start)
        {
            printString = string.Concat(start, "W");
        }

        /// <summary>
        /// 本月最后一天
        /// </summary>
        public void WithLastDay()
        {
            printString = "L";
        }

        /// <summary>
        /// 类型
        /// </summary>
        internal override CronCakeCycle Key
        {
            get { return CronCakeCycle.Day; }
        }
    }

    /// <summary>
    /// 月
    /// </summary>
    public class MonthCronCake : BaseCronCake
    {
        /// <summary>
        /// 校验指定的值
        /// </summary>
        /// <param name="ranage">要检验的列表</param>
        protected override void checkSpecified(IEnumerable<int> ranage)
        {
            base.checkSpecified(ranage);
            int max = ranage.Last();
            if (max > 12)
                throw new ArgumentException("Parameter error ", "ranage");
        }
        /// <summary>
        /// 不指定
        /// </summary>
        public void WithNoSpecified()
        {
            printString = Question.ToString();
        }

        /// <summary>
        /// 类型
        /// </summary>
        internal override CronCakeCycle Key
        {
            get { return CronCakeCycle.Month; }
        }
    }

    /// <summary>
    /// 周
    /// </summary>
    public class WeekCronCake : BaseCronCake
    {
        /// <summary>
        /// 星期日
        /// </summary>
        public const int Sunday = 1;
        /// <summary>
        /// 星期一
        /// </summary>
        public const int Monday = 2;
        /// <summary>
        /// 星期二
        /// </summary>
        public const int Tuesday = 3;
        /// <summary>
        /// 星期三
        /// </summary>
        public const int Wednesday = 4;
        /// <summary>
        /// 星期四
        /// </summary>
        public const int Thursday = 5;
        /// <summary>
        /// 星期五
        /// </summary>
        public const int Friday = 6;
        /// <summary>
        /// 星期六
        /// </summary>
        public const int Saturday = 7;

        /// <summary>
        /// 校验指定的值
        /// </summary>
        /// <param name="ranage">要检验的列表</param>
        protected override void checkSpecified(IEnumerable<int> ranage)
        {
            base.checkSpecified(ranage);
            int max = ranage.Last();
            if (max > 7)
                throw new ArgumentException("Parameter error ", "ranage");
        }

        /// <summary>
        /// 不指定
        /// </summary>
        public void WithNoSpecified()
        {
            printString = Question.ToString();
        }

        /// <summary>
        /// 第start周的星期几
        /// </summary>
        /// <param name="start"></param>
        /// <param name="week">星期</param>
        public void WithWeekly(int start, int week)
        {
            if (start <= 0 || start > 4)
                throw new ArgumentException("Parameter error ", "start");
            if (week < Sunday || week > Saturday)
                throw new ArgumentException("Parameter error ", "week");
            printString = string.Concat(week, "#", start);
        }

        /// <summary>
        /// 本月最后一个星期几
        /// </summary>
        /// <param name="week">星期</param>
        public void WithLastWeekly(int week)
        {
            if (week < Sunday || week > Saturday)
                throw new ArgumentException("Parameter error ", "week");
            printString = string.Concat(week, "L");
        }

        /// <summary>
        /// 类型
        /// </summary>
        internal override CronCakeCycle Key
        {
            get { return CronCakeCycle.Week; }
        }
    }

    /// <summary>
    /// 年
    /// </summary>
    public class YearCornCake : BaseCronCake
    {
        /// <summary>
        /// 使用递增的方式
        /// </summary>
        /// <param name="start">开始</param>
        /// <param name="interval">递增间隔</param>
        public override void WithIncrease(int start, int interval)
        {
            throw new NotSupportedException("The method or operation is not implemented.");
        }

        /// <summary>
        /// 类型
        /// </summary>
        internal override CronCakeCycle Key
        {
            get { return CronCakeCycle.Year; }
        }
    }

Cron表达式是一个字符串,字符串以5或6个空格隔开,分为6或7个域,每一个域代表一个含义,Cron有如下两种语法格式: Seconds Minutes Hours DayofMonth Month DayofWeek Year或 Seconds Minutes Hours DayofMonth Month DayofWeek 每一个域可出现的字符如下: Seconds:可出现", - * /"四个字符,有效范围为0-59的整数 Minutes:可出现", - * /"四个字符,有效范围为0-59的整数 Hours:可出现", - * /"四个字符,有效范围为0-23的整数 DayofMonth:可出现", - * / ? L W C"八个字符,有效范围为0-31的整数 Month:可出现", - * /"四个字符,有效范围为1-12的整数或JAN-DEc DayofWeek:可出现", - * / ? L C #"八个字符,有效范围为1-7的整数或SUN-SAT两个范围。1表示星期天,2表示星期一, 依次类推 Year:可出现", - * /"四个字符,有效范围为1970-2099年 每一个域都使用数字,但还可以出现如下特殊字符,它们的含义是: (1)*:表示匹配该域的任意值,假如在Minutes域使用*, 即表示每分钟都会触发事件。 (2)?:只能用在DayofMonth和DayofWeek两个域。它也匹配域的任意值,但实际不会。因为DayofMonth和DayofWeek会相互影响。例如想在每月的20日触发调度,不管20日到底是星期几,则只能使用如下写法: 13 13 15 20 * ?, 其中最后一位只能用?,而不能使用*,如果使用*表示不管星期几都会触发,实际上并不是这样。 (3)-:表示范围,例如在Minutes域使用5-20,表示从5分到20分钟每分钟触发一次 (4)/:表示起始时间开始触发,然后每隔固定时间触发一次,例如在Minutes域使用5/20,则意味着5分钟触发一次,而25,45等分别触发一次. (5),:表示列出枚举值值。例如:在Minutes域使用5,20,则意味着在5和20分每分钟触发一次。 (6)L:表示最后,只能出现在DayofWeek和DayofMonth域,如果在DayofWeek域使用5L,意味着在最后的一个星期四触发。 (7)W:表示有效工作日(周一到周五),只能出现在DayofMonth域,系统将在离指定日期的最近的有效工作日触发事件。例如:在 DayofMonth使用5W,如果5日是星期六,则将在最近的工作日:星期五,即4日触发。如果5日是星期天,则在6日(周一)触发;如果5日在星期一到星期五中的一天,则就在5日触发。另外一点,W的最近寻找不会跨过月份 (8)LW:这两个字符可以连用,表示在某个月最后一个工作日,即最后一个星期五。 (9)#:用于确定每个月第几个星期几,只能出现在DayofMonth域。例如在4#2,表示某月的第二个星期三。 举几个例子: 0 0 2 1 * ? * 表示在每月的1日的凌晨2点调度任务 0 15 10 ? * MON-FRI 表示周一到周五每天上午10:15执行作业 0 15 10 ? 6L 2002-2006 表示2002-2006年的每个月的最后一个星期五上午10:15执行作 0 0 10,14,16 * * ? 每天上午10点,下午2点,4点 0 0/30 9-17 * * ? 朝九晚五工作时间内每半小时 0 0 12 ? * WED 表示每个星期三中午12点 "0 0 12 * * ?" 每天中午12点触发 "0 15 10 ? * *" 每天上午10:15触发 "0 15 10 * * ?" 每天上午10:15触发 "0 15 10 * * ? *" 每天上午10:15触发 "0 15 10 * * ? 2005" 2005年的每天上午10:15触发 "0 * 14 * * ?" 在每天下午2点到下午2:59期间的每1分钟触发 "0 0/5 14 * * ?" 在每天下午2点到下午2:55期间的每5分钟触发 "0 0/5 14,18 * * ?" 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发 "0 0-5 14 * * ?" 在每天下午2点到下午2:05期间的每1分钟触发 "0 10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44触发 "0 15 10 ? * MON-FRI" 周一至周五的上午10:15触发 "0 15 10 15 * ?" 每月15日上午10:15触发 "0 15 10 L * ?" 每月最后一日的上午10:15触发 "0 15 10 ? * 6L" 每月的最后一个星期五上午10:15触发 "0 15 10 ? * 6L 2002-2005" 2002年至2005年的每月的最后一个星期五上午10:15触发 "0 15 10 ? * 6#3" 每月的第三个星期五上午10:15触发
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值