心跳监测器 IHeartBeatChecker -- ESBasic 可复用的.NET类库(09)

1.缘起:

    假设我们的C/S系统中服务端与客户端之间采用UDP进行通信,那么服务端如何知道每个客户端当前是否仍然在线了?有可能某个客户端一直没有退出,但是在很长一段时间内都没有与服务端作任何通信,那么服务端就应该认为这个客户端已经离线了吗?为了能让服务端掌握每个客户端是否在线的状态,我们可以这样做,只要客户端一启动起来,就每隔一段时间间隔(如10秒)就向服务端发一个“我还在线”的消息,以表明自己的状态。而服务端如果在一个更大的时间间隔内(如20秒)都没有收到某个客户端的任何消息,则可以判定这个客户端已经离线了。

    这就是我们常用的“心跳”机制,客户端每隔一段时间间隔发的那个消息就称为“心跳消息”,只要心跳还在,就表示自己还是Alive的,否则就是断线/下线了。

       心跳监测器的形象示意图如下: 

    

2.适用场合:

       在很多基于非连接的通信系统中,心跳机制是经常使用的方案。其最主要的目的是让服务端能比较及时的掌握到每个客户端当前的在线状态,因为这个信息是相当重要的,而且这个状态改变时服务端知道得越及时越好。ESBasic.Threading.Application.IHeartBeatChecker(心跳检测器)便是用于对每个客户端的心跳进行监控以掌握每个客户端的在线状态的。它经常使用在类似下面的这些场合:

(1)服务端无法感知客户端的离线或意外掉线(如网络中断、系统重启等),而这个信息对于服务端而言却是非常重要的。

(2)服务端可以感知客户端的离线或意外掉线,但是这种感知有延迟,而且延迟可能非常大(比如几分钟),其程度已经超过了服务端能接受的范围。比如,基于TCP的C/S系统,客户端之间与服务端之间有防火墙等相关设备的存在,客户端掉线时,服务端与防火墙之间对应的连接仍然存在,所以服务端认为客户端仍然在线,这种状况可能要持续几秒钟到几分钟不等。

(3)以上所说的服务端/客户端可以认为是一个广义的定义,只要是通信的双方(如P2P)需要知道对方的在线状态,那么都可以使用心跳机制来解决。

 

3.设计思想与实现

      IHeartBeatChecker接口定义如下:       

    public interface IHeartBeatChecker 
    {
        /// <summary>
        /// SurviveSpanInSecs 在没有心跳到来时,可以存活的最长时间。SurviveSpanInSecs小于等于0,表示存活时间为无限长,而不需要进行心跳检查
        /// </summary>
        int SurviveSpanInSecs { get; set; }

        /// <summary>
        /// DetectSpanInSecs 隔多长时间进行一次状态检查。
        /// </summary>
        int DetectSpanInSecs { get;set; }

        /// <summary>
        /// Initialize 初始化并启动心跳监测器。
        /// </summary>
        void Initialize();

        /// <summary>
        /// RegisterOrActivate 注册一个新的客户端或激活它(收到心跳消息)。
        /// </summary>       
        void RegisterOrActivate(string id);

        /// <summary>
        /// Unregister 服务端主动取消对目标客户端的监测。
        /// </summary>        
        void Unregister(string id);

        /// <summary>
        /// Clear 清空所有的监测。
        /// </summary>
        void Clear();

        /// <summary>
        /// SomeOneTimeOuted  当在规定的时间内没有任何消息过来,那么将会触发该事件。
        /// 注意:该事件的处理函数严禁抛出任何异常。
        /// </summary>
        event CbSimpleStr SomeOneTimeOuted;
    }

 

    

