Linux TCP客户端(定时发送/接受,断线重连)

有问题请指出,谢谢

#include <iostream>
#include <cstring>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <errno.h>
#include <signal.h>
#include <chrono>
#include <thread>
#include <fcntl.h>

int sock = -1;                    // 全局变量套接字
const int kMaxRetryCount = 60;    // 最大重试次数
const int kReconnectInterval = 1; // 重连间隔时间(秒)
const int kConnectTimeout = 2;    // 连接等待时间(秒)

#define TCP_SERVER_IP "192.168.8.115"
#define TCP_SERVER_PORT 60000
#define RECV_TIMER 500                                   //(MS)
#define SEND_TIMER 1000                                  //(MS)

void handleSignal(int signum)
{
    if (signum == SIGPIPE)
    {
        std::cerr << "Broken Pipe signal received" << std::endl;
    }
}

// 连接服务器
bool connectToServer(const std::string &ip, int port)
{
    sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if (sock == -1)
    {
        std::cerr << "Failed to create socket: " << strerror(errno) << std::endl;
        return false;
    }
    struct sockaddr_in serverAddr;
    memset(&serverAddr, 0, sizeof(serverAddr));
    serverAddr.sin_family = AF_INET;
    serverAddr.sin_port = htons(port);
    serverAddr.sin_addr.s_addr = inet_addr(ip.c_str());
    // 非阻塞模式连接服务端
    int flags = fcntl(sock, F_GETFL, 0);      // 获取套接字当前的文件状态标志(file status flags)
    fcntl(sock, F_SETFL, flags | O_NONBLOCK); // 设置套接字为非阻塞模式
    int connect_result = connect(sock, (struct sockaddr *)&serverAddr, sizeof(serverAddr));
    if (connect_result == -1 && errno == EINPROGRESS) // 临时错误:如果连接正在进行,则继续等待
    {
        fd_set write_fds;
        FD_ZERO(&write_fds);
        FD_SET(sock, &write_fds);
        struct timeval timeout
        {
        };
        timeout.tv_sec = kConnectTimeout; // 连接等待时间
        connect_result = select(sock + 1, NULL, &write_fds, NULL, &timeout);
        if (connect_result > 0 && FD_ISSET(sock, &write_fds)) // 可写:说明连接完成,可切换回阻塞模式
        {
            flags = fcntl(sock, F_GETFL, 0);
            fcntl(sock, F_SETFL, flags & ~O_NONBLOCK);
        }
        else if (connect_result == 0)
        {
            // 超时:关闭套接字,并处理超时情况
            std::cerr << "connect to server timeout: " << strerror(errno) << std::endl;
            close(sock);
            return false;
        }
        else
        {
            // 出错:关闭套接字,并处理错误情况
            std::cerr << "Failed to connect to server: " << strerror(errno) << std::endl;
            close(sock);
            return false;
        }
    }
    else if (connect_result == -1)
    {
        // 错误:关闭套接字,并处理错误情况
        std::cerr << "Failed to connect to server: " << strerror(errno) << std::endl;
        close(sock);
        return false;
    }
    else
    {
        // 连接已完成,可切换回阻塞模式
        flags = fcntl(sock, F_GETFL, 0);
        fcntl(sock, F_SETFL, flags & ~O_NONBLOCK);
    }
    signal(SIGPIPE, handleSignal);
    std::cout << "Connected to server" << std::endl;
    return true;
}

// 断线重连
bool reconnectToServer(const std::string &ip, int port)
{
    for (int retryCount = 0; retryCount < kMaxRetryCount; ++retryCount)//每次重连计数
    {
        std::cerr << "Attempt #" << retryCount + 1 << " to connect to server..." << std::endl;
        if (connectToServer(ip, port))
        {
            return true;
        }
        std::cerr << "Failed to connect to server, retrying in " << kReconnectInterval << " seconds..." << std::endl;
        std::this_thread::sleep_for(std::chrono::seconds(kReconnectInterval));//重连间隔时间
    }
    std::cerr << "Failed to connect to server after " << kMaxRetryCount << " retries" << std::endl;
    return false;
}

