简述

定时器是系统常用的组件之一,程序员可以根据自己的需求定制一个定时器类型,也可以使用.net内建的定时器类型。在.net中一共为程序员提供了3种定时器:

  • System.Windows.Forms.Timer类型
  • System.Threading.Timer类型
  • System.Timers.Timer类型

一、System.Windows.Forms.Timer

从这个定时器的命名空间可以看出,.net设计这个定时器的目的是为了方便程序员在Window Form中使用的定时器。当一个System.Windows.Forms.Timer类被构造时,当前定时器会和当前线程进行关联。而当计时器的计满后,一个定时器消息将被插入到当前线程的消息队列中。当前线程逐一处理消息中的所有消息,并一一派发给各自的处理方法。这样的机制和利用工作者进程定时有很大的区别。

System.Windows.Forms.Timer类型并没有涉及多线程的操作,定时器的设置、定时方法的执行都在同一个线程之上。这就意味着System.Windows.Forms.Timer并不能准确计时,尤其当消息阻塞时,定时器的误差将会更大,因为定时器消息只能等待在前面的所有消息处理完后才能得到处理。但是因为System.Windows.Forms.Timer类型的定时器并不涉及多线程的操作,因此是线程安全的,不会发生回调方法重入的问题。

主要使用步骤如下:

1.       System.Windows.Forms.Timer myTimer = new System.Windows.Forms.Timer();//实例化 

2.      myTimer.Tick += new EventHandler(函数名); //给timer挂起事件

3.      myTimer.Enabled = true;//使timer可用

4.       myTimer.Interval = n; //设置时间间隔,以毫秒为单位

5.       myTimer.Stop(); //如果要暂停计时则使用Stop()方法

6.       myTimer.Enabled = false;//若要停止使用timer,则使之不可用

System.Windows.Forms.Timer例程:

 
 
  1. using System.Windows.Forms;
  2. namespace WindowsFormsApplication1
  3. {
  4. public partial class Form1 : Form
  5. {
  6. public Form1()
  7. {
  8. InitializeComponent();
  9. }
  10. //实例化一个timer
  11. System.Windows.Forms.Timer myTimer = new System.Windows.Forms.Timer();
  12. private void StartTimeBtn_Click(object sender, EventArgs e)//开始计时
  13. {
  14. //给timer挂起事件
  15. myTimer.Tick += new EventHandler(Callback);
  16. //使timer可用
  17. myTimer.Enabled = true;
  18. //设置时间间隔,以毫秒为单位
  19. myTimer.Interval = 1000; //1s
  20. }
  21. private void StopTimeBtn_Click(object sender, EventArgs e)//停止计时
  22. {
  23. //计时开始
  24. myTimer.Stop();
  25. }
  26. //回调函数
  27. private void Callback(object sender, EventArgs e)
  28. {
  29. //获取系统时间 20:16:16
  30. textBox1.Text = DateTime.Now.ToLongTimeString().ToString();
  31. }
  32. }
  33. }

二、System.Threading.Timer

这个定时器类的使用相对复杂,但同时它也是最优化的非常常用的一个定时器类型。System.Threading.Timer的定时方法在独立的线程上执行,定时时间更为准确。使用回调方法的计时器,而且由线程池线程服务,简单且对资源要求不高。所有的对象有一个线程控制,当下一个定时到达时,该线程会负责在线程中获得一个新的工作者线程,用以执行相应的回调方法。

虽然这个定时器是相对最优化的一个定时器类型,但是从其机制上来讲,其并不是线程安全的,可能会出现回调方法重入的问题。解释下方法重入,是一个有关多线程编程的概念,意思大概是:同一程序中,多个线程同时运行时,就可能发生同一个方法被多个进程同时调用的情况。当这个方法中存在一些非线程安全的代码时,方法重入会导致数据发生同步错误的bug。

 主要使用步骤如下: 

1、实例构造一个线程定时器。

System.Threading.Timer  mytimer =  new System.Threading.Timer(new  System.Threading.TimerCallback(timerCall), null, 0, 1000);
 
 

★timer构造函数资料:

 
 
  1. public Timer(
  2. TimerCallback callback, // TimerCallback 委托,表示要执行的方法。
  3. object state,    //一个包含回调方法要使用的信息的对象,或者为 null
  4. int dueTime, //第一次执行调用方法等待时间 // 指定零 ( 0) 可立即启动计时器。
  5. //指定零 (-1) 表示本定时器被禁用。
  6. //Change(Int32,Int32)方法可以改写定时器参数
  7. int period // 执行间隔时间
  8.            //如果 period 为零 (0) 或 -1 毫秒,而且dueTime 为正,则只会调用一次方法;
  9. //然后计时器的定时行为将被禁用,使用 Change 可以改写定时器参数。
  10. )

2、编写timerCall回调函数

     格式:private  void  timerCall(object   xxxxx)  { .......; .......;}

3、使用Change(Int32,Int32)方法来修改定时器参数实现停止、重新开始等。