根据上述对心跳监测器的介绍,我们知道需要定时检查每个客户端的状态,看在规定的时间间隔内是否有“心跳”消息过来。我们可以借助循环引擎(ICycleEngine)来进行定时检查。从IHeartBeatChecker接口定义,你有看到它并没有从ICycleEngine继承,那表明心跳监测器不需要被反复的Start、Stop。相反的,IHeartBeatChecker提供了一个Initialize方法,用于初始化和启动监测器。监测器一旦启动就会在随系统的生命周期运行,这和我们的绝大部分需求是完全一致的。

       DetectSpanInSecs属性表示需要间隔多少秒检测一次客户端的状态,这个属性的值将被直接传递给循环引擎的同名属性。

       SurviveSpanInSecs属性表示在没有心跳到来时,客户端可以存活的最长时间。这个时间通常要比客户端定时发送“心跳”消息的时间间隔大一些。

     当心跳监测器发现某个客户端在规定的时间内没有心跳消息过来,那么将会触发SomeOneTimeOuted事件以通知服务端目标客户端掉线了。

  HeartBeatChecker实现了IHeartBeatChecker接口,其实现要注意以下几点:

(1)HeartBeatChecker继承自BaseCycleEngine,它借助于循环引擎来进行任务状态的循环检测。

(2)为了允许在多线程的环境中回调定时器,HeartBeatChecker必须对内部集合(dicIDTime)进行加锁控制。

(3)为了在初始化的时候启动监测器,其在Initialize方法中调用了循环引擎的Start方法。

      

4. 使用时的注意事项

(1)如果服务端已经确切知道客户端已经离线(比如,客户端向服务端发送“我要退出了”的消息),那么服务端可以调用IHeartBeatChecker. Unregister方法来主动清除对目标客户端的监测。

(2)SomeOneTimeOuted事件的处理函数不要抛出任何异常,否则会导致后续的客户端掉线事件无法被触发。这点从我们的实现源码就可以看到,一旦一个SomeOneTimeOuted抛出异常,foreach将会被迫中断。而且,更严重的是,会导致循环引擎的停止运行――监测器会停止运行。

(3)不一定只有心跳消息到来时,才调用RegisterOrActivate方法来激活对应的客户端。实际上,我们只要收到来自客户端的任何消息时,都可以调用RegisterOrActivate方法来激活它。

(4)如何设置SurviveSpanInSecs属性和DetectSpanInSecs属性的值,取决于我们系统的需求。服务端要求感受客户端掉线越及时,那么DetectSpanInSecs就要设得越小,而且客户端发送心跳的时间间隔也要越小,SurviveSpanInSecs也要相应的小。SurviveSpanInSecs的设定取决于客户端发送心跳的时间间隔和可以允许的最大网络延时。可以采用如下公式:SurviveSpanInSecs = 客户端发送心跳时间间隔 + 允许的最大网络延时
 

5.扩展

       心跳监测器IHeartBeatChecker暂时没有任何扩展。

注: ESBasic已经开源,点击这里下载源码。
    ESBasic开源前言

 

 

时间: 2024-10-16 03:22:49

心跳监测器 IHeartBeatChecker -- ESBasic 可复用的.NET类库(09)的相关文章

ESBasic 可复用的.NET类库(00) -- 开源前言(附下载)

自从03年正式使用.NET开发以来,已经走过了6个年头,这期间我积累了几套类库和框架,ESBasic便是其中最基础的一个类库.ESBasic是Enterprise Service Basic的缩写,虽然也简写为ESB,但是它和Enterprise Service Bus(企业服务总线)没有任何关系.ESBasic是我能够快速和高效开发应用程序的利器之一,开这个专门的blog是想将它介绍给大家,希望能对大家有所启发. ESBasic覆盖的内容包括:对象管理.插件.网络(Socket).多线程.Em

优先级管理器 IPriorityManager -- ESBasic 可复用的.NET类库(14)

