C#执行定时计算限制操作(计时器)

2024-04-08 08:58

本文主要是介绍C#执行定时计算限制操作(计时器),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

c#执行定时计算限制操作(计时器)

在.Net Framework Class Library(FCL)中,System.Threading命名空间下定义了一个Timer类,这就是常用的一个计时器。实际上FCL总共提供了如下几种计时器:

1、System.Threading.Timer

在实际的开发中,这个类出现和使用频率非常高,下面就重点谈谈它的一些基础概念和应用实例。

(1)、常用的构造器

1
2
3
4
public  Timer(TimerCallback callback, object  state, int  dueTime, int  period);
public  Timer(TimerCallback callback, object  state, long  dueTime, long  period);
public  Timer(TimerCallback callback, object  state, TimeSpan dueTime, TimeSpan period);
public  Timer(TimerCallback callback, object  state, uint  dueTime, uint  period);

参数说明:

a、callback:望文生意,肯定表示一个回调,它是标识希望由一个线程池线程回调的方法。当然它的类型必须和System.Threading.TimerCallback委托类型匹配,如下所示:

1
public  delegate  void  TimerCallback( object  state);

b、state:每次调用回调方法,向回调方法传递的状态数据,如没有,可以为null;

c、dueTime:在首次调用回调方法之前要等待多少毫秒。如希望立刻调用回调方法,该参数指定为0即可。

d、period:指定了以后每次调用回调方法之前要等待多少毫秒(理解成下一次和本次调用的时间间隔即可)。如果为该参数传递Timeout.Infinite(或者直接写-1),线程池线程只调用回调方法一次(那也就没有必要用计时器了)。

 

(2)、基本工作原理

在内部(原文应该指CLR内),线程池为所有Timer对象只使用了一个线程。这个线程知道下一个Timer对象在什么时候到期(计时器还有多久触发)。下一个Timer对象到期时,线程就会唤醒,在内部调用ThreadPool的QueueUserWorkItem,将一个工作项添加到线程池的队列中,使你的回调方法得到调用。

注意:如果回调方法的执行时间很长,计时器可能在上个回调还没有完成的时候再次触发(对于执行时间很长的任务,实际开发中通常不使用线程池,而是直接使用new一个Thread)。这可能会造成多个线程池线程同时调用你的方法(方法的重叠覆盖?)。为了解决这个问题,Jeffrey Richter建议我们这样使用Timer:

a、为period指定Timeout.Infinite。这样,计时器就只触发一次;

b、在回调方法中,调用Timer的Change方法来指定一个新的dueTime,并再次为period指定Timeout.Infinite。Change方法的几个重载版本:

1
2
3
4
5
6
7
public  bool  Change( int  dueTime, int  period);
public  bool  Change( long  dueTime, long  period);
public  bool  Change(TimeSpan dueTime, TimeSpan period);
public  bool  Change( uint  dueTime, uint  period);

Timer还有一个Dispose方法,允许完全取消计时器。

(3)、示例代码

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
31
32
internal  static  class  TimerDemo
{
     private  static  Timer s_timer;
     public  static  void  Main()
     {
         Console.WriteLine( "Main thread: starting a timer" );
         using  (s_timer = new  Timer(ComputeBoundOp, 5, 0, Timeout.Infinite))
         {
             Console.WriteLine( "Main thread: Doing other work here..." );
             Thread.Sleep(10000);  // Simulating other work (10 seconds)
         } // Calls Dispose to cancel the timer now
         Console.Read();
     }
     // This method's signature must match the TimerCallback delegate
     private  static  void  ComputeBoundOp(Object state)
     {
         // This method is executed by a thread pool thread
         Console.WriteLine( "In ComputeBoundOp: state={0}" , state);
         Thread.Sleep(1000);  // Simulates other work (1 second)
         // Have the Timer call this method again in 2 seconds
         s_timer.Change(2000, Timeout.Infinite);
         // When this method returns, the thread goes back
         // to the pool and waits for another work item
     }
}

这个示例是<<CLR via C#>>中的源码,虽然简单,却贯穿实现了创建Timer,Timer定时工作到Timer的销毁一整个生命周期。using语句我们可以使用try finally替换,在finally语句块中显式调用Dispose方法。注意,程序中输出了四次“In ComputeBoundOp: state=5”,您不妨多想一想为什么只有4次。而如果我们把Main方法的这一行:

1
Thread.Sleep(10000);  // Simulating other work (10 seconds)

注释掉,程序有时候会抛出“程序无法访问已释放对象”的异常,而且每次输出的结果可能还不一样。之所以这么说,是因为using语句不能保证静态Timer立刻被垃圾回收器回收。您不妨自己动手一试。

 

2、System.Timers.Timer

