上一篇文章我们讲诉了自定义线程执行器和任务处理器

我们继续来讲解自定义线程的定时执行器,我们在很多场景下需要做到某些状态或者数据进行更新,如果事情很多很杂,很时候时候会创建很多不同的定时器那么势必会照成系统的消耗和性能低下的问题!今天我们来解决这一问题。

首先我们创建定时任务执行器基类

 /// <summary>
     ///
     /// </summary>
     public abstract class TimerTaskBase : BaseTask
     {

         /// <summary>
         /// 开始执行的时间
         /// </summary>
         public long StartTime { get; set; }

         /// <summary>
         /// 是否一开始执行一次
         /// </summary>
         public bool IsStartAction { get; set; }

         /// <summary>
         /// 结束时间
         /// </summary>
         public long EndTime { get; set; }

         /// <summary>
         /// 执行次数
         /// </summary>
         public int ActionCount { get; set; }

         /// <summary>
         /// 已经执行的次数
         /// </summary>
         public int AActionCount { get; set; }

         /// <summary>
         /// 间隔执行时间
         /// </summary>
         public int IntervalTime { get; set; }

         /// <summary>
         /// 制定执行次数的定时任务
         /// </summary>
         /// <param name="startTime">0表示立即执行,否则延迟执行,填写开始时间</param>
         /// <param name="intervalTime">执行间隔时间,小于10毫秒,当10毫秒处理</param>
         /// <param name="isStartAction">是否一开始执行一次</param>
         /// <param name="actionCount">需要执行的次数</param>
         public TimerTaskBase(long startTime, int intervalTime, bool isStartAction, int actionCount)
         {
             this.StartTime = startTime;
             this.IntervalTime = intervalTime;
             this.IsStartAction = isStartAction;
             this.ActionCount = actionCount;
             ;
         }

         /// <summary>
         /// 制定结束时间的定时任务
         /// </summary>
         /// <param name="startTime">0表示立即执行,否则延迟执行,填写开始时间</param>
         /// <param name="intervalTime">执行间隔时间,小于10毫秒,当10毫秒处理</param>
         /// <param name="endTime">执行结束时间</param>
         /// <param name="isStartAction">是否一开始执行一次</param>
         public TimerTaskBase(long startTime, int intervalTime, long endTime, bool isStartAction)
         {
             this.StartTime = startTime;
             this.IntervalTime = intervalTime;
             this.IsStartAction = isStartAction;
             ;
             this.EndTime = endTime;
         }

         /// <summary>
         /// 制定开始时间,无限执行任务
         /// </summary>
         /// <param name="startTime">0表示立即执行,否则延迟执行,填写开始时间</param>
         /// <param name="intervalTime">执行间隔时间,小于10毫秒,当 10 毫秒处理 建议 10 毫秒的倍数</param>
         /// <param name="isStartAction">是否一开始执行一次</param>
         public TimerTaskBase(long startTime, int intervalTime, bool isStartAction)
         {
             this.StartTime = startTime;
             this.IntervalTime = intervalTime;
             this.IsStartAction = isStartAction;
             ;
             ;
         }

         public TimerTaskBase()
         {
             // TODO: Complete member initialization
         }
     }

上面的代码实现了,开始时间,间隔时间,结束时间和执行次数 的控制

那么我们来看看定时器线程的设计

 public class TimerThread
     {
         public TimerThread()
         {
             System.Threading.Thread thread = new System.Threading.Thread(new System.Threading.ThreadStart(Run));
             thread.IsBackground = true;
             thread.Start();
         }

         /// <summary>
         /// 任务队列
         /// </summary>
         private List<TimerTaskBase> taskQueue = new List<TimerTaskBase>();

         /// <summary>
         /// 加入任务
         /// </summary>
         /// <param name="t"></param>
         public void AddTask(TimerTaskBase t)
         {
             if (t.IsStartAction)
             {
                 //满足添加队列前先执行一次
                 t.Run();
             }
             lock (taskQueue)
             {
                 taskQueue.Add(t);
             }
         }

         public long GetDate()
         {
             return Convert.ToInt64(System.DateTime.Now.ToString("yyyyMMddHHmmssfff"));
         }

         //这里的线程同步器,不是用来通知的,
         //只是用来暂停的,因为Thread.Sleep() 消耗开销比较大
         ManualResetEvent mre = new ManualResetEvent(false);

         /// <summary>
         /// 重构函数执行器
         /// </summary>
         private void Run()
         {
             ///无限循环执行函数器
             while (true)
             {
                 )
                 {
                     IEnumerable<TimerTaskBase> collections = null;
                     lock (taskQueue)
                     {
                         //拷贝一次队列 预防本次轮训检查的时候有新的任务添加
                         //否则循环会出错 集合被修改无法迭代
                         collections = new List<TimerTaskBase>(taskQueue);
                     }
                     //开始迭代
                     foreach (TimerTaskBase tet in collections)
                     {
                         int actionCount = tet.AActionCount;
                         long timers = GetDate();
                          && timers > tet.EndTime) || (tet.ActionCount >  && actionCount >= tet.ActionCount))
                         {
                             //任务过期
                             lock (taskQueue)
                             {
                                 taskQueue.Remove(tet);
                             }
                             continue;
                         }
                         //获取最后一次的执行时间
                         long lastactiontime = tet.TempAttribute.getlongValue("lastactiontime");
                          && Math.Abs(timers - lastactiontime) < tet.IntervalTime)
                         {
                             continue;
                         }
                         //记录出来次数
                         tet.AActionCount++;
                         //记录最后执行的时间
                         tet.TempAttribute.setValue("lastactiontime", timers);

                         //上面的代码执行情况是非常几乎不用考虑消耗问题

                         //下面是任务的执行需要考虑消耗,

                         //这里我们不考虑执行耗时问题,
                         //我们我这里没有涉及到后台线程池
                         //也没有具体的业务逻辑,所以都放到这里统一执行
                         tet.Run();
                     }
                 }
                 //暂停10毫秒后再次检查
                 mre.WaitOne();
             }
         }
     }