// 发送数据
bool sendData(const char *message)
{
    ssize_t sendSize = send(sock, message, strlen(message), 0); // 非阻塞I/O操作send()
    if (sendSize < 0)
    {
        std::cerr << "Failed to send message to server: " << strerror(errno) << std::endl;
        return false;
    }
    else if (sendSize < strlen(message))
    {
        std::cerr << "Incomplete message sent to server" << std::endl;
        return false;
    }
    std::cout << "Sent message to server: " << message << std::endl;
    return true;
}

// 接收数据
bool receiveData()
{
    char buffer[1024] = {0};
    ssize_t recvSize = recv(sock, buffer, sizeof(buffer) - 1, MSG_DONTWAIT); // 非阻塞I/O操作recv()
    if (recvSize < 0)
    {
        // 如果send()返回-1并且errno等于EWOULDBLOCK或EAGAIN,则说明非阻塞I/O操作仍在进行
        if (errno != EAGAIN && errno != EWOULDBLOCK)
        {
            std::cerr << "Failed to receive message from server: " << strerror(errno) << std::endl;
            return false;
        }
    }
    else if (recvSize == 0)
    {
        std::cerr << "Server closed connection" << std::endl;
        return false;
    }
    buffer[recvSize] = '\0';
    std::cout << "Received Size: " << (int)recvSize << std::endl;
    return true;
}