4、使用Dispose()方法释放定时器资源。

System.Threading.Timer例程:

 
 
  1. public startTimer()
  2. {
  3. //定义一个对象
  4. System.Threading.Timer timer = new System.Threading.Timer(
  5. new System.Threading.TimerCallback(Say), null,
  6. 0, 1000); //1S定时器
  7. }
  8. //函数形式参数必须是object格式
  9. public void Say(object a )
  10. {
  11. Console.WriteLine( "你好");
  12. }

 

 
 
  1. using System;
  2. using System.Threading;
  3. class TimerExample
  4. {
  5. static void Main()
  6. {
  7. //创建一个事件来指示超时计数阈值
  8. //定时器回调。
  9. AutoResetEvent autoEvent = new AutoResetEvent( false);
  10. StatusChecker statusChecker = new StatusChecker( 10);
  11. // 创建一个推断的委托,调用计时器的方法。
  12. TimerCallback tcb = statusChecker.CheckStatus;
  13. // 创建一个指示委托调用的计时器。
  14. Console.WriteLine( "{0} Creating timer.\n",DateTime.Now.ToString( "h:mm:ss.fff"));
  15. Timer stateTimer = new Timer(tcb, autoEvent, 1000, 250);
  16. // 当自动事件信号,改变周期到每一个
  17. autoEvent.WaitOne( 5000, false);
  18.      stateTimer.Change( 0, 500);
  19. Console.WriteLine( "\nChanging period.\n");
  20. //当autoevent信号的第二次,处置
  21. autoEvent.WaitOne( 5000, false);
  22. stateTimer.Dispose();
  23. Console.WriteLine( "\nDestroying timer.");
  24. }
  25. }
  26. class StatusChecker
  27. {
  28. private int invokeCount;
  29. private int maxCount;
  30. public StatusChecker(int count)
  31. {
  32. invokeCount = 0;
  33. maxCount = count;
  34. }
  35. // 此方法由定时器委托调用。
  36. public void CheckStatus(Object stateInfo)
  37. {
  38. AutoResetEvent autoEvent = (AutoResetEvent)stateInfo;
  39. Console.WriteLine( "{0} Checking status {1,2}.",
  40. DateTime.Now.ToString( "h:mm:ss.fff"),
  41. (++invokeCount).ToString());
  42. if(invokeCount == maxCount)
  43. {
  44. // 重置计数器和信号主信号。
  45. invokeCount = 0;
  46. autoEvent.Set();
  47. }
  48. }
  49. }

三、 System.Timers.Timer类

这是一个相对较旧的类型,它和System.Threading.Timer一样,可以由工作者线程来执行回调方法,但同时它也可以在IDE环境中被拖到窗体控件上,这个时候它的行为非常类似于System.Windows.Forms.Timer类型,在消息过多时其定时并不准确。  System.Timers.Timer可以视为System.Threading.Timer的一个包装,其类型设计相对古老,不建议使用该定时器。

主要使用步骤如下: 

 
 
  1. 1. System.Timers.Timer myTimer = new System.Timers.Timer( 1000 ); //实例化 设置时间间隔
  2. 2. myTimer .Elapsed += new System.Timers.ElapsedEventHandler(函数名); //给timer挂起事件
  3. 3. myTimer .AutoReset = true; //设置是执行一次(false)还是一直执行(true);
  4. 4. myTimer .Enabled = true; //是否执行System.Timers.Timer.Elapsed事件;

 System.Timers.Timer例程代码:

 
 
  1. System.Timers.Timer t = newSystem.Timers.Timer( 5000); //设置时间间隔为5秒
  2. private void Form1_Load(object sender,EventArgs e)
  3. {
  4. t.Elapsed += newSystem.Timers.ElapsedEventHandler(Timer_TimesUp);
  5. t.AutoReset = false; //每到指定时间Elapsed事件是触发一次(false),还是一直触发(true)
  6. }
  7. private void btnStart_Click(objectsender, EventArgs e)
  8. {
  9. t.Enabled = true; //是否触发Elapsed事件
  10. t.Start();
  11. }
  12. private void Timer_TimesUp(objectsender, System.Timers.ElapsedEventArgs e)
  13. {
  14. //到达指定时间5秒触发该事件输出 HelloWorld!!!!
  15. System.Diagnostics.Debug.WriteLine( "HelloWorld!!!!");
  16. }
  17. private void btnStop_Click(objectsender, EventArgs e)
  18. {
  19. t.Stop();
  20. System.Diagnostics.Debug.WriteLine( "未到指定时间5秒提前终结!!!");
  21. }

参考链接:http://www.cnblogs.com/DebugLZQ/archive/2012/08/05/2623669.html

     https://blog.csdn.net/zhouqt/article/details/51779264

      http://www.360doc.com/content/11/0812/11/1039473_139824496.shtml

原文链接:https://www.cnblogs.com/Tanghongchang/p/9644766.html

原文作者:酒不醉心