客户端长连接断线重连机制

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
// testsocketclient.cpp : 定义控制台应用程序的入口点。
//
 
#include "stdafx.h"
 
#include <winsock2.h>
#include <stdio.h>
#include <mstcpip.h>
#pragma comment(lib, "Ws2_32.lib")
SOCKET sockClient = 0;
static int conn_status = 0;
#define IP_ADDRESS ("192.168.207.131")
#define PORT (9099)
#include <iostream>
using namespace std;
int socket_tcp_alive(int socket)
{
    int ret = 0;
 
    int keep_alive = 1;
    ret = setsockopt(socket, SOL_SOCKET, SO_KEEPALIVE, (char*)&keep_alive, sizeof(keep_alive));
 
    if (ret == SOCKET_ERROR)
    {
        printf("setsockopt failed: %d \n", WSAGetLastError());
        return -1;
    }
 
    struct tcp_keepalive in_keep_alive = { 0 };
    unsigned long ul_in_len = sizeof(struct tcp_keepalive);
    struct tcp_keepalive out_keep_alive = { 0 };
    unsigned long ul_out_len = sizeof(struct tcp_keepalive);
    unsigned long ul_bytes_return = 0;
 
    in_keep_alive.onoff = 1;                    /*打开keepalive*/
    in_keep_alive.keepaliveinterval = 5000; /*发送keepalive心跳时间间隔-单位为毫秒*/
    in_keep_alive.keepalivetime = 1000;         /*多长时间没有报文开始发送keepalive心跳包-单位为毫秒*/
 
    ret = WSAIoctl(socket, SIO_KEEPALIVE_VALS, (LPVOID)&in_keep_alive, ul_in_len,
        (LPVOID)&out_keep_alive, ul_out_len, &ul_bytes_return, NULL, NULL);
 
    if (ret == SOCKET_ERROR)
    {
        printf("WSAIoctl failed: %d \n", WSAGetLastError());
        return -1;
    }
 
    return 0;
}
int tcp_connect_server()
{
    int save_errno;
    SOCKADDR_IN  server_addr;
    conn_status = 0;
    memset(&server_addr, 0, sizeof(server_addr));
 
    server_addr.sin_family = AF_INET;
    server_addr.sin_addr.s_addr = inet_addr(IP_ADDRESS);
    server_addr.sin_port = htons(PORT);
 
    sockClient = ::socket(PF_INET, SOCK_STREAM, 0);
    if (sockClient == -1)
        return sockClient;
 
 
    conn_status = ::connect(sockClient, (struct sockaddr*)&server_addr, sizeof(server_addr));
 
    if (conn_status == -1)
    {
        save_errno = errno;   //清理 
        closesocket(sockClient);
        errno = save_errno; //the close may be error 
        return -1;
    }
 
    //  evutil_make_socket_nonblocking(sockfd);
 
    return sockClient;
}
 
DWORD WINAPI monitorThread(LPVOID pM)
{
    while (1)
    {
        char szRecvBuf[10] = { 0 };
    //  send(sockClient, "this is qiang ge1233", strlen("this is qiang ge1234") + 1, 0);
        int nRet = recv(sockClient, szRecvBuf, 1, MSG_PEEK); // 注意, 最后一个参数必须是MSG_PEEK, 否则会影响主线程接收信息
        if (nRet <= 0)
        {
            printf("监测到啦: nRet is %d\n", nRet);
            closesocket(sockClient);
            sockClient = NULL;
            sockClient = tcp_connect_server();
            if (conn_status != -1)
            socket_tcp_alive(sockClient);
             
         // break;
        }
        Sleep(5000);
 
    }
    return 0;
}
 
 
void main()
{
    //加载套接字库,版本协商
    WORD wVersionRequired;
    WSADATA wsaData;
    int err;
    int status, save_errno;
    //请求库
    //WSAStartup( WORD wVersionRequired, WSADATA lpWSAData );
    wVersionRequired = MAKEWORD(1, 1);
    err = WSAStartup(wVersionRequired, &wsaData);
 
    if (err != 0)
    {
        //不等于0就退出
        return;
    }
 
    //高字节和低字节都不为1
    if (LOBYTE(wsaData.wVersion) != 1 || HIBYTE(wsaData.wVersion) != 1)
    {
        WSACleanup();
        return;
    }
    sockClient = socket(AF_INET, SOCK_STREAM, 0);
    //设定服务器的地址信息
    SOCKADDR_IN addrSrv;
    addrSrv.sin_addr.S_un.S_addr = inet_addr(IP_ADDRESS);
    addrSrv.sin_family = AF_INET;
    addrSrv.sin_port = htons(PORT);
    while (1)
    {
        conn_status = connect(sockClient, (SOCKADDR*)&addrSrv, sizeof(SOCKADDR));
        if (conn_status == -1)
    {
        save_errno = errno;   //清理 
        errno = save_errno; //the close may be error 
        Sleep(3000);
        cout << "one lian test" << endl;
        continue;
    }
    else
    {
        socket_tcp_alive(sockClient);
        break;
    }
     
    }
    HANDLE handle = CreateThread(NULL, 0, monitorThread, NULL, 0, NULL);
    
//  send(sockClient, "this is qiang ge", strlen("this is qiang ge") + 1, 0);
      
    getchar();
  
  
 
    closesocket(sockClient);
    WSACleanup();
    system("PAUSE");
}

  

posted on   lydstory  阅读(1857)  评论(0编辑  收藏  举报

编辑推荐:
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
阅读排行:
· 全程不用写代码,我用AI程序员写了一个飞机大战
· DeepSeek 开源周回顾「GitHub 热点速览」
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了

导航

< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

统计

点击右上角即可分享
微信分享提示