最近做一个小项目,项目中有一个定时服务,需要向对方定时发送数据,时间间隔是1.5s,然后就想到了用C#的Timer类,我们知道Timer

确实非常好用,因为里面有非常人性化的start和stop功能,在Timer里面还有一个Interval,就是用来设置时间间隔,然后时间间隔到了就会触

发Elapsed事件,我们只需要把callback函数注册到这个事件就可以了,如果Interval到了就会触发Elapsed,貌似一切看起来很顺其自然,但是

有一点一定要注意,callback函数本身执行也是需要时间的,也许这个时间是1s,2s或者更长时间,而timer类却不管这些,它只顾1.5s触发一下

Elapsed,这就导致了我的callback可能还没有执行完,下一个callback又开始执行了,也就导致了没有达到我预期的1.5s的效果,并且还出现了

一个非常严重的问题,那就是线程激增,非常恐怖。

下面举个例子,为了简化一下,我就定义一个task任务,当然项目中是多个task任务一起跑的。

一:问题产生

为了具有更高的灵活性,我定义了一个CustomTimer类继承自Timer,然后里面可以放些Task要跑的数据,这里就定义一个Queue。

 namespace Sample
{
class Program
{
static void Main(string[] args)
{
TimerCustom timer = new TimerCustom(); timer.Interval = ; timer.Elapsed += (obj, evt) =>
{
TimerCustom singleTimer = obj as TimerCustom; if (singleTimer != null)
{
if (singleTimer.queue.Count != )
{
var item = singleTimer.queue.Dequeue(); Send(item);
}
}
}; timer.Start(); Console.Read();
} static void Send(int obj)
{
//随机暂定8-10s
Thread.Sleep(new Random().Next(, )); Console.WriteLine("当前时间:{0},定时数据发送成功!", DateTime.Now);
}
} class TimerCustom : System.Timers.Timer
{
public Queue<int> queue = new Queue<int>(); public TimerCustom()
{
for (int i = ; i < short.MaxValue; i++)
{
queue.Enqueue(i);
}
}
}
}

二:解决方法

1.  从上图看,在一个任务的情况下就已经有14个线程了,并且在21s的时候有两个线程同时执行了,我的第一反应就是想怎么把后续执行callback的

线程踢出去,也就是保证当前仅让两个线程在用callback,一个在执行,一个在等待执行,如果第一个线程的callback没有执行完,后续如果来了第三

个线程的话,我就把这第三个线程直接踢出去,直到第一个callback执行完后,才允许第三个线程进来并等待执行callback,然后曾今的第二个线程开

始执行callback,后续的就以此类推。。。

然后我就想到了用lock机制,在customTimer中增加lockMe,lockNum,isFirst字段,用lockMe来锁住,用lockNum来踢当前多余的要执行callback

的线程,用isFirst来判断是不是第一次执行该callback,后续callback的线程必须先等待1.5s再执行。

 namespace Sample
{
class Program
{
static void Main(string[] args)
{
TimerCustom timer = new TimerCustom(); timer.Interval = ; timer.Elapsed += (obj, evt) =>
{
TimerCustom singleTimer = obj as TimerCustom; if (singleTimer != null)
{
//如果当前等待线程>2,就踢掉该线程
if (Interlocked.Read(ref singleTimer.lockNum) > )
return; Interlocked.Increment(ref singleTimer.lockNum); //这里的lock只能存在一个线程等待
lock (singleTimer.lockMe)
{
if (!singleTimer.isFirst)
{
Thread.Sleep((int)singleTimer.Interval);
} singleTimer.isFirst = false; if (singleTimer.queue.Count != )
{
var item = singleTimer.queue.Dequeue(); Send(item); Interlocked.Decrement(ref singleTimer.lockNum);
}
}
}
}; timer.Start(); Console.Read();
} static void Send(int obj)
{
Thread.Sleep(new Random().Next(, )); Console.WriteLine("当前时间:{0},邮件发送成功!", DateTime.Now);
}
} class TimerCustom : System.Timers.Timer
{
public Queue<int> queue = new Queue<int>(); public object lockMe = new object(); public bool isFirst = true; /// <summary>
/// 为保持连贯性,默认锁住两个
/// </summary>
public long lockNum = ; public TimerCustom()
{
for (int i = ; i < short.MaxValue; i++)
{
queue.Enqueue(i);
}
}
}
}