定时器为什么没有使用上一篇文章讲诉的自定义线程呢,是因为,上一篇文章的自定义线程是基于队列处理的,先进先出执行,而我们的定时器任务并非是基于队列的。所以需要单独定义。

那么我们先来实现一个每一秒执行的任务,

  /// <summary>
     /// 每秒执行的任务
     /// </summary>
     public class SecondsTimerTask : TimerTaskBase
     {
         /// <summary>
         /// 定义一秒执行一次的
         /// </summary>
         public SecondsTimerTask()
             : , , false)
         {

         }

         public override void Run()
         {
             Console.WriteLine(DateTime.Now.ToString("HH:mm:ss:ffff: ") + "我是 每秒 执行的任务");
         }
     }

我们来测试一下看看效果

     class Program
     {
         static void Main(string[] args)
         {
             TimerThread timerThread = new TimerThread();
             timerThread.AddTask(new SecondsTimerTask());
             Console.ReadLine();
         }
     }

还算是我们的预想的效果吧,每秒执行一次

接下来我们创建每分钟执行一次

     public class MinuteTimerTask : TimerTaskBase
     {
         /// <summary>
         /// 定义一分钟执行一次的
         /// </summary>
         public MinuteTimerTask()
             : ,  * , false)
         {

         }

         public override void Run()
         {
             Console.WriteLine(DateTime.Now.ToString("HH:mm:ss:ffff: ") + "我是 每分钟 执行的任务");
         }
     }

按照执行次数的定时器

     /// <summary>
     /// 我是按照执行次数结束的
     /// </summary>
     public class CountTimerTask : TimerTaskBase
     {
         /// <summary>
         /// 定义一秒执行一次的
         /// </summary>
         public CountTimerTask()
             : , , )
         {

         }

         public override void Run()
         {
             Console.WriteLine(DateTime.Now.ToString("HH:mm:ss:ffff: ") + "我是 次数 执行的任务 我要执行:" + this.ActionCount + " 次 执行了: " + this.AActionCount + " 次");
         }
     }

测试一下

    class Program
     {
         static void Main(string[] args)
         {
             TimerThread timerThread = new TimerThread();
             timerThread.AddTask(new SecondsTimerTask());

             timerThread.AddTask(new MinuteTimerTask());
             timerThread.AddTask(new CountTimerTask());

             Console.ReadLine();
         }
     }

运行结果:

到此我们的定时器线程执行器已经完成,满足我们需要做到定时更新状态,合作检查数据等应用场景!

不知道对你有没有帮助呢?

请不了吝啬你的小手,给予一个评论吧,帮助到你了请给与鼓励,如果有不足之处还请多多指教!

