作者:feiying008

在开发一套视觉系统时,发现系统内存一直不断增加,直至系统内存爆满。一开始还以为是程序内存泄露,是图像操作算法写的有问题,但是,发现如果电机轴如果 不运行的状态下,每隔一秒进行视觉运算,发现内存增加后,但操作完会立即释放内存。而不会一直增加。这个误区耽误了我很长时间,一直以为是视觉算法的写法 有问题。但排除了这个可能后,就在想为什么加上电机轴的运转后,就内存直上不下了呢。有时经过数日的摸索,终于发现,原来罪魁祸首是.NET 的Timer定时器。因为在电机轴运行时,要时刻监控电机轴的运行状态,从而根据不同的状态做出相应的动作。而实现这个功能就是用Timer计时器。我用 的是System.Timer。这个计时器的Elapsed事件是启用的一个新的线程进行间隔操作,问题就出在这里。每次间隔触发就启用一个多线程,那多 线程的数量就不断暴增,而启用的多线程的内存回收是有.net 垃圾回收机制自动进行回收的。而这个回收时间是不确定的。所以,就导致了内存暴增的情况。嗯,就说到这里,算是一个总结。

昨天又做了一些验证,上 面是说法不正确,Elapsed事件是启用了一个线程,但此线程是从次线程启用的,所以,严格意义上来讲是不会出现线程无线增加的情况的。那暴增的内存是 从哪里来的呢?经过实验验证,如果将Timer的Interval值设定为小与100的值,则内存会不断增加,值越小,增加的越快。如果值大于100的 话,则值也会增加,但每隔一段时间都会降下去,我想应该是被.net垃圾回收机制给回收了。这才是真正的真因。

下面这篇文章很详细地讲解了定时器的知识,当然,.net 环境下一共有四种定时器。但是最优的还是下面讲解的这个。

在系统开发过程中经常用到定时器进行定时处理,比如比较常见的邮件群发、实时更新论坛的在线人数、文章数、点击率等。  很多情况下,我们不能对某一状态或者某一行为进行实时监控,所以就希望系统能够实现这一功能。通过多线程技术可以使得定时器的性能更高。

尽管定时器能够自动处理或者一些批处理操作,但是定时器也给系统带来一定的安全隐患,特别是当定时进行的操作出现bug时,如果没有对Exception 做出及时的处理,系统资源将会大大的浪费,严重的情况下,可能导致系统崩溃。因此,对于定时器的使用一定要慎重,至少要保证定时处理的行为出现异常的可能 性很小,并在出现Exception的情况下及时处理。

System.Threading.Timer 是一个非常常用的定时器类,是一个使用回调方法的计时器,而且由线程池线程服务,简单且对资源要求不高。

public Timer (
    TimerCallback callback,
    Object state,
    TimeSpan dueTime,
    TimeSpan period
)

参数

callback

一个 TimerCallback 委托,表示要执行的方法。

state

一个包含回调方法要使用的信息的对象,或者为 空引用(在 Visual Basic 中为 Nothing)。

dueTime

TimeSpan,表示在 callback 参数调用它的方法之前延迟的时间量。指定 -1 毫秒以防止启动计时器。指定零 (0) 以立即启动计时器。

period

在调用 callback 所引用的方法之间的时间间隔。指定 -1 毫秒可以禁用定期终止。

方法、原理

  • 使用 TimerCallback 委托指定希望 Timer 执行的方法。计时器委托在构造计时器时指定,并且不能更改。此方法不在创建计时器的线程上执行,而是在系统提供的 ThreadPool 线程上执行。
  • 创建计时器时,可以指定在第一次执行方法之前等待的时间量(截止时间)以及此后的执行期间等待的时间量(时间周期)。可以使用 Change 方法更改这些值或禁用计时器。
  • 当不再需要计时器时,请使用 Dispose 方法释放计时器持有的资源。如果希望在计时器被释放时接收到信号,请使用接受 WaitHandle 的 Dispose(WaitHandle) 方法重载。计时器已被释放后,WaitHandle 便终止。
  • 由计时器执行的回调方法应该是可重入的,因为它是在 ThreadPool 线程上调用的。

