GS LiveMgr心跳管理类

struct LiveMgr
{
private:
    int                            m_nCount;            ///< 管理数量
    std::vector<int>            m_vecChannels;        ///< 所有channel
    std::shared_ptr<I_Timer>    m_spTimer;            ///< 定时器
 public:
     std::function<void(int channel)> m_fnTimeOutDisconnect;
    void Init(int nMaxGcNumb);
    bool IsLive(int nChannelId)
    {
        return !!(m_vecChannels[nChannelId]);
    }
    void OnLive(int nChannelId)
    {
        m_vecChannels[nChannelId] = GetTickCount();
    }
    void Add(int nChannelId)
    {
        m_nCount++;
         m_vecChannels[nChannelId] = GetTickCount();
    }
    void Remove(int nChannelId)
    {
        m_nCount--;
        m_vecChannels[nChannelId] = 0;
    }
    void Check();
    int GetLinks()
    {
        return m_nCount;
    }

};
这个是心跳管理类
GS初始化的时候
m_LiveMgr.m_fnTimeOutDisconnect = std::bind(&GameServer::TimeOutDisconnect, this, ph::_1);
m_LiveMgr.Init(GetMaxGcNumb());

void LiveMgr::Init(int max_gc_numb)
{
    ///创建内部定时检查的定时器,5s执行一次,不过为了现在方便把定时器关闭了
    m_spTimer.reset(GetPlug(TimerFactory)->createTimer());
    m_spTimer->regTimer(std::bind(&LiveMgr::Check, this));
    m_spTimer->setInterval(5 * 1000);
    //为调试先关掉
    //m_timer->start();

    m_vecChannels.resize(max_gc_numb);//m_vecChannels保存的所有channel最新的包跟新时间,注意这个默认初始化为0
    m_nCount = 0;
}
那这个是什么时候更新的呢
if(rPkt.is_data)
{
    if(!rPkt.data)
        return false;

    GameChannel* pGC = m_vecChannel[rPkt.channel_id];
    if(pGC)
        pGC->OnReceiveData(rPkt.data, rPkt.size);

    m_LiveMgr.OnLive(rPkt.channel_id);//看来这个是每接收一个包的时候将channel的最新时间更新一下
}
void OnLive(int nChannelId)
{
    m_vecChannels[nChannelId] = GetTickCount();
}
看看这个5s定时回调的函数做了什么
void LiveMgr::Check()
{
    size_t uNow = GetTickCount();
    for (size_t i = 0; i < m_vecChannels.size(); ++i)//遍历所有channel,这个效率???
    {
        size_t uLastTick = m_vecChannels[i];
        //uLastTick为0表示没有连接,这句就说明连接了并且15没有接收到数据了,就认为其断线了,会走下线处理的
        //想当时测试的时候把网线拔了,此时libevent是不能收到事件的,GS他的连接还是在上面的,如果把定时器打开,15秒后会将其删除的
        if(uLastTick && uNow - uLastTick > 15 * 1000)
        {
            m_fnTimeOutDisconnect(i);
            Remove(i);//需要将m_vecChannels对应位置赋值为0,表示没有连接,刚才我还在想这边把socket关闭了,不会受到断开通知,这里处理了就对了
        }
    }
}
//看看这个function做了什么
做相应的下线处理,并关闭网络层的socket
void GameServer::TimeOutDisconnect(int nChannelId)
{
    try
    {
        GameChannel* pGC = m_vecChannel[nChannelId];
        pGC->OnDisconnect();//玩家下线处理,回头看玩家下线在详看
        if(!pGC->m_pMap)
        {
            //PushFreeQueue(pGC);
            //m_vecChannel[nChannelId] = NULL;
            AutoFreeGC(pGC);
        }
        m_spDataLayer->Close(nChannelId);//这个属于主动断开客户端的连接,libevent那边调用close_channel放入下线队列中,主线程去关闭socket,这个过程应该很清楚了
    }
    catch (...)
    {
        DWORD e = GetLastError();
        Plug::PlugMessageBox(L"m_spTCPServer->close_channel异常了啊!");
    }
}
//我在想,当时踢玩家下线,是因为那边放入释放队列,60s之后关闭socket,此时客户端收到通知
但对于拔网线,那客户端好像也能收到通知,服务端心跳检测关闭时永远都不能收到通知的,那客户端怎么会收到通知的呢,来试验一下
时间: 2024-12-23 21:30:51

GS LiveMgr心跳管理类的相关文章

用户和组管理类命令