1.缘起:     假设我们的订单处理系统所要处理的订单是有优先级的,也就是说,不同的订单类型所要求被处理的紧迫程度不同,对那些优先级高的注单要先处理,对于优先级低的注单可稍后处理.对于处于同一优先级的订单了,就按照其到达的先后顺序进行处理.     这是一个典型的管理具有优先级的对象的需求,注单就是具有优先级(With Priority)的对象.我设计了ESBasic.ObjectManagement.Managers.IPriorityManager优先级管理器(确切地说,应该称之为"具有优

Round缓存管理器RoundCacheManager--ESBasic 可复用的.NET类库(26)

1.缘起:     在增量自动获取器章节的缘起部分,我们曾提到增量缓存,本节我们将深入探讨它以及用于管理增量缓存的管理器.我们还是以增量自动获取器章节提到的例子作为基础,并做更进一步的讨论.       OK,现在让我们开始这有趣的旅程. 首先,基于前面例子给出的上下文,我们知道IIncreaseAutoRetriever获取的增量是用于累积当天的已成交订单报表的."当天已成交报表"就是一个典型的增量缓存,每当有新的增量到来,都会累加到上面. 我们假设今天是2009.07.08,那么我

循环任务切换器 CircleTaskSwitcher -- ESBasic 可复用的.NET类库(06)

 1.缘起:     假设我的订单处理系统有这样的需求:将一天24小时分为4个时段,凌晨2:15到8:30采用A类型的处理器处理接收到的订单,8:30到14:00采用B类型的处理器,14:00到20:00采用C类型的处理器,20:00到第二天凌晨2:15采用D类型的处理器.     即我们的订单处理器需要在任一天的2:15.8:30.14:00.20:00这四个时刻发生切换,这就是一个循环切换器所要做的工作.     我设计了ESBasic.Threading.Application. ICir

对象获取器IObjectRetriever -- ESBasic 可复用的.NET类库(17)

1.缘起: ESBasic中许多管理对象的容器都用到了这个ESBasic.ObjectManagement.IObjectRetriever接口,所以单独将其提出来介绍一下. 当我们向对象容器(Container)请求某个对象时,也许目标对象还未加载到容器中,这可能是因为容器在初始化的时候就没有加载这个对象,也有可能是因为这个对象是容器初始化以后新增到数据库(当然也有可能是其它的持久化存储)的.在这种情况下,对象容器就可以借助IObjectRetriever来将目标对象从数据库等持久化存储中加载

工作者引擎 IWorkerEngine -- ESBasic 可复用的.NET类库(05)

1.缘起:     假设我们的系统在运行的过程中,源源不断的有新的任务需要处理(比如订单处理),而且这些任务的处理是相互独立的,没有前后顺序依赖性(顺序依赖性是指,必须在任务A处理结束后才可开始B任务),那么我们就可以使用多个线程来同时处理多个任务.每个处理任务的线程称为"工作者(线程)".      我设计了ESBasic.Threading.Engines.IWorkerEngine工作者引擎,其目的就是使用多个线程来并行处理任务,提高系统的吞吐能力.       工作者引擎的形象

回调定时器ICallbackTimer --ESBasic 可复用的.NET类库(07)

 1.缘起:     举个例子也许就能够说清楚回调定时器的用途.假设我的订单系统接收各种不同类型的订单,当订单A进来时,系统根据订单的类型和其它特征进行综合判断后,决定A订单要在2秒之后被方法M1处理:接下来收到的B订单经过同样的判断后,决定要在10秒后被方法M2处理,--.这时候就可以用回调定时器来管理这些将要被延迟一定时间再执行的任务.     当然,我们可以使用定时器或前面介绍的循环引擎来实现这样的功能,只不过我们自己需要手动管理注册的定时回调任务,并且定时检查每一个未处理订单是否已经到了

双向映射 IBidirectionalMapping -- ESBasic 可复用的.NET类库(11)

1.缘起:     假设我们的用户管理系统要求用户的ID和Name都必须是唯一的,并且用户的ID和Name一经确定就不能被修改.而且管理系统经常需要根据ID来查找Name,也经常需要根据Name来查找ID.根据这样的需求,我们可以考虑使用一个Dictionary来将ID和Name缓存起来,通常ID作为Key,Name作为Value.这样便可实现通过ID查询Name的快速查找,但是,通过Name查找ID就不是那么快了,因为涉及到对Dictionary的Values做遍历的操作.那么,有可能使得通过

对象管理器 IObjectManager -- ESBasic 可复用的.NET类库(12)

1.缘起: 我们经常需要对一些动态对象进行管理,最常见的例子就是在线用户管理.当一个用户成功登陆到服务器后,我们就需要将其管理起来:当他退出后,就不再需要再管理他了.这就是所谓动态对象的含义,这些对象并不是一直需要被管理,只有当其被激活后,才需要被管理.它们总是在"激活"状态和"非激活"状态之间不断地切换. 我设计了对象管理器ESBasic.ObjectManagement.Managers.IObjectManager来管理类似的动态对象.这个类是ESBasic提