从图中可以看到,已经没有同一秒出现重复任务的发送情况了,并且线程也给压制下去了,乍一看效果不是很明显,不过这是在一个任务的情况

下的场景,任务越多就越明显了,所以这个就达到我要的效果。

2. 从上面的解决方案来看,其实我们的思维已经被问题约束住了,当时我也是这样,毕竟坑出来了,就必须来填坑,既然在callback中出现线程

  蜂拥的情况,我当然要想办法管制了,其实这也没什么错,等问题解决了再回头考虑下时,我们会发现文章开头说的Timer类有强大的Stop和

Start功能,所以。。。。这个时候思维就跳出来了,何不在callback执行的时候把Timer关掉,执行完callback后再把Timer开启,这样不就

可以解决问题吗?好吧,说干就干。

 namespace Sample
{
class Program
{
static void Main(string[] args)
{
TimerCustom timer = new TimerCustom(); timer.Interval = ; timer.Elapsed += (obj, evt) =>
{
TimerCustom singleTimer = obj as TimerCustom; //先停掉
singleTimer.Stop(); if (singleTimer != null)
{
if (singleTimer.queue.Count != )
{
var item = singleTimer.queue.Dequeue(); Send(item); //发送完成之后再开启
singleTimer.Start();
}
}
}; timer.Start(); Console.Read();
} static void Send(int obj)
{
Thread.Sleep(new Random().Next(, )); Console.WriteLine("当前时间:{0},邮件发送成功!", DateTime.Now);
}
} class TimerCustom : System.Timers.Timer
{
public Queue<int> queue = new Queue<int>(); public object lockMe = new object(); /// <summary>
/// 为保持连贯性,默认锁住两个
/// </summary>
public long lockNum = ; public TimerCustom()
{
for (int i = ; i < short.MaxValue; i++)
{
queue.Enqueue(i);
}
}
}
}

从图中可以看到,问题同样得到解决,而且更简单,精妙。

最后总结一下:解决问题的思维很重要,但是如果跳出思维站到更高的抽象层次上考虑问题貌似也很难得。。。