1.列出当前系统上所有已经登陆的用户的用户名,注意:同一个用户登陆多次,则显示一次即可. who |cut -d' ' -f1 | sort | uniq 2.取出最后登陆到当前系统的用户的相关信息 w|tail-n1或者who|tail-n1 3.取出当前系统上被用户当做其默认shell的最多的那个shell cat /etc/passwd | awk -F':' '{print $7}'|uniq –c 结果为:/sbin/nologin awk -F':' '{a[$7]++}END{fo

DownloadManager 下载管理类

演示 简介 从Android 2.3开始新增了一个下载管理类,在SDK的文档中我们查找android.app.DownloadManager可以看到.下载管理类可以长期处理多个HTTP下载任务,客户端只需要给出请求的Uri和存放目标文件的位置即可,下载管理使用了一个AIDL服务器,所以可以放心的在后台执行,同时实例化的方法需要使用getSystemService(Context.DOWNLOAD_SERVICE) ,我们可以轻松的通过新增的这个API实现Android平台上的文件下载操作. Do

转 学生管理类

看上去很工整. <!DOCTYPE html><html><head> <title>学生管理类</title></head><body><form method="post"> 学号:<input type="text" name="number"><br/> 姓名:<input type="text"

linux用户和管理类常用命令

1.列出当前系统上所有已经等了的用户的用户名,注意:同一个用户登录多次,则只显示一次即可. 2.取出最后登录到当前系统的用户的相关信息 3.取出当前系统上被用户当作其默认shell的最多的那个shell. 4.将/etc/passwd中的第三个字段数值最大的后10个用户的信息全部改为大写后保存至/tmp/maxusers.txt文件中. 5.取出当前主机的IP地址,提示:对ifconfig命令进行切分.  6.取出/etc/目录下所有以.com结尾的文件的文件名,并将其名字转换为大写后保存至/t

1、自学——Linux的学习进度与任务【时间日期管理类的命令】

时间日期管理类的命令 时间日期管理类的命令:  # date :显示 # date MMDDhhmm[CC]YY.ss :设定 # date [+FORMAT] %Y:4位年份 %y:  2位年份 %M:  分 %m:月 %d:  日 %H:  时 %S:  秒 %T: 时间(以冒号隔开) %D:日期 %F: 日期(以横杆隔开) unix元年:1970-01-01 00:00:00 %s:timestamp:从unix元年开始到此刻所经历的秒数 # cal   :    显示日历 # cal #

Linux学习笔记之常用用户管理类命令

开始这个话题之前应该先了解下/etc/shadow这个文件中的构造,对于列所代表的意思大体有个了解,先以ftpuser账户来介绍:后边会给出一些相应的例题来做使用说明补充(好吧我承认,这,又是一篇博客作业^_^万恶的标题党). ftpuser:$6$T9WUEls/$AK4hTO1bZ7MXW7VoMej8e7tVG1qqxPy5axVJAAIz4m6uuzZmoYDcQjUG2XlXx7klx/i8yJ1luZDtoKYwJAMa4.:16929:0:99999:7:5:: 总共有9个字段被"

动态内存管理类

内存管理类需要包括以下几个基本操作的正确性 添加元素:判断管理的空间大小是否能够添加新元素,如不够,则使用allocator分配内存,并将旧数据移动到新内存,然后释放旧内存,并更新内存首指针.第一个可用内存指针.尾指针位置. 对象拷贝:使用allocator的allocate分配内存,相关的uninitialized_copy拷贝元素到新的位置,并更新内存首指针.第一个可用内存指针.尾指针位置. 内存释放:使用allocator的destroy销毁内存,并使用deallocate执行内存回收操作

下载的管理类MyDownloadManager

import android.content.Intent; import android.net.Uri; import java.io.File; import java.io.FileOutputStream; import java.io.InputStream; import java.util.ArrayList; import java.util.HashMap; /** * 下载的管理类 */ public class MyDownloadManager { public sta

用户和组管理类命令总结

用户和组管理类命令总结 一.用户管理 1.用户创建:useradd 2.用户属性修改:usermod 3.修改用户属性:chage 4.删除用户:userdel 5.给用户添加密码:passwd 6.切换用户或以其他身份执行:su 二.组管理 7.组创建:groupadd 8.组属性修改:groupmod 9.组删除:groupdel 10.组密码:gpasswd 11.临时切换基本组:newgrp 三.查看信息 12.查看用户相关的ID信息:id 13.其他查看信息的命令:chfn.chsh.