自己开发能在asp.net项目正常使用的定时器WebTimer,让定时器听话起来

简述:

iis是一个很不错的服务器,有很多很好用的特性来支持网站运行,但有时候这些特性却会影响到我们开发者的一些操作。比如我们需要定时运行做一些操作,但由于iis的利用应用程序池来管理这种方式会让网站所在的进程在空闲(一段时间没人访问)时注销该线程,所以定时器是无法正常运行的。但利用asp.net的Cache机制是可以巧妙的实现能正常使用的定时器的。

运行效果:

请打开该链接

该网站定时每一个小时给该网页添加一条记录,而且估计我这个站点几百年都不会有一个人来访问一次的,大部分时间都是处于空闲状态的。

怎么使用:

网站添加WebTimerLib类库的引用

在合适的位置编写以下一行代码

 WebTimerLib.WebTimer timer = new WebTimerLib.WebTimer(new System.Threading.TimerCallback(TimeCallback), null, TimeSpan.FromSeconds(2), TimeSpan.FromHours(1));

 或者以下一行代码,推荐这行

WebTimerLib.WebTimer timer = new WebTimerLib.WebTimer(new System.Threading.TimerCallback(TimeCallback), null, TimeSpan.FromSeconds(2), TimeSpan.FromSeconds(60), "20130728");

这行代码表示2秒后没一个小时运行一次TimeCallback这个方法

具体每个参数会在源码里有注释

源码:

源码是在.net 3.5的环境下开发的,估计.net2.0也是可以编译通过的。

类库下载地址

WebTimer类的代码: 

  1 using System;
  2 using System.Collections.Generic;
  3 using System.Text;
  4 using System.Threading;
  5 using System.Web.Caching;
  6 using System.Web;
  7 
  8 namespace WebTimerLib
  9 {
 10     /// <summary>
 11     /// 可以正常在web使用的定时器
 12     /// </summary>
 13     public class WebTimer : IDisposable
 14     {
 15         /// <summary>
 16         /// 初始化 Timer 类的新实例,使用 TimeSpan 值来度量时间间隔。
 17         /// </summary>
 18         /// <param name="callback">一个 TimerCallback 委托,表示要执行的方法。 </param>
 19         /// <param name="state">一个包含回调方法要使用的信息的对象,或者为 空引用(在 Visual Basic 中为 Nothing)。 </param>
 20         /// <param name="dueTime">TimeSpan,表示在 callback 参数调用它的方法之前延迟的时间量。指定 -1 毫秒以防止启动计时器。指定零 (0) 以立即启动计时器。 </param>
 21         /// <param name="period">在调用 callback 所引用的方法之间的时间间隔。指定 -1 毫秒可以禁用定期终止。 </param>
 22         /// <param name="timerID">定时器标识符,不能重复</param>
 23         /// <remarks>
 24         /// 如果 dueTime 为零 (0),则立即调用 callback。如果 dueTime 是 -1 毫秒,则不会调用 callback;计时器将被禁用,但通过调用 Change 方法可以重新启用计时器。
 25         /// 如果 period 为零 (0) 或 -1 毫秒,而且 dueTime 为正,则只会调用一次 callback;计时器的定期行为将被禁用,但通过使用 Change 方法可以重新启用该行为。
 26         /// 为 callback 指定的方法应是可重入的,这是因为 ThreadPool 线程会调用该方法。该方法在以下两种情况下可以同时在两个线程池线程中执行:一是计时器间隔小于执行该方法所需的时间;二是所有线程池线程都在使用,并且多次对该方法进行排队。
 27         /// </remarks>
 28         public WebTimer(TimerCallback callback, Object state, TimeSpan dueTime, TimeSpan period, string timerID)
 29         {
 30             _callback = callback;
 31             _state = state;
 32             _dueTime = dueTime;
 33             _period = period;
 34             _cacheID = timerID;
 35             Run(dueTime);
 36         }
 37 
 38         /// <summary>
 39         /// 初始化 Timer 类的新实例,使用 TimeSpan 值来度量时间间隔, 建议使用WebTimer(TimerCallback callback, Object state, TimeSpan dueTime, TimeSpan period, string timerID),避免造成更改定时器操作的时候,旧操作仍旧在运行
 40         /// </summary>
 41         /// <param name="callback">一个 TimerCallback 委托,表示要执行的方法。 </param>
 42         /// <param name="state">一个包含回调方法要使用的信息的对象,或者为 空引用(在 Visual Basic 中为 Nothing)。 </param>
 43         /// <param name="dueTime">TimeSpan,表示在 callback 参数调用它的方法之前延迟的时间量。指定 -1 毫秒以防止启动计时器。指定零 (0) 以立即启动计时器。 </param>
 44         /// <param name="period">在调用 callback 所引用的方法之间的时间间隔。指定 -1 毫秒可以禁用定期终止。 </param>
 45         /// <param name="timerID">定时器标识符,重复的话会取消之前的任务</param>
 46         /// <remarks>
 47         /// 如果 dueTime 为零 (0),则立即调用 callback。如果 dueTime 是 -1 毫秒,则不会调用 callback;计时器将被禁用,但通过调用 Change 方法可以重新启用计时器。
 48         /// 如果 period 为零 (0) 或 -1 毫秒,而且 dueTime 为正,则只会调用一次 callback;计时器的定期行为将被禁用,但通过使用 Change 方法可以重新启用该行为。
 49         /// 为 callback 指定的方法应是可重入的,这是因为 ThreadPool 线程会调用该方法。该方法在以下两种情况下可以同时在两个线程池线程中执行:一是计时器间隔小于执行该方法所需的时间;二是所有线程池线程都在使用,并且多次对该方法进行排队。
 50         /// </remarks>
 51         public WebTimer(TimerCallback callback, Object state, TimeSpan dueTime, TimeSpan period)
 52         {
 53             _callback = callback;
 54             _state = state;
 55             _dueTime = dueTime;
 56             _period = period;
 57             Run(dueTime);
 58         }
 59 
 60         private TimerCallback _callback;
 61         private object _state;
 62         private TimeSpan _dueTime;
 63         private TimeSpan _period;
 64         private bool _isDisposing = false;
 65         private DateTime _NewDoCallbackTime;
 66 
 67         protected void Run(TimeSpan dueTime)
 68         {
 69             _NewDoCallbackTime = DateTime.Now.Add(dueTime);
 70             HttpRuntime.Cache.Insert(CacheID, this, null, _NewDoCallbackTime, Cache.NoSlidingExpiration, CacheItemPriority.NotRemovable, new CacheItemRemovedCallback(CacheItemRemovedCallback));
 71 
 72         }
 73 
 74         private void CacheItemRemovedCallback(String key, Object value, CacheItemRemovedReason reason)
 75         {
 76             WebTimer timer = (WebTimer)value;
 77             if (timer._isDisposing)
 78                 return;
 79             if (reason == CacheItemRemovedReason.Expired)
 80             {
 81                 if (timer._period < TimeSpan.FromSeconds(0))
 82                     _isDisposing = true;
 83                 timer._callback(_state);
 84                 timer.Run(timer._period);
 85             }
 86             else
 87             {
 88                 timer.Run(_NewDoCallbackTime - DateTime.Now);
 89             }
 90         }
 91 
 92         /// <summary>
 93         /// 更改计时器的启动时间和方法调用之间的时间间隔,使用 TimeSpan 值度量时间间隔。 
 94         /// </summary>
 95         /// <param name="dueTime">一个 TimeSpan,表示在调用构造 Timer 时指定的回调方法之前的延迟时间量。指定负 -1 毫秒以防止计时器重新启动。指定零 (0) 以立即重新启动计时器。</param>
 96         /// <param name="period">在构造 Timer 时指定的回调方法调用之间的时间间隔。指定 -1 毫秒可以禁用定期终止。 </param>
 97         /// <returns></returns>
 98         public bool Change(TimeSpan dueTime, TimeSpan period)
 99         {
100             this._dueTime = dueTime;
101             this._period = period;
102             try
103             {
104                 HttpRuntime.Cache.Remove(CacheID);
105                 this.Run(this._dueTime);
106                 return true;
107             }
108             catch (Exception)
109             {
110                 return false;
111             }
112         }
113 
114         private string _cacheID;
115         protected string CacheID
116         {
117             get
118             {
119                 if (_cacheID == null) _cacheID = GetHashCode().ToString();
120                 return _cacheID;
121             }
122         }
123 
124         #region IDisposable 成员
125 
126         public void Dispose()
127         {
128             _isDisposing = true;
129         }
130 
131 
132 
133         #endregion
134 
135     }
136 }

