简述:
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)