C# 线程系列三 定时器线程的更多相关文章

  1. 线程系列08,实现线程锁的各种方式,使用lock,Montor,Mutex,Semaphore以及线程死锁

    当涉及到多线程共享数据,需要数据同步的时候,就可以考虑使用线程锁了.本篇体验线程锁的各种用法以及线程死锁.主要包括: ※ 使用lock处理数据同步※ 使用Monitor.Enter和Monitor.E ...

  2. 【java线程系列】java线程系列之java线程池详解

    一线程池的概念及为何需要线程池: 我们知道当我们自己创建一个线程时如果该线程执行完任务后就进入死亡状态,这样如果我们需要在次使用一个线程时得重新创建一个线程,但是线程的创建是要付出一定的代价的,如果在 ...

  3. 【java线程系列】java线程系列之线程间的交互wait()/notify()/notifyAll()及生产者与消费者模型

    关于线程,博主写过java线程详解基本上把java线程的基础知识都讲解到位了,但是那还远远不够,多线程的存在就是为了让多个线程去协作来完成某一具体任务,比如生产者与消费者模型,因此了解线程间的协作是非 ...

  4. pThreads线程(三) 线程同步--条件变量

    条件变量(Condition Variables) 参考资料:http://game-lab.org/posts/posix-thread-cn/#5.1 条件变量是什么? 条件变量为我们提供了另一种 ...

  5. java线程系列之三(线程协作)

    本文来自:高爽|Coder,原文地址:http://blog.csdn.net/ghsau/article/details/7433673,转载请注明. 上一篇讲述了线程的互斥(同步),但是在很多情况 ...

  6. java多线程与线程并发三:线程同步通信

    本文章内容整理自:张孝祥_Java多线程与并发库高级应用视频教程. 有些时候,线程间需要传递消息,比如下面这道面试题: 子线程循环10次,然后主线程循环100次,然后又回到子线程循环50次,然后再回到 ...

  7. Java多线程系列三——实现线程同步的方法

    两种实现线程同步的方法 方法 特性 synchronized 不需要显式地加解锁,易实现 ReentrantLock 需要显式地加解锁,灵活性更好,性能更优秀,结合Condition可实现多种条件锁 ...

  8. (Java多线程系列三)线程间通讯

    Java多线程间通讯 多线程之间通讯,其实就是多个线程在操作同一个资源,但是操作的动作不同. 1.使用wait()和notify()方法在线程中通讯 需求:第一个线程写入(input)用户,另一个线程 ...

  9. 死磕 java线程系列之创建线程的8种方式

    (手机横屏看源码更方便) 问题 (1)创建线程有哪几种方式? (2)它们分别有什么运用场景? 简介 创建线程,是多线程编程中最基本的操作,彤哥总结了一下,大概有8种创建线程的方式,你知道吗? 继承Th ...

随机推荐

  1. 利用jmSlip写一个移动端顶部日历选择组件

    可滚动选日期,并限制哪些日期可选和不可选. 主要用来根据后台返回生成一个日期选择器. 具体实现可关注jmslip: https://github.com/jiamao/jmSlip 示例:http:/ ...

  2. Sensor(LIGHT)

    package com.example.sensor01; import java.util.List; import android.hardware.Sensor; import android. ...

  3. Ubuntu Desktop开发生产环境搭建

    Ubuntu Desktop开发生产环境搭建 1   开发生产环境搭建 在本节内容开始前,先定义一下使用场合,没有哪种系统或者设备是万能的,都有它的优点和缺点,能够在具体的使用场景,根据自身的需求来取 ...

  4. 每周一书-编写高质量代码:改善C程序代码的125个建议

    首先说明,本周活动有效时间为2016年8月28日到2016年9月4日.本周为大家送出的书是由机械工业出版社出版,马伟编著的<编写高质量代码:改善C程序代码的125个建议>. 编辑推荐 10 ...

  5. nim的引用和指针

    nim语言的引用和其他语言的指针有点相似 可以提供一种“多对一”的关系 这就意味着不同的引用可以指向同一个内存位置 nim区分可被追踪的引用和不可被追踪的引用 不可被追踪的引用又称为指针 可被追踪的引 ...

  6. Yii CModel中rules验证 获取错误信息

    在model中定义 public function rules(){ return array( array('Name,Email','required'), array('Email','uniq ...

  7. [Unity3D入门]入门级游戏项目"坦克狙击手"更新

    [Unity3D入门]入门级游戏项目"坦克狙击手"更新 在上一篇中我分享了一个尚未完全写好的入门级unity3d项目"坦克狙击手". 本文介绍最新版的" ...

  8. 在js中对时间类型格式化字符串

    Date.prototype.toString = function (format) { if (format == null) { format = "yyyy-MM-dd HH:mm: ...

  9. ECMAScript5 Array新增方法

    数组在各个编程语言中的重要性不言而喻,但是在之前的JavaScript中数组(JavaScript 数组详解)虽然功能已经很强大,但操作方法并不完善,在ECMAScript5中做了适当的补充. Arr ...

  10. 十分钟搞定CSS选择器

    在最近的web开发中是不是就会用到一些选择器,发现很多尤其是CSS3新增的不太熟悉,在此总结一下. 优先级 不同级别 1. 在属性后面使用 !important 会覆盖页面内任何位置定义的元素样式. ...