补充:

对于我的书写能力我表示歉意,小时候语文没学好,文笔不好请见谅;能力水平有限,有什么说错的地方请多多指教;

关于转载,我不喜欢别人转载我的文章,觉得有价值的话就收藏到笔记之类软件,不会给搜索引擎收录的地方,我不太关注我的文章的版权问题,但我实在不喜欢自己在网上搜索资料的时候,搜到满满的一页是自己曾写过的一篇文章,实在恶心到我自己,我就注意到了一个情况,很多转载都不是为了学习的,而是一些公司不知道基于什么想法就转了过去,这样造成了搜索自己想要的资源会越来越难。但我发现如果是搜英文的资料的时候就很少有重复的,o(︶︿︶)o 唉,实在纠结啊!

补充(2013-7-28 7:08):

喉咙发炎,所以喝水太多,于是半夜起床,看了下定时器的运行情况,发现了奇怪的情况,如下图

有问题,怎么能安心入眠呢?于是一个经典的场面出现了,一个酷毙(苦逼)的程序猿在夜深人静的时候努力的敲着代码。

问题在那里呢?认真看运行结果,是不是发现好像有两个定时器在运行呢?没错,就是有两个定时器在运行了。

原来是我的运行结果用了Cache来保存,运行的站点是个免费的国外空间,估计我们半夜的时候是人家那里烈阳当头的时候,所以出现了内存紧张,我存着结果的Cache给回收了。Cache给回收之后,下次有人访问的时候就会重新设置一个定时器。

于是发现了一个题外话,有个哥们在1:51分的时候来关顾过我的网站,哦,这个网站运行环境的时候估计和我们有12个小时的时差。

问题描述出来了,那接下来就是解决

小弟不用Cache了,直接存在一个文本文件里。

于是问题解决了,并且我改了定时器的运行周期,半小时一次,然后我们继续等待测试结果吧。

最后,我对代码做了点修改,修改的范围在53到58行之间,之前有使用的孩纸,就重新复制粘贴下吧,没有的话就不用管这句话了

更新说明(2013-07-28 9:33)

重载一个构造函数

WebTimer(TimerCallback callback, Object state, TimeSpan dueTime, TimeSpan period, string timerID)

原文地址:https://www.cnblogs.com/sheepswallow/p/3219612.html