TCP socket心跳包示例程序_xqhrs232的专栏-CSDN博客_setsockopt 心跳包

原文地址::TCP socket心跳包示例程序_神奕的专栏-CSDN博客_tcp心跳包

相关文章

1、Linux网络编程--服务端判断客户端断开的经验方法 ----Linux网络编程--服务端判断客户端断开的经验方法_志存高远-CSDN博客_info.tcpi_state

2、Socket心跳包机制及SO_KEEPALIVE套接口选项----Socket心跳包机制及SO_KEEPALIVE套接口选项_u012252959的博客-CSDN博客

3、TCP Keepalive HOWTO----TCP Keepalive HOWTO

在做游戏开发时,经常需要在应用层实现自己的心跳机制,即定时发送一个自定义的结构体(心跳包),让对方知道自己还活着,以确保连接的有效性。

在TCP socket心跳机制中,心跳包可以由服务器发送给客户端,也可以由客户端发送给服务器,不过比较起来,前者开销可能更大。—— 这里实现的是由客户端给服务器发送心跳包,基本思路是:

1) 服务器为每个客户端保存了IP和计数器count,即map<fd, pair<ip, count>>。服务端主线程采用 select 实现多路IO复用,监听新连接以及接受数据包(心跳包),子线程用于检测心跳:

  • 如果主线程接收到的是心跳包,将该客户端对应的计数器 count 清零;
  • 在子线程中,每隔3秒遍历一次所有客户端的计数器 count: 
    • 若 count 小于 5,将 count 计数器加 1;
    • 若 count 等于 5,说明已经15秒未收到该用户心跳包,判定该用户已经掉线;

2) 客户端则只是开辟子线程,定时给服务器发送心跳包(本示例中定时时间为3秒)。

下面是Linux下一个socket心跳包的简单实现:

/*************************************************************************> File Name: Server.cpp> Author: SongLee> E-mail: lisong.shine@qq.com> Created Time: 2016年05月05日 星期四 22时50分23秒> Personal Blog: http://songlee24.github.io/************************************************************************/
#include<netinet/in.h>   // sockaddr_in
#include<sys/types.h>    // socket
#include<sys/socket.h>   // socket
#include<arpa/inet.h>
#include<unistd.h>
#include<sys/select.h>   // select
#include<sys/ioctl.h>
#include<sys/time.h>
#include<iostream>
#include<vector>
#include<map>
#include<string>
#include<cstdlib>
#include<cstdio>
#include<cstring>
using namespace std;
#define BUFFER_SIZE 1024enum Type {HEART, OTHER};struct PACKET_HEAD
{Type type;int length;
};void* heart_handler(void* arg);class Server
{
private:struct sockaddr_in server_addr;socklen_t server_addr_len;int listen_fd;    // 监听的fdint max_fd;       // 最大的fdfd_set master_set;   // 所有fd集合,包括监听fd和客户端fd   fd_set working_set;  // 工作集合struct timeval timeout;map<int, pair<string, int> > mmap;   // 记录连接的客户端fd--><ip, count>
public:Server(int port);~Server();void Bind();void Listen(int queue_len = 20);void Accept();void Run();void Recv(int nums);friend void* heart_handler(void* arg);
};Server::Server(int port)
{bzero(&server_addr, sizeof(server_addr));server_addr.sin_family = AF_INET;server_addr.sin_addr.s_addr = htons(INADDR_ANY);server_addr.sin_port = htons(port);// create socket to listenlisten_fd = socket(PF_INET, SOCK_STREAM, 0);if(listen_fd < 0){cout << "Create Socket Failed!";exit(1);}int opt = 1;setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
}Server::~Server()
{for(int fd=0; fd<=max_fd; ++fd){if(FD_ISSET(fd, &master_set)){close(fd);}}
}void Server::Bind()
{if(-1 == (bind(listen_fd, (struct sockaddr*)&server_addr, sizeof(server_addr)))){cout << "Server Bind Failed!";exit(1);}cout << "Bind Successfully.\n";
}void Server::Listen(int queue_len)
{if(-1 == listen(listen_fd, queue_len)){cout << "Server Listen Failed!";exit(1);}cout << "Listen Successfully.\n";
}void Server::Accept()
{struct sockaddr_in client_addr;socklen_t client_addr_len = sizeof(client_addr);int new_fd = accept(listen_fd, (struct sockaddr*)&client_addr, &client_addr_len);if(new_fd < 0){cout << "Server Accept Failed!";exit(1);}string ip(inet_ntoa(client_addr.sin_addr));    // 获取客户端IPcout << ip << " new connection was accepted.\n";mmap.insert(make_pair(new_fd, make_pair(ip, 0)));// 将新建立的连接的fd加入master_setFD_SET(new_fd, &master_set);if(new_fd > max_fd){max_fd = new_fd;}
}   void Server::Recv(int nums)
{for(int fd=0; fd<=max_fd; ++fd){if(FD_ISSET(fd, &working_set)){bool close_conn = false;  // 标记当前连接是否断开了PACKET_HEAD head;recv(fd, &head, sizeof(head), 0);   // 先接受包头if(head.type == HEART){mmap[fd].second = 0;        // 每次收到心跳包,count置0cout << "Received heart-beat from client.\n";}else{// 数据包,通过head.length确认数据包长度 }   if(close_conn)  // 当前这个连接有问题,关闭它{close(fd);FD_CLR(fd, &master_set);if(fd == max_fd)  // 需要更新max_fd;{while(FD_ISSET(max_fd, &master_set) == false)--max_fd;}}}}
}void Server::Run()
{pthread_t id;     // 创建心跳检测线程int ret = pthread_create(&id, NULL, heart_handler, (void*)this);if(ret != 0){cout << "Can not create heart-beat checking thread.\n";}max_fd = listen_fd;   // 初始化max_fdFD_ZERO(&master_set);FD_SET(listen_fd, &master_set);  // 添加监听fdwhile(1){FD_ZERO(&working_set);memcpy(&working_set, &master_set, sizeof(master_set));timeout.tv_sec = 30;timeout.tv_usec = 0;int nums = select(max_fd+1, &working_set, NULL, NULL, &timeout);if(nums < 0){cout << "select() error!";exit(1);}if(nums == 0){//cout << "select() is timeout!";continue;}if(FD_ISSET(listen_fd, &working_set))Accept();   // 有新的客户端请求elseRecv(nums); // 接收客户端的消息}
}// thread function
void* heart_handler(void* arg)
{cout << "The heartbeat checking thread started.\n";Server* s = (Server*)arg;while(1){map<int, pair<string, int> >::iterator it = s->mmap.begin();for( ; it!=s->mmap.end(); ){   if(it->second.second == 5)   // 3s*5没有收到心跳包,判定客户端掉线{cout << "The client " << it->second.first << " has be offline.\n";int fd = it->first;close(fd);            // 关闭该连接FD_CLR(fd, &s->master_set);if(fd == s->max_fd)      // 需要更新max_fd;{while(FD_ISSET(s->max_fd, &s->master_set) == false)s->max_fd--;}s->mmap.erase(it++);  // 从map中移除该记录}else if(it->second.second < 5 && it->second.second >= 0){it->second.second += 1;++it;}else{++it;}}sleep(3);   // 定时三秒}
}int main()
{Server server(15000);server.Bind();server.Listen();server.Run();return 0;
}
/*************************************************************************> File Name: Client.cpp> Author: SongLee> E-mail: lisong.shine@qq.com> Created Time: 2016年05月05日 星期四 23时41分56秒> Personal Blog: http://songlee24.github.io/************************************************************************/
#include<netinet/in.h>   // sockaddr_in
#include<sys/types.h>    // socket
#include<sys/socket.h>   // socket
#include<arpa/inet.h>
#include<sys/ioctl.h>
#include<unistd.h>
#include<iostream>
#include<string>
#include<cstdlib>
#include<cstdio>
#include<cstring>
using namespace std;
#define BUFFER_SIZE 1024enum Type {HEART, OTHER};struct PACKET_HEAD
{Type type;int length;
};void* send_heart(void* arg); class Client
{
private:struct sockaddr_in server_addr;socklen_t server_addr_len;int fd;
public:Client(string ip, int port);~Client();void Connect();void Run();friend void* send_heart(void* arg);
};Client::Client(string ip, int port)
{bzero(&server_addr, sizeof(server_addr));server_addr.sin_family = AF_INET;if(inet_pton(AF_INET, ip.c_str(), &server_addr.sin_addr) == 0){cout << "Server IP Address Error!";exit(1);}server_addr.sin_port = htons(port);server_addr_len = sizeof(server_addr);// create socketfd = socket(AF_INET, SOCK_STREAM, 0);if(fd < 0){cout << "Create Socket Failed!";exit(1);}
}Client::~Client()
{close(fd);
}void Client::Connect()
{cout << "Connecting......" << endl;if(connect(fd, (struct sockaddr*)&server_addr, server_addr_len) < 0){cout << "Can not Connect to Server IP!";exit(1);}cout << "Connect to Server successfully." << endl;
}void Client::Run()
{pthread_t id;int ret = pthread_create(&id, NULL, send_heart, (void*)this);if(ret != 0){cout << "Can not create thread!";exit(1);}
}// thread function
void* send_heart(void* arg)
{cout << "The heartbeat sending thread started.\n";Client* c = (Client*)arg;int count = 0;  // 测试while(1) {PACKET_HEAD head;head.type = HEART;head.length = 0;    send(c->fd, &head, sizeof(head), 0);sleep(3);     // 定时3秒++count;      // 测试:发送15次心跳包就停止发送if(count > 15)break;}
}int main()
{Client client("127.0.0.1", 15000);client.Connect();client.Run();while(1){string msg;getline(cin, msg);if(msg == "exit")break;cout << "msg\n";}return 0;
}

可以看出,客户端启动以后发送了15次心跳包,然后停止发送心跳包。在经过一段时间后(3s*5),服务器就判断该客户端掉线,并断开了连接。

TCP socket心跳包示例程序相关推荐

  1. golang中tcp socket粘包问题和处理

    http://www.01happy.com/golang-tcp-socket-adhere/ 在用golang开发人工客服系统的时候碰到了粘包问题,那么什么是粘包呢?例如我们和客户端约定数据交互格 ...

  2. web socket 心跳包的实现方案

    web socket 心跳包的实现方案05/30/2010 现在网络环境错综复杂,socket心跳包是获得健康强壮的连接的有效解决方案,今天,我们就在web socket中实现心跳包方案,是的,尽管我 ...

  3. 三菱Q系列PLC编程TCP Socket套接字程序

    三菱Q系列PLC编程TCP Socket套接字程序 用于和上位机通讯 支持掉线自动后自动重连 附详细注释和指令说明! 编号:969665940010231沉默物语

  4. java websocket心跳包_web socket 心跳包的实现方案

    web socket 心跳包的实现方案05/30/2010 现在网络环境错综复杂,socket心跳包是获得健康强壮的连接的有效解决方案,今天,我们就在web socket中实现心跳包方案,是的,尽管我 ...

  5. java socket 心跳包_socket中的短连接与长连接,心跳包示例详解

    TCP连接简介 当网络通信时采用TCP协议时,在真正的读写操作之前,server与client之间必须建立一个连接, 当读写操作完成后,双方不再需要这个连接时它们可以释放这个连接, 连接的建立是需要三 ...

  6. java长连接心跳包_socket中的短连接与长连接,心跳包示例详解

    TCP连接简介 当网络通信时采用TCP协议时,在真正的读写操作之前,server与client之间必须建立一个连接, 当读写操作完成后,双方不再需要这个连接时它们可以释放这个连接, 连接的建立是需要三 ...

  7. Socket心跳包机制

    心跳包的发送,通常有两种技术 方法1:应用层自己实现的心跳包 由应用程序自己发送心跳包来检测连接是否正常,大致的方法是:服务器在一个 Timer事件中定时 向客户端发送一个短小精悍的数据包,然后启动一 ...

  8. Socket心跳包机制 .

    心跳包的发送,通常有两种技术 方法1:应用层自己实现的心跳包 由应用程序自己发送心跳包来检测连接是否正常,大致的方法是:服务器在一个 Timer事件中定时 向客户端发送一个短小精悍的数据包,然后启动一 ...

  9. Socket心跳包机制总结

         跳包之所以叫心跳包是因为:它像心跳一样每隔固定时间发一次,以此来告诉服务器,这个客户端还活着.事实上这是为了保持长连接,至于这个包的内容,是没有什么特别规定的,不过一般都是很小的包,或者只包 ...

最新文章

  1. Jenkins 2.9.1 安装文档
  2. 什么是缓存一致性问题?如何解决呢?
  3. spark1.3.1使用基础教程
  4. servlet.jar--jar not loaded错误
  5. php维护session,维护带有cookie的PHP session_start()
  6. uniapp添加网站favicon文件
  7. 欲取代硬盘?SSD固态存储器前景分析
  8. Android2.2+opencv3.1配置实现
  9. 香港股票交易成本计算器 android,股票交易手续费计算器
  10. 新版 AS9100D:2016航空航天质量管理体系标准变化内容
  11. 保研之路——中山大学数据科学与计算机学院直硕夏令营
  12. windows自带备份驱动
  13. POJO类toString()方法
  14. 阿里云ddns过程记录
  15. 文件夹重命名失败,删除失败,提示‘文件正在被使用‘
  16. LeetCode 6118. 最小差值平方和
  17. python时间戳转换成时间_Python时间,日期,时间戳之间转换,时间转换时间戳,Python时间戳转换时间,Python时间转换时间戳...
  18. 北京市中小学信息学竞赛汇总
  19. 【Tableau 图表大全1.0】之条形图
  20. error C2275 将此类型用作表达式非法

热门文章

  1. eclipse 开发 scala
  2. 玩转MySQL之Linux下的简单操作(服务启动与关闭、启动与关闭、查看版本)
  3. SAP-MM:发票、贷方凭证、事后借记、后续贷记
  4. 前端项目难点及解决方法_预埋件施工重点难点的解决方法
  5. android 回退函数,详解React Native监听Android回退按键与程序化退出应用
  6. flask 检测post是否为空_使用Flask搭建一个校园论坛-4
  7. 查询已有链表的hashmap_原创 | 面试不再慌,看完这篇保证让你写HashMap跟玩一样...
  8. gis 联合 融合_GIS技术进化 | 我们为何需要跨平台GIS技术体系?
  9. 利用scp 远程上传下载文件/文件夹
  10. 7-17 爬动的蠕虫 (15 分)