备注:

在超过 dueTime 以后及此后每隔 period 时间间隔,都会调用一次 callback 参数所指定的委托。
      如果 dueTime 为零 (0),则立即调用 callback。如果 dueTime 是 -1 毫秒,则不会调用 callback;计时器将被禁用,但通过调用 Change 方法可以重新启用计时器。
      如果 period 为零 (0) 或 -1 毫秒,而且 dueTime 为正,则只会调用一次 callback;计时器的定期行为将被禁用,但通过使用 Change 方法可以重新启用该行为。

最简单的定时器

using System;
using System.Threading;

public class TestTimer
{
    /// <summary>
    /// 定时器
    /// </summary>
    private Timer iTimer;
    /// <summary>
    /// constructor
    /// </summary>
    public TestTimer()
    {
        iTimer = new System.Threading.Timer(new TimerCallback(Doing));
        iTimer.Change(TimeSpan.FromMinutes(5), TimeSpan.FromMinutes(5));
    }
    /// <summary>
    /// 
    /// </summary>
    /// <param name="nObject"></param>
    public void Doing(object nObject)
    {
        //do something
    }
}

 一个比较完整的计时器:

下面是我设计的一个简单实例。在一个问卷调查系统中,每一张问卷都有其终止日期,当到达了终止日期时,需要系统自动将其关闭。这就需要定时器对问卷的状态
和终止日期进行实时监控,及时关闭。这里采用了一个简单的单件模式来操作、控制定时器。 这里主要的操作包括定时器开始、终止、执行一次。


    /// <summary>
    /// 管理类
    /// </summary>
    public class PaperManager
    {
        /// <summary>
        /// 定时器
        /// </summary>
        private Timer iTimer;
        /// <summary>
        /// 启动时间
        /// </summary>
        private TimeSpan dueTime;
        /// <summary>
        /// 方法调用间隔
        /// </summary>
        private TimeSpan period;
        /// <summary>
        /// 委托
        /// </summary>
        private TimerCallback timerDelegate;   
        /// <summary>
        /// 静态实例
        /// </summary>
        private static readonly PaperManager self = new PaperManager();
        /// <summary>
        /// 构造函数
        /// </summary>
        public PaperManager()
        {
            timerDelegate = new TimerCallback(CheckStatus);
        }
        /// <summary>
        /// 
        /// </summary>
        /// <returns></returns>
        public static PaperManager getInstance()
        {
            return self;
        }
        /// <summary>
        /// 设置启动时间间隔
        /// </summary>
        /// <param name="days">天</param>
        /// <param name="hours">小时</param>
        /// <param name="minutes">分钟</param>
        /// <param name="seconds">秒</param>
        /// <param name="milisecond">毫秒</param>
        public void setDueTime(int days, int hours, int minutes, int seconds, int milisecond)
        {
            dueTime = new TimeSpan(days, hours, minutes, seconds, milisecond);
        }
        /// <summary>
        /// 设置回调时间间隔
        /// </summary>
        /// <param name="days">天</param>
        /// <param name="hours">小时</param>
        /// <param name="minutes">分钟</param>
        /// <param name="seconds">秒</param>
        /// <param name="milisecond">毫秒</param>
        public void setPeriod(int days, int hours, int minutes, int seconds, int milisecond)
        {
            period = new TimeSpan(days, hours, minutes, seconds, milisecond);
        }
        /// <summary>
        /// 开始
        /// </summary>
        public void Start()
        {
            AutoResetEvent autoEvent = new AutoResetEvent(false);
            dueTime = TimeSpan.FromSeconds(0);
            period = TimeSpan.FromSeconds(10);
            iTimer = new Timer(timerDelegate, autoEvent, dueTime, period);
            autoEvent.WaitOne(5000, false);
            iTimer.Change(dueTime, period);
        }
        /// <summary>
        /// 停止
        /// </summary>
        public void Stop()
        {
            iTimer.Dispose();
        }
        /// <summary>
        /// 执行一次
        /// </summary>
        public void ExcuteOneTime()
        {
            if (iTimer != null)
            {
                iTimer.Dispose();
            }
            //如果 period 为零 (0) 或 -1 毫秒,而且 dueTime 为正,则只会调用一次 callback;
            //计时器的定期行为将被禁用,但通过使用 Change 方法可以重新启用该行为。
            setDueTime(0, 0, 0, 0, 1);
            setPeriod(0, 0, 0, 0, -1);
            AutoResetEvent autoEvent = new AutoResetEvent(false);
            iTimer = new Timer(timerDelegate, autoEvent, dueTime, period);
            autoEvent.WaitOne(5000, false);
            iTimer.Change(dueTime, period);
        }
        /// <summary>
        /// 行为
        /// </summary>
        /// <param name="nObject"></param>
        public void CheckStatus(object nObject)
        {
            AutoResetEvent autoEvent = (AutoResetEvent)nObject;
            if (ExcuteUpdate())
            {                
                autoEvent.Set();
            }
        }
        /// <summary>
        /// 更新
        /// </summary>
        /// <returns></returns>
        private bool ExcuteUpdate()
        {
            try
            {
                //应该从数据库获得Paper对象的集合,这里简略
                //List<Paper> paperList = getPaperList();
                List<Paper> paperList = new List<Paper>();
                foreach (Paper item in paperList)
                {
                    if (item.EndTime <= DateTime.Now)
                    {
                        if (item.Status == Paper.StatusOfNormal)
                        {
                            item.Status = Paper.StatusOfTerminate;
                        }
                    }
                }
                ////执行数据更新,这里省略
                return true;
            }
            catch
            {
                return false;
            }
        }    
    }

