https://blog.csdn.net/hoiven/article/details/51362582

如果你需要使用规律的时间间隔重复执行一些方法,最简单的方式是使用定时器(timer)。

.NET Framework 提供了 4 种定时器。下边两个类是通用的多线程定时器:

(1)System.Threading.Timer
(2)System.Timers.Timer
另外两个是专用的单线程定时器:

(3)System.Windows.Forms.Timer (Windows Forms 的定时器)
(4)System.Windows.Threading.DispatcherTimer (WPF 的定时器)
多线程定时器更加强大、精确并且更加灵活,而单线程定时器对于一些简单的更新 Windows Forms 和 WPF 控件的任务来说是安全的,并且更加便捷。

1.多线程定时器

System.Threading.Timer是最简单的多线程定时器:它仅仅有一个构造方法和两个普通方法(取悦于极简主义者,还有本书作者!)。在接下来的例子中,一个定时器在 5 秒钟之后调用Tick方法来打印 “ tick… “,之后每秒打印一次直到用户按下回车键:

using System;
using System.Threading;class Program
{static void Main(){// 首次间隔 5000ms,之后间隔 1000msTimer tmr = new Timer (Tick, "tick...", 5000, 1000);Console.ReadLine();tmr.Dispose();     // 停止定时器并执行清理工作
 }static void Tick (object data){// 这里运行在一个线程池线程上Console.WriteLine (data);     // 打印 "tick..."
 }
}

之后可以通过调用Change方法来改变定时器的时间间隔。如果你希望定时器只触发一次,可以指定Timeout.Infinite作为构造方法的最后一个参数。

2、.NET Framework 在System.Timers命名空间下提供了另一个名字相同的定时器类。它只是封装了 System.Threading.Timer,并在使用完全相同的底层引擎的前提下提供额外的便利。下面是增加功能的简介:

(1)实现了Component,允许用于 Visual Studio 的设计器中。
(2)Interval属性代替了Change方法。
(3)Elapsed事件代替了回调委托。
(4)Enabled属性用于开始或停止定时器(默认值是false)。
(5)Start和Stop方法,避免对Enabled属性感到困惑。
(6)AutoReset标识来指定是否为可重复的事件(默认为true)。
SynchronizingObject属性提供Invoke和BeginInvoke方法,用于在 WPF 和 Windows Forms 控件上安全调用方法。

using System;
using System.Timers;  // 命名空间是 Timers 而不是 Threadingclass SystemTimer
{static void Main(){Timer tmr = new Timer();    // 无需任何参数tmr.Interval = 500;tmr.Elapsed += tmr_Elapsed;  // 使用事件代替委托tmr.Start();          // 开启定时器
  Console.ReadLine();tmr.Stop();          // 停止定时器
  Console.ReadLine();tmr.Start();          // 重启定时器
  Console.ReadLine();tmr.Dispose();         // 永久停止定时器
 }static void tmr_Elapsed (object sender, EventArgs e){Console.WriteLine ("Tick");}
}

多线程定时器使用线程池来允许少量线程服务多个定时器。这意味着,回调方法或Elapsed事件每次可能会在不同的线程上触发。此外,不论之前的Elapsed是否完成执行,Elapsed总是几乎按时触发。因此,回调方法或事件处理器必须是线程安全的。

多线程定时器的精度依赖于操作系统,通常是在 10-20 ms 的区间。如果需要更高的精度,你可以使用本地互操作(native interop)来调用 Windows 多媒体定时器,可以让精度提升到 1 ms。它定义在 winmm.dll 中,首先调用timeBeginPeriod来通知操作系统你需要更高的定时器精度,然后调用timeSetEvent来启动多媒体定时器。当使用完成后,调用timeKillEvent停止定时器,最后调用timeEndPeriod通知操作系统你不在需要更高的定时器精度了。可以通过搜索关键字 dllimport winmm.dll timesetevent 在网上找到完整的例子。

3、它们暴露的成员都像System.Timers.Timer一样(Interval、Tick、Start和Stop),并且用法也类似。但是不同之处在于其内部是如何工作的。它们不是使用线程池来产生定时器事件,WPF 和 Windows Forms 定时器依赖于 UI 模型的底层消息循环机制(message pumping mechanism)。意味着Tick事件总是在创建该定时器的那个线程触发,在通常的程序中,它也就是管理所有 UI 元素和控件的那个线程。