int main(int argc, char *argv[])
{
    std::string ip = TCP_SERVER_IP;
    int port = TCP_SERVER_PORT;
    // 连接服务器
    if (!connectToServer(ip, port))
    {
        if (!reconnectToServer(ip, port))
        {
            std::cerr << "Terminating program..." << std::endl;
            return -1;
        }
    }
    // 定义发送和接收数据的消息
    const char *sendMsg = "Hello, server!";
    const char *recvMsg = "Server, hello!";
    // 定时器
    std::chrono::time_point<std::chrono::steady_clock> time_recv_start = std::chrono::steady_clock::now();  // 记录开始时间
    std::chrono::time_point<std::chrono::steady_clock> time_send_start = std::chrono::steady_clock::now(); // 记录开始时间
    std::chrono::time_point<std::chrono::steady_clock> current_time = time_send_start;// 记录当前时间
    long long time_recv_duration, time_send_duration;
    while (true)
    {
        current_time = std::chrono::steady_clock::now();                                                           // 记录当前时间
        time_recv_duration = std::chrono::duration_cast<std::chrono::milliseconds>(current_time - time_recv_start).count(); // 计算经过的时间
        if (time_recv_duration >= RECV_TIMER)
        {
            time_recv_start = std::chrono::steady_clock::now(); // 记录开始时间
            // 接收数据
            if (!receiveData())
            {
                close(sock);
                std::cerr << "Failed to communicate with server, reconnecting..." << std::endl;
                if (!reconnectToServer(ip, port))
                {
                    std::cerr << "Terminating program..." << std::endl;
                    return -1;
                }
            }
        }
        current_time = std::chrono::steady_clock::now();                                                             // 记录当前时间
        time_send_duration = std::chrono::duration_cast<std::chrono::milliseconds>(current_time - time_send_start).count(); // 计算经过的时间
        if (time_send_duration >= SEND_TIMER)                                                                                            // 如果经过的时间大于等于100ms
        {
            time_send_start = std::chrono::steady_clock::now(); // 记录开始时间
            // 发送数据
            if (!sendData(sendMsg))
            {
                close(sock);
                std::cerr << "Failed to communicate with server, reconnecting..." << std::endl;
                if (!reconnectToServer(ip, port))
                {
                    std::cerr << "Terminating program..." << std::endl;
                    return -1;
                }
            }
        }
        std::this_thread::sleep_for(std::chrono::milliseconds(1));//1ms
    }
    return 0;
}

 

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在LabVIEW中编写TCP/IP断线重连功能可以采用以下步骤: 1. 首先,建立一个TCP/IP连接。使用TCP/IP VIs或TCP/IP函数将与目标设备建立一个稳定的连接。确保设置了适当的IP地址和端口号。 2. 在连接建立之后,使用循环来监测连接状态。您可以使用TCP/IP VIs中的一些函数来检测连接状态。比如,使用“TCP/IP Is Connected”函数检查连接是否仍然处于活动状态。 3. 如果连接状态检测到连接断开,即连接不再处于活动状态,您可以使用“TCP/IP Close Connection”函数来关闭当前的连接。 4. 接下来,使用“TCP/IP Open Connection”函数来重新建立TCP/IP连接。在此之前,请确保您已预先设置好正确的IP地址和端口号。 5. 当连接重新建立后,继续进行数据传输或执行其他操作。 6. 将这些步骤放入一个循环中,确保在连接断开时能够自动重连。可以设置一个标志位,当连接断开时,标志位为真,循环检测到标志位为真时重复执行以上步骤。 7. 最后,在程序中加入适当的错误处理机制,以处理连接失败和其他异常情况。您可以使用“TCP/IP Get Last Error”函数来获取最后一个错误代码,并进行相应的处理。 通过以上步骤,您就可以在LabVIEW中实现TCP/IP断线重连功能,以确保与目标设备的稳定通信。 ### 回答2: 要在LabVIEW中编写TCP/IP断线重连功能,可以按照以下步骤进行: 1. 创建一个TCP/IP客户端连接的VI。使用LabVIEW的TCP/IP工具包,可以从函数面板中找到TCP Open Connection函数。将该函数配置为客户端模式,并填入远程主机的IP地址和端口号。 2. 在主程序中创建一个循环,用于监测连接状态。可以使用一个While循环实现,设置循环条件为“连接断开”。 3. 在循环中,添加一个TCP Read函数,用于检测连接状态。将该函数配置为读取指定字节数,并将读取到的数据与一个常数进行比较。如果读取到的数据与常数不匹配,即表示连接已断开。 4. 在连接断开的条件下,使用TCP Close Connection函数关闭当前连接。 5. 添加一个等待函数,设置等待时间为几秒钟,以避免频繁尝试重连。 6. 在循环中添加一个TCP Open Connection函数,重新建立连接。设置连接的IP地址和端口号。 7. 使用一个条件结构,将连接建立的结果进行判断。如果连接成功,则将“连接断开”的条件设置为假,退出循环。如果连接失败,则继续循环,进行下一次尝试。 8. 在主程序中,执行需要发送或接收数据的操作,并在循环内进行。 上述步骤简要说明了在LabVIEW中编写TCP/IP断线重连的方法,具体实现还需要根据具体需求进行适当的修改和调整。同时,在编写过程中需要注意异常处理,确保程序能够正确处理连接异常和重连操作。 ### 回答3: 为了实现LabVIEW编写TCP/IP断线重连功能,需要进行以下步骤: 1. 建立TCP/IP连接: 使用LabVIEW中的TCP/IP相关VIs来建立与远程主机的TCP/IP连接。这可以通过使用"TCP Open Connection" VI来实现。在这一步骤中,需要提供目标IP地址和端口号。 2. 执行数据传输操作: 在建立TCP/IP连接后,可以使用LabVIEW中的TCP/IP通信VIs来实现数据的收发。这可以通过使用"TCP Read"和"TCP Write"这样的VIs来实现。 3. 检测连接状态: 通过定期检查连接状态,可以判断是否与远程主机断开连接。使用"TCP/IP Connection State.vi" 可以获得当前连接状态。如果连接状态显示为"断开连接",则需要进行断线重连操作。 4. 断线重连: 一旦检测到连接断开,可以使用"TCP Close Connection" VI来关闭当前连接。然后再次使用"TCP Open Connection" VI来重新建立连接。这个过程可以在一个循环结构中进行,直到成功建立连接。 总结:通过使用LabVIEW中的TCP/IP相关VIs,我们可以实现TCP/IP断线重连功能。这需要在建立连接、数据传输和连接状态检测的基础上进行实现。重连的过程包括关闭当前连接,并重新建立连接。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值