connect 的断线重连设计

客户端会面临服务器崩溃的情况,我们可以试着写一个客户端重连的代码,模拟并理 解一些客户端行为,比如游戏客户端等。

TcpClient.cc

  • 采用状态机,实现一个简单的 tcp client 可以实现重连效果
cpp 复制代码
#include <iostream>
#include <string>
#include <cstring>
#include <cstdlib>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

using namespace std;

void Usage(const std::string &process)
{
    std::cout << "Usage: " << process << " server_ip server_port"
<< std::endl;
}

enum class Status // C++11 强类型枚举
{
    NEW, // 新建状态,就是单纯的连接
    CONNECTING, // 正在连接,仅仅方便查询 conn 状态
    CONNECTED, // 连接或者重连成功
    DISCONNECTED, // 重连失败
    CLOSED // 连接失败,经历重连,无法连接
};

class ClientConnection
{
public:
    ClientConnection(uint16_t serverport, const std::string &serverip)
    : _sockfd(-1),
    _serverport(serverport),
    _serverip(serverip),
    _retry_interval(1),
    _max_retries(5),
    _status(Status::NEW)
    {}

    void Connect()
    {
        // 1. 创建 socket
        _sockfd = socket(AF_INET, SOCK_STREAM, 0);
        if (_sockfd < 0)
        {
            cerr << "socket error" << endl;
            exit(1);
        }
        // 2. 要不要 bind?必须要有 Ip 和 Port, 需要 bind,但是不需要用户显示的 bind,client 系统随机端口
        // 发起连接的时候,client 会被 OS 自动进行本地绑定
        // 2. connect
        struct sockaddr_in server;
        memset(&server, 0, sizeof(server));
        server.sin_family = AF_INET;
        server.sin_port = htons(_serverport);
        // p:process(进程), n(网络) -- 不太准确,但是好记忆
        inet_pton(AF_INET, _serverip.c_str(), &server.sin_addr);
        // 1. 字符串 ip->4 字节 IP 2. 网络序列
        int n = connect(_sockfd, (struct sockaddr *)&server,
        sizeof(server)); // 自动进行 bind 哦!
        if (n < 0)
        {
            Disconnect(); // 恢复_sockfd 的默认值,是连接没有成功,不代表 sockfd 创建没有成功
            _status = Status::DISCONNECTED; // 没有连接成功
            return;
        }
        _status = Status::CONNECTED; // 连接成功
    }

    int SocketFd()
    {
        return _sockfd;
    }

    void Reconnect()
    {
        _status = Status::CONNECTING; // 正在重连
        int count = 0;
        while (count < _max_retries)
        {
            Connect(); // 重连
            if (_status == Status::CONNECTED)
            {
                return;
            }
            sleep(_retry_interval);
            count++;
            std::cout << "重连次数: " << count << ", 最大上限: " <<
            _max_retries << std::endl;
        }
        _status = Status::CLOSED; // 重连失败,可以关闭了
    }

    void Disconnect()
    {
        if (_sockfd != -1)
        {
            close(_sockfd);
            _status = Status::CLOSED;
            _sockfd = -1;
        }
    }

    Status GetStatus()
    {
        return _status;
    }

    void Process()
    {
        // 简单的 IO 即可
        while (true)
        {
            string inbuffer;
            cout << "Please Enter# ";
            getline(cin, inbuffer);
            if(inbuffer.empty()) continue;
            ssize_t n = write(_sockfd, inbuffer.c_str(),
            inbuffer.size());
            if (n > 0)
            {
                char buffer[1024];
                ssize_t m = read(_sockfd, buffer, sizeof(buffer) -1);
                if (m > 0)
                {
                    buffer[m] = 0;
                    cout << "echo messsge -> " << buffer << endl;
                }
                else if (m == 0) // 这里证明 server 端掉线了
                {
                    _status = Status::DISCONNECTED;
                    break;
                }
                else
                {
                    std::cout << "read m : " << m << "errno: " <<
                    errno << "errno string: " << strerror(errno) << std::endl;
                    _status = Status::CLOSED;
                    break;
                }
            }
            else
            {
                std::cout << "write n : " << n << "errno: " <<
                errno << "errno string: " << strerror(errno) << std::endl;
                _status = Status::CLOSED;
                break;
            }
        }
    }

    ~ClientConnection()
    {
        Disconnect();
    }

private:
    int _sockfd;
    uint16_t _serverport; // server port 端口号
    std::string _serverip; // server ip 地址
    int _retry_interval; // 重试时间间隔
    int _max_retries; // 重试次数
    Status _status; // 连接状态
};

class TcpClient
{
public:
    TcpClient(uint16_t serverport, const std::string &serverip) :
    _conn(serverport, serverip)
    {}

    void Execute()
    {
        while (true)
        {
            switch (_conn.GetStatus())
            {
                case Status::NEW:
                    _conn.Connect();
                    break;
                case Status::CONNECTED:
                    std::cout << "连接成功, 开始进行通信." << std::endl;
                    _conn.Process();
                    break;
                case Status::DISCONNECTED:
                    std::cout << "连接失败或者对方掉线,开始重连." << std::endl;
                    _conn.Reconnect();
                    break;
                case Status::CLOSED:
                    _conn.Disconnect();
                    std::cout << "重连失败, 退出." << std::endl;
                    return; // 退出
                default:
                    break;
            }
        }
    }

    ~TcpClient()
    {}

private:
    ClientConnection _conn; // 简单组合起来即可
};

// class Tcp
// ./tcp_client serverip serverport
int main(int argc, char *argv[])
{
    if (argc != 3)
    {
        Usage(argv[0]);
        return 1;
    }
    std::string serverip = argv[1];
    uint16_t serverport = stoi(argv[2]);
    TcpClient client(serverport, serverip);
    client.Execute();
    return 0;
}

测试重连

相关推荐
网硕互联的小客服38 分钟前
Apache 如何支持SHTML(SSI)的配置方法
运维·服务器·网络·windows·php
落日漫游1 小时前
K8s核心组件全解析
运维·docker·运维开发
Demisse4 小时前
[Linux] Linux文件系统基本管理
linux·运维·服务器
BAOYUCompany4 小时前
暴雨服务器:以定制化满足算力需求多样化
运维·服务器
青岛佰优联创新科技有限公司6 小时前
移动板房的网络化建设
服务器·人工智能·云计算·智慧城市
禁默6 小时前
进程替换:从 “改头换面” 到程序加载的底层逻辑
linux·运维·服务器
gameatp6 小时前
从 Windows 到 Linux 服务器的全自动部署教程(免密登录 + 压缩 + 上传 + 启动)
linux·服务器·windows
一匹电信狗6 小时前
【C++】异常详解(万字解读)
服务器·c++·算法·leetcode·小程序·stl·visual studio
Python私教8 小时前
Docker in Test:用一次性的真实环境,终结“测试永远跑不通”魔咒
运维·docker·容器
张3蜂8 小时前
深度解读 Browser-Use:让 AI 驱动浏览器自动化成为可能
运维·人工智能·自动化