单线程计时器比较安全,对于更新 Windows Forms controls或者WPF这种简单任务来说更方便。在WPF或Windows Forms中安全的调用方法的SynchronizingObject对象。
单线程计时器是被设计成属于他们执行环境的计时器,如果你在一个Windows服务应用程序中使用Windows Forms的Timer,timer 事件并不会被触发,只有在对应的环境下才会被触发。
像System.Timers.Timer一样,他们也提供了相同的成员(Interval,Tick,Start,Stop),但是他们内部的工作原理不同,WPF和Windows Forms的计时器使用消息循环机制来取代线程池产生消息的机制。

你可以不必考虑线程安全。
新的Tick在之前的Tick完成执行前不会触发。
你可以直接在Tick时间事件的处理代码中更新 UI 控件,而不需要调用Control.Invoke或Dispatcher.Invoke。
这听起来好的难以置信,直到你意识到使用这些定时器的程序并不是真正的多线程,不会有并行执行。一个线程服务于所有定时器,并且还处理 UI 事件。这带来了单线程定时器的缺点:

除非Tick事件处理器执行的很快,否则 UI 会失去响应。
这使得 WPF 和 Windows Forms 定时器仅适用于小任务,通常就是那些更新 UI 外观的任务(例如,显示时钟或倒计时)。否则,你就需要多线程定时器。

在精度方面,单线程定时器与多线程定时器类似(几十毫秒),但是通常精度更低,因为它们会被其它 UI 请求(或其它定时器事件)推迟。

单线程计时器基于Windows消息循环,应用程序会同步的处理计时器的消息。会发现UI界面相应速度比较慢。解决这个问题的方法是使用多线程计时器。
单线程计时器的缺点:除非Tick事件的处理代码执行的非常快,否则UI界面会变得响应很慢。所以 WPF和Windows Forms的计时器都非常适合小任务,尤其是界面更新的任务。例如时钟和计数显示。否则,你需要一个多线程计时器

用法举例

System.Timers.Timer是多线程定时器,如果一个Timer没有处理完成,到达下一个时间点,新的Timer同样会被启动,所以在使用Timer时需要注意,当设定的时间间隔远大于任务时间时,能够实现只在一个线程中执行,但是不能确保一定是一个。

https://blog.csdn.net/yl2isoft/article/details/51346469

如何prevent System.Timers.Timer的触发事件从一个线程池排队来执行?(How to prevent System.Timers.Timer from queuing for execution on a thread pool?)

There is a problem with standard System.Timers.Timer behaviour. The timer raise Elapsed event with some interval. But when time of execution inside Elapsed event handler exceed timer interval then thread pool begin queuing event handling. This is a problem in my case. This is because with my Elapsed event handler I fetch some data from database and doing something with it and finally save results back to database. But data handling should be provided only once. So, is there a way to prevent from queuing elapse events for System.Timers.Timer.(怎样只让事件执行一次)

As illustration for this issue you can consider next test program:

public class EntryPoint
{private static void TimeProc(object state, ElapsedEventArgs e){Console.WriteLine("Current time {0} on the thread {1}", DateTime.Now, Thread.CurrentThread.ManagedThreadId);Thread.Sleep(20000);}static void Main(string[] args){Console.WriteLine("Press <Enter> for finishing\n\n");ThreadPool.SetMaxThreads(10, 10);System.Timers.Timer MyTimer = new System.Timers.Timer(1000);MyTimer.Elapsed += new ElapsedEventHandler(TimeProc);MyTimer.Start();Console.ReadLine();MyTimer.Stop();}
}

解决方法:

public class EntryPoint{private static System.Timers.Timer MyTimer;private static void TimeProc(object state, ElapsedEventArgs e){Console.WriteLine("Current time {0} on the thread {1}", DateTime.Now, Thread.CurrentThread.ManagedThreadId);Thread.Sleep(20000);  MyTimer.Enabled = true;}static void Main(string[] args){Console.WriteLine("Press <Enter> for finishing\n\n");ThreadPool.SetMaxThreads(10, 10);MyTimer = new System.Timers.Timer(1000);   MyTimer.AutoReset = false;MyTimer.Elapsed += new ElapsedEventHandler(TimeProc);   MyTimer.Enabled = true;Console.ReadLine();}}

以上方法似乎不行,不知道有什么能够确保只在一个线程中执行触发函数的方法。

https://www.cnblogs.com/dotnet261010/p/7113523.html

https://www.cnblogs.com/yang-fei/p/6169089.html

转载于:https://www.cnblogs.com/janghe/p/8981282.html