这个类据说基本上就是System.Threading.Timer的包装过的“整过容的副产品”。当计时器到期触发时,会导致CLR将事件放到线程池的队列中。该类派生自System.ComponentModel的Component类,允许VS将计时器对象放到设计平面上,并公开了它的属性和事件(不就是一个控件吗,难道?)。

 

3、System.Windows.Forms.Timer

构造一个该类的实例,相当于告诉Windows将一个计时器和调用线程关联。当这个计时器被触发时,Windows将一条及时消息注入线程的消息队列。线程必须执行一个消息泵来提取这些消息,并把它们派遣给想要的回调方法。注意,所有这些工作都只由一个线程完成——设置计时器的线程保证就是执行回调方法的线程。这还意味着你的计时器方法不会由多个线程并发执行。

 

4、System.Windows.Threading.DispatcherTimer

这个类是System.Windows.Forms的Timer在WPF和SilverLight应用程序中的等价物。通俗地说,这个类也就是System.Windows.Forms的Timer在WPF和SilverLight下的一个马甲。擅于改头换面新瓶装旧酒在MS的技术领域中也不是一天两天的事情了。

 

在Jeffrey Richter 的<<CLR via C#>>中关于计时器有很精练的说明,其中他说“事实上,我个人从来不用System.Timers.Timer类,建议你也不要用它”,他的这种态度蛮好玩的。反正对于这个类我倒是用过不少,而且一直没出过什么差错,就算是”山寨“或者”马甲“也不是从来都是一无是处。

 

参考:

Jeffrey Richter <<CLR via C#>>

http://msdn.microsoft.com/zh-cn/library/system.threading.timer.aspx

原文地址:

http://www.cnblogs.com/jeffwongishandsome/archive/2010/11/15/1877837.html

这篇关于C#执行定时计算限制操作(计时器)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/885131

相关文章

SQL中JOIN操作的条件使用总结与实践

《SQL中JOIN操作的条件使用总结与实践》在SQL查询中,JOIN操作是多表关联的核心工具,本文将从原理,场景和最佳实践三个方面总结JOIN条件的使用规则,希望可以帮助开发者精准控制查询逻辑... 目录一、ON与WHERE的本质区别二、场景化条件使用规则三、最佳实践建议1.优先使用ON条件2.WHERE用

Linux链表操作方式

《Linux链表操作方式》:本文主要介绍Linux链表操作方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、链表基础概念与内核链表优势二、内核链表结构与宏解析三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势六、典型应用场景七、调试技巧与

C#如何去掉文件夹或文件名非法字符

《C#如何去掉文件夹或文件名非法字符》:本文主要介绍C#如何去掉文件夹或文件名非法字符的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#去掉文件夹或文件名非法字符net类库提供了非法字符的数组这里还有个小窍门总结C#去掉文件夹或文件名非法字符实现有输入字

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据

Java Multimap实现类与操作的具体示例

《JavaMultimap实现类与操作的具体示例》Multimap出现在Google的Guava库中,它为Java提供了更加灵活的集合操作,:本文主要介绍JavaMultimap实现类与操作的... 目录一、Multimap 概述Multimap 主要特点:二、Multimap 实现类1. ListMult

C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式

《C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式》Markdown凭借简洁的语法、优良的可读性,以及对版本控制系统的高度兼容性,逐渐成为最受欢迎的文档格式... 目录为什么要将文档转换为 Markdown 格式使用工具将 Word 文档转换为 Markdown(.

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

Python中文件读取操作漏洞深度解析与防护指南

《Python中文件读取操作漏洞深度解析与防护指南》在Web应用开发中,文件操作是最基础也最危险的功能之一,这篇文章将全面剖析Python环境中常见的文件读取漏洞类型,成因及防护方案,感兴趣的小伙伴可... 目录引言一、静态资源处理中的路径穿越漏洞1.1 典型漏洞场景1.2 os.path.join()的陷

C#代码实现解析WTGPS和BD数据

《C#代码实现解析WTGPS和BD数据》在现代的导航与定位应用中,准确解析GPS和北斗(BD)等卫星定位数据至关重要,本文将使用C#语言实现解析WTGPS和BD数据,需要的可以了解下... 目录一、代码结构概览1. 核心解析方法2. 位置信息解析3. 经纬度转换方法4. 日期和时间戳解析5. 辅助方法二、L

Python使用Code2flow将代码转化为流程图的操作教程

《Python使用Code2flow将代码转化为流程图的操作教程》Code2flow是一款开源工具,能够将代码自动转换为流程图,该工具对于代码审查、调试和理解大型代码库非常有用,在这篇博客中,我们将深... 目录引言1nVflRA、为什么选择 Code2flow?2、安装 Code2flow3、基本功能演示