最近用Timer踩了一个坑,分享一下避免别人继续踩的更多相关文章

  1. 【转】最近用Timer踩了一个坑,分享一下避免别人继续踩

    [转]最近用Timer踩了一个坑,分享一下避免别人继续踩 最近做一个小项目,项目中有一个定时服务,需要向对方定时发送数据,时间间隔是1.5s,然后就想到了用C#的Timer类,我们知道Timer 确实 ...

  2. 使用ffmpeg视频编码过程中踩的一个坑

           今天说说使用ffmpeg在写视频编码程序中踩的一个坑,这个坑让我花了好多时间,回头想想,非常多时候一旦思维定势真的挺难突破的.以下是不对的编码结果:                   ...

  3. 记新人从 excel 文件中读取字典数据踩的一个坑

    原本是打算今天分享一下最近学习接口自动化的心得体会,然而在我写模板的时候,却被一个坑拦我大半天,心得体会不得不 延期再分享了.在我无数次调试无数次看log,终于发现并解决这个问题了.下面记录一下踩的坑 ...

  4. toFixed()一不小心踩了一个坑

    toFixed,多么简单的一个函数,昨天突发奇想做两道算法题练练手.结果,踩到了一个从未遇到的坑! \n 简单来讲是要对输入的很多组数据,自己写一个函数做个处理,把每次函数处理的结果要相加求和.最后输 ...

  5. Javascript之旅——第八站:说说instanceof踩了一个坑

    前些天写js遇到了一个instanceof的坑,我们的页面中有一个iframe,我在index页面中计算得到了一个array,然后需要传递到Flight页面 这个嵌套的iframe中的一个函数(Sea ...

  6. java 反射的踩的一个坑

    今天工作的时候用到了一个反射.其业务简单描述为:系统启动时将需要定时调用的方法签名保存到数据库中,开启线程定时从数据库中读取对应的方法签名,通过反射生成实例后调用方法.完成一定的定时任务. 写到的方法 ...

  7. mongodb 的Cursor 作为 stream 的时候,读出来的数据数字开头的key没法访问(又踩了一个坑)

    mongdb 用Cursor 读取数据的时候,直接用流读出来的数据key是数字开头的话,就是独具不到,用Object.keys() 把所有的key 打印出来的话如下:怎么会是这样的呢? 查看了一下文档 ...

  8. 看了下opengl相关的资料,踩了一个坑,记录一下

    2019/03/10 下午看了下关于opengl的资料,然后把敲了下代码,然后程序报错了.代码如下: #include <glad/glad.h> #include <GLFW/gl ...

  9. leaflet control.layers踩的一个坑

    Control.Layers方法 该方法可以创建一个切换图层的工具, L.control.layers(baseLayers, overlayers).addTo(map); baseLayers参数 ...

随机推荐

  1. java的spilt(“,”)方法bug处理

    java split方法以逗号分隔如字符串",,,,,," 这样会得到一个空的数组 String str ={1,2,3,,,,, } String[] str1 =spilt(& ...

  2. 软件项目发展历史<人月神话>这本书好

    几乎是计算机软件开发的发展历史     人月神话,增加人手并不一定能提高开发速度. 原因在于,有些任务是无法分解的,存在先后顺序.无法同步进行. 增加人手,增加的是沟通成本,相互牵制.可以分解的任务就 ...

  3. loadrunner录制webservice接口需要用户名密码时的解决方法

    当loadrunner进行webservice测试,进入访问地址,限制要输入用户名密码才能登陆时,添加以下代码: web_set_user("acerpc/admin"," ...

  4. http异步请求的一种调用示例

    在异步编程中,经常会调用已经写好的异步方法.这时会有一个需求:根据异步方法的返回值,做一些别的操作. 1.0 重新开启一个异步方法,在这个新的异步方法内部,调用需要请求的异步方法.示例: static ...

  5. 设计模式之 面向对象的养猪厂的故事,C#演示(一)

    对于设计模式, 从本质上说, 其最大的用途就是适应需求的变化. 因为有了设计模式,我们可以在设计阶段就为未来可能发生的变化留下足够的空间. 我们通过一个建造现代化养猪场的故事, 来讨论一下设计模式与需 ...

  6. java内存模型-基础

    基础 并发编程模型的分类 在并发编程中,我们需要处理两个关键问题:线程之间如何通信及线程之间如何同步(这里的线程是指并发执行的活动实体).通信是指线程之间以何种机制来交换信息.在命令式编程中,线程之间 ...

  7. canvas粒子效果-snow

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  8. 我所知道的Javascript

    javascript到了今天,已经不再是我10多年前所认识的小脚本了.最近我也开始用javascript编写复杂的应用,所以觉得有必要将自己的javascript知识梳理一下.同大家一起分享javas ...

  9. 12款支持移动设备的响应式 WordPress 主题

    响应式和现代设计风格的多用途 WordPress 主题与能够非常灵活的适应所有设备.而高级主题能够更大可能性的轻松定制.所有的主题是完全响应式的,您可以从主题选项中启用响应模式. 今天,这个列表收集了 ...

  10. 定时器相关 setTimeout setInterval 函数节流

    这个问题也是在参加百度的前端技术学院中遇到的 任务中需要用js实现动画  导师给的评价中setInterval会导致bug 当时不理解   下面把自己学习的过程分享出来 再次理解单线程   老是说js ...