C# 三个Timer

栏目: ASP.NET · 发布时间: 4年前

内容简介:在C#中存在3种常用的这个

在C#中存在3种常用的 Timer

  1. System.Windows.Forms.Timer
  2. System.Timers.Timer
  3. System.Threading.Timer

零、System.Windows.Forms.Timer

这个 Timer 是单线程的,也就是说只要它运行,其他线程就要等着。

C# 三个Timer

这个 Timer 有如下特点:

  1. 完全基于UI线程,定时器触发时,操作系统把定时器消息插入线程消息队列中,调用线程执行一个消息泵提取消息,然后发送到回调方法Tick中;
  2. 使用 Start Stop 启动和停止 Timer;
  3. UI操作过长会导致 Tick 丢失;
  4. 可以使用委托Hook Tick事件;
  5. 精确度不高;
  6. 通过将 Enabled 设置为 True ,使 Timer 自动运行

从上面的第一个特点可以得知,该 Timer 会造成 WinForm UI 假死,因此如果需要定时处理大量计算或者大量IO操作的任务,不建议使用该 Timer ,接下来我们看一个例子体会一下在IO操作的情况下出现的假死情况:

我们在Form中放入两个Button 一个Lable和一个Timer

C# 三个Timer

private void Button_Click(object sender, EventArgs e)
{
    timer.Interval = 1000;
    timer.Tick += Timer_Tick;
    timer.Start();
}

private void Timer_Tick(object sender, EventArgs e)
{
    for (int i = 0; i < 10000; i++)
    {
        File.AppendAllText(Directory.GetCurrentDirectory()+"test.txt", i.ToString());
        this.label_output.Text = "当前操作:插入数字" + i;
    }
}

我们单击计算按钮,我们会发现WinForm出现了假死(无法移动窗口、按钮无法点击等)

一、System.Timers.Timer

该 Timer 是基于服务器的计时器,是为在多线程环境中用于辅助线程而设计的,可以在线程间移动来处理引发的 Elapsed 事件,比上一个计时器更加精确。

该 Timer 有如下特点:

  1. 通过 Elapsed 设置回掉处理事件,且 Elapsed 是运行在 ThreadPool 上的;
  2. 通过 Interval 设置间隔时间;
  3. AutoReset 设置为 False 时,只在到达第一次时间间隔后触发 Elapsed 事件;
  4. 是一个多线程计时器;
  5. 无法直接调用 WinForm 上的控件,需要使用 委托
  6. 主要用在 Windows 服务中。

同样我们通过代码来看一下该 Timer 计时器怎么使用:

System.Timers.Timer timersTimer = new System.Timers.Timer();
private void Button_Click(object sender, EventArgs e)
{
    timersTimer.Interval = 1000;
    timersTimer.Enabled = true;
    timersTimer.Elapsed += TimersTimer_Elapsed;
    timersTimer.Start();
}

private void TimersTimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
{
    for (int i = 0; i < 10000; i++)
    {
        this.BeginInvoke(new Action(() =>
        {
            this.label_output.Text="当前时间:"+DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss");
        }), null);
    }
}

private void Button1_Click(object sender, EventArgs e)
{
    timersTimer.Stop();
}

运行上面代码,会发现WinForm界面假死的情况消失了。

二、System.Threading.Timer

该 Timer 同样也是一个多线程的计时器,它有如下特点:

  1. 多线程
  2. 和前两个计时器相比没有 StartStop 方法,如果要停止计时器,必须调用 Dispose 方法来销毁 Timer 对象;
  3. 调用 Dispose 方法后并不能马上停止所有的计时器,这是因为间隔时间小于执行时间时多个线程运行造成的,多个线程无法同时停止;

C# 三个Timer

  1. 是一个轻量级的计时器;
  2. 所有的参数全部在构造函数中进行了设置;
  3. 可以设置启动时间;
  4. 不建议再 WinForm 程序中使用。

我们来看一下代码(在控制台应用程序中输入以下代码):

static System.Threading.Timer threadingTimer;
static int numSum = 0;
static void Main(string[] args)
{
    threadingTimer = new System.Threading.Timer(new System.Threading.TimerCallback(threadingTimer_Elapsed), null, 0, 1000);
    Console.Read();
}
private static void threadingTimer_Elapsed(object state)
{
    for (int i = 0; i < 10000; i++)
    {
        numSum++;
        Console.WriteLine("输出数字:"+i);
    }

    if (numSum > 10000)
    {
        threadingTimer.Dispose();
        Console.WriteLine("结束");
    }
}

注意:当我们不再需要多线程Timer计时器的时候,我们可以调用 Dispose 方法释放所占有的资源。但是因为Timer计时器是按线程池线程来安排回调执行的,因此回调可能发生在 Dispose方法的重载被调用之后,所以我们可以使用可使用 Dispose(WaitHandle) 方法等待所有回掉完成。


以上所述就是小编给大家介绍的《C# 三个Timer》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

标签

标签

Gene Smith / 张军、陈军亮 / 机械工业出版社 / 2012-6 / 59.00元

本书对标记系统这一概念的内涵和外延进行了系统化的、深入浅出的阐述。从什么是标记系统、标记系统有什么价值,到标记系统的架构和与其他分类系统的对比,再到标签的呈现方式和标记系统的实现细节,作者都用通俗易懂的语言进行了阐述,并附有详细的示例和具体的案例研究。本书的每一章都涵盖了标记系统的一个方面,主要内容包括:标记系统的模型、价值、架构,标签的分类、可视化、管理方法,最后介绍标记系统设计方法。本书带领读......一起来看看 《标签》 这本书的介绍吧!

JS 压缩/解压工具
JS 压缩/解压工具

在线压缩/解压 JS 代码

URL 编码/解码
URL 编码/解码

URL 编码/解码

HEX HSV 转换工具
HEX HSV 转换工具

HEX HSV 互换工具