C#中的三种timer相关推荐

  1. css中的三种基本定位机制

    css中的三种基本定位机制 a.普通文档流 b.定位:相对定位 绝对定位 固定定位 c.浮动 1.普通流中,元素位置由文档顺序和元素性质决定,块级元素从上到下依次排列,框之间的垂直距离由框的垂直mar ...

  2. Oracle的join默认为,Oracle中的三种Join方法详解

    这里将为大家介绍Oracle中的三种Join方法,Nested loop join.Sort merge join和Hash join.整理出来以便帮助大家学习. 基本概念 Nested loop j ...

  3. oracle hash join outer,CSS_浅谈Oracle中的三种Join方法,基本概念 Nested loop join: Outer - phpStudy...

    浅谈Oracle中的三种Join方法 基本概念 Nested loop join: Outer table中的每一行与inner table中的相应记录join,类似一个嵌套的循环. Sort mer ...

  4. (POST请求中的三种数据请求格式.application/x-www-form-urlencoded和multipart/form-data和application/json)

    (POST请求中的三种数据请求格式.application/x-www-form-urlencoded和multipart/form-data和application/json) applicatio ...

  5. MySQL buffer pool中的三种链

    三种page.三种list.LRU控制调优 一.innodb buffer pool中的三种页 1.free page:从未用过的页 2.clean page:干净的页,数据页的数据和磁盘一致 3.d ...

  6. 061 hive中的三种join与数据倾斜

    一:hive中的三种join 1.map join 应用场景:小表join大表 一:设置mapjoin的方式: )如果有一张表是小表,小表将自动执行map join. 默认是true. <pro ...

  7. .net core 注入中的三种模式:Singleton、Scoped 和 Transient

    从上篇内容不如题的文章<.net core 并发下的线程安全问题>扩展认识.net core注入中的三种模式:Singleton.Scoped 和 Transient 我们都知道在 Sta ...

  8. C#中的三种委托方式:Func委托,Action委托,Predicate委托

    C#中的三种委托方式:Func委托,Action委托,Predicate委托. Func,Action,Predicate全面解析 首先来说明Func委托,通过MSDN我们可以了解到,Func委托有如 ...

  9. 分析Java中的三种不同变量的区别

    1.首先分析Java中的三种不同变量的区别,如下表所示   概念 默认值 其他 类变量 也叫静态变量,是类中独立于方法之外的变量 用static 修饰 有默认初始值,系统自动初始化. 如boolean ...

最新文章

  1. 物体检测中的评价指标【文末赠书】
  2. golang 文件 文件夹 创建 读取 移动 复制 写入 遍历
  3. 入门一班 20181024 io监控free ps 网络状态 抓包
  4. java dbrecord_JFinal 独创 Db + Record 模式
  5. C#控制管理VisualSVN Server
  6. 工作392-选择Hbuilder x导入项目
  7. 二级vb笔试题库__全国计算机等级考试,2012年9月全国计算机二级VB笔试标准预测试卷试题六...
  8. vue中Axios网络请求之Vue知识点归纳(十)
  9. Maven 项目关于 plexus-utils:jar的错误解决
  10. Android字数限制的EditText实现方案研究
  11. 【体系结构】LGWR进程触发机制的理解
  12. 学习和研究下unity3d的四元数 Quaternion
  13. nvidia控制面板官方版-nvidia控制面板附安装教程
  14. 用flash做古诗动画_Flash制作跟我学 用遮罩技术制作古诗动画-FLASH课件制作(FLASH课件制作教程)-flash课件吧(湖北金鹰)...
  15. 计算机丢失MSVCR100.dll文件的解决办法
  16. linux安全-用户行为监控
  17. 《利用Python进行数据分析》第七章——数据清洗与准备
  18. 中秋福利!三维重建/SLAM/点云/相机标定/深度估计/缺陷检测课程
  19. 灵魂画手教你浅拷贝与深拷贝
  20. md500代码,异步电机,基于28379D,带无速度传感器控制,参数辨识,同步调制等功能

热门文章

  1. phpcmsv9 幻灯片管理模块_UTF8
  2. 30个Oracle语句优化规则详解
  3. 计算机文化基础教程教案,新版《计算机文化基础》教案.doc
  4. Linux下调试python
  5. 【知识发现】隐语义模型LFM算法python实现(三)
  6. MapReduce基础开发之十读写ORC File
  7. Pandas简明教程:二、Pandas基本数据结构-DataFrame与Series
  8. Apache Ignite与Apache Hive的个人理解与总结
  9. c语言单词翻译大全,c语言单词翻译
  10. 手机型号大全资料_电子元器件知识资料大全