这是问卷的实体类,只是简单的列出必要的属性。


    /// <summary>
    /// 实体类
    /// </summary>
    public class Paper
    {
        /// <summary>
        /// 终止时间
        /// </summary>
        public DateTime EndTime;
        /// <summary>
        /// 状态
        /// </summary>
        public int Status;
        /// <summary>
        /// 正常
        /// </summary>
        public const int StatusOfNormal = 1;
        /// <summary>
        /// 终止
        /// </summary>
        public const int StatusOfTerminate = 2;
        /// <summary>
        /// 
        /// </summary>
        /// <param name="status"></param>
        /// <param name="endTime"></param>
        public Paper(int status, DateTime endTime)
        {
            Status = status;
            EndTime = endTime;
        }
    }

.net中 Timer定时器的更多相关文章

  1. 浅析linux内核中timer定时器的生成和sofirq软中断调用流程(转自http://blog.chinaunix.net/uid-20564848-id-73480.html)

    浅析linux内核中timer定时器的生成和sofirq软中断调用流程 mod_timer添加的定时器timer在内核的软中断中发生调用,__run_timers会spin_lock_irq(& ...

  2. 浅析linux内核中timer定时器的生成和sofirq软中断调用流程【转】

    转自:http://blog.chinaunix.net/uid-20564848-id-73480.html 浅析linux内核中timer定时器的生成和sofirq软中断调用流程 mod_time ...

  3. asp.net中Timer定时器在web中无刷新的使用

    最近在做一个项目的时候,web端的数据需要与数据源进行实时同步,并保证数据的准确性,当时,考虑到使用ajax异步刷新技术.但后来在网上查找相关资料时,发现这样做,太浪费资源了,因为ajax的提交请求不 ...

  4. C#中Timer定时器的使用示例

    关于C#中timer类 在C#里关于定时器类就有3个: 1.定义在System.Windows.Forms里 2.定义在System.Threading.Timer类里 3.定义在System.Tim ...

  5. 关于C#中Timer定时器的重入问题解决方法(也适用于多线程)

    项目中用到了定时器随着服务启动作定时任务,按指定的准点时间定时执行相关操作,但是在指定准点时间内我只想让它执行一次,要避免重入问题的发生. 首先简单介绍一下timer,这里所说的timer是指的Sys ...

  6. 【Boost】boost库中timer定时器 2

    博客转载自:http://blog.csdn.net/yockie/article/details/40386145 先跟着boost文档中asio章节的指南中的几个例子学习一下使用: 所有的Asio ...

  7. 【Boost】boost库中timer定时器 1

    博客转载自:http://blog.csdn.net/liujiayu2/article/details/50384537 同步Timer asio中提供的timer名为deadline_timer, ...

  8. C#中WebService 的 Timer定时器过段时间后自动停止运行

    我用.net做的一个Timer定时器,定时获取短信并给予回复,但大概过了十几个小时以后,Timer定时器会自动停止,再发送短信就不能收到回复,需要在服务器中重新运行定时器才可以,请教各位! 我是在.n ...

  9. Java 中Timer和TimerTask 定时器和定时任务使用的例子

    转自:http://blog.csdn.net/kalision/article/details/7692796 这两个类使用起来非常方便,可以完成我们对定时器的绝大多数需求 Timer类是用来执行任 ...

随机推荐

  1. [解决] Error Code: 1044. Access denied for user 'root'@'%' to database

    今天在测试集群用的mysql上,遇到个权限的问题: SQLException : SQL state: 42000 com.mysql.jdbc.exceptions.jdbc4.MySQLSynta ...

  2. dbms_output.put与put_line

    BEGIN DBMS_OUTPUT.ENABLE (buffer_size => NULL);--with no limit on the output. dbms_output.put('a' ...

  3. Java的Comparable接口的一个陷阱

    转载自:http://my.oschina.net/jack230230/blog/56339 Java的Comparable接口提供一个对实现了这个接口的对象列表进行排序的办法.原始的排序对于简单的 ...

  4. 2017 济南综合班 Day 7

     a 两个指针L.R R开始指向恰好[R,n]有不超过k个逆序对的地方 随着L的右移,R指针只会右移 逆序对有2部分 1.L左侧与其他位置形成的逆序对 2.R右侧与其他位置形成的逆序对 用树状数组分别 ...

  5. 快速搭建rabbitmq单节点并配置使用

    安装erlang环境 wget http://erlang.org/download/otp_src_20.3.tar.gz tar xf otp_src_20.3.tar.gz && ...

  6. Divergent Change(发散式变化)---要重构的信号

    “ 当你看着一个类说,呃,如果新加入一个数据库,我必须修改这三个函数:如果新出现一种金融工具,我必须修改这四个函数.那么,此时也许将这个类分成两个会更好,这么一来每个对象就可以只因一种变化而需要修改. ...

  7. bzoj 2276: [Poi2011]Temperature——单调队列

    Description 某国进行了连续n天的温度测量,测量存在误差,测量结果是第i天温度在[l_i,r_i]范围内. 求最长的连续的一段,满足该段内可能温度不降 第一行n 下面n行,每行l_i,r_i ...

  8. quartz的简介

    1. 介绍  Quartz是OpenSymphony开源组织在Job scheduling领域又一个开源的任务调度框架,是完全由java开发的一个开源的任务日程管理系统,“任务进度管理器”就是一个在预 ...

  9. Caffe学习笔记2

    Caffe学习笔记2-用一个预训练模型提取特征 本文为原创作品,未经本人同意,禁止转载,禁止用于商业用途!本人对博客使用拥有最终解释权 欢迎关注我的博客:http://blog.csdn.net/hi ...

  10. vue路由-动态路由和嵌套路由

    一.动态路由 我们经常需要把某种模式匹配到的所有路由,全都映射到同个组件.例如,我们有一个 User 组件,对于所有 ID 各不相同的用户,都要使用这个组件来渲染.那么,我们可以在 vue-route ...