【问题标题】:TCP C Program Autoreconnect to clientTCP C 程序自动重新连接到客户端
【发布时间】:2020-04-11 19:23:29
【问题描述】:

我目前有一个使用 TCP 远程连接的传感器。问题是远程位置不断断开连接,我不知道如何纠正我的代码。我对我在做什么有 3/10 的想法,这段代码中的大部分内容来自互联网。

问题是,我如何实现自动重新连接到我的 tcp 客户端的代码?尝试了很多方法,但似乎都不起作用。但是,如果连接是无缝的,则代码可以完美运行。

#include <stdio.h> 
#include <netdb.h> 
#include <netinet/in.h> 
#include <stdlib.h> 
#include <string.h> 
#include <sys/socket.h> 
#include <sys/types.h> 
#include <time.h>
#define MAX 1000 
#define PORT 14344
#define SA struct sockaddr 

int main() 
{ 
        time_t t;
    time(&t);
    int sockfd, connfd, len; 
    struct sockaddr_in servaddr, cli; 
    unsigned char buff[MAX]; 
    int n,dd,a,b,c; 
    char timebuffer [80];
    time_t rawtime;
    struct tm * timeinfo;
    time ( &rawtime );
    timeinfo = localtime ( &rawtime );
        strftime (timebuffer,80,"/home/%m%d%y_%H%M%S",timeinfo);  
    FILE *fil1;
    fil1=fopen(timebuffer,"wb");

    printf("%s",ctime(&t));

    sockfd = socket(AF_INET, SOCK_STREAM, 0); 
    if (sockfd == -1) { 
        printf("socket creation failed...\n"); 
        exit(0); } 
    else
        printf("Socket successfully created..\n"); 

    bzero(&servaddr, sizeof(servaddr)); 
    servaddr.sin_family = AF_INET; 
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY); 
    servaddr.sin_port = htons(PORT); 

    if ((bind(sockfd, (SA*)&servaddr, sizeof(servaddr))) != 0) { 
        printf("socket bind failed...\n"); 
        exit(0); } 
    else    
    {
        printf("Socket successfully binded..\n"); 

        if ((listen(sockfd, 5)) != 0) { 
            printf("Listen failed...\n"); 
            exit(0); } 
        else
            printf("Server listening..\n");

        len = sizeof(cli); 

        connfd = accept(sockfd, (SA*)&cli, &len); 

        if (connfd < 0) { 
            printf("server accept failed...\n"); 
            exit(0); } 
        else    
            printf("server accepted...\n");

        while (1){
            bzero(buff, MAX); 
            dd=read(connfd, buff, sizeof(buff)); 
            printf("\n"); 

            for(a=0; a<=dd-1; a++){fprintf(fil1, "%c", buff[a]);printf("%02X", buff[a]);}

            printf("\n"); 
            bzero(buff, MAX);
            }
    }

        close(connfd);   
}

【问题讨论】:

  • 你发布的代码是socket服务器的,它监听传入的连接并从中读取,你应该发布你想要修复的tcp客户端的代码
  • 对不起。我的客户是USRIOT设备usriot.com/products/rs232-to-ethernet-converter.html我只能在我这边接收数据。
  • 服务器无法连接到客户端。只有客户端可以连接到服务器。所以你的问题真的没有意义。相反,您可能应该首先解决连接断开的原因。
  • 我明白了。它是一个卫星链接,因为它太远了,而且它依赖于天气。一旦遇到恶劣天气或停电,链接就会失效并再次接收它需要手动重新激活上面的代码。另一边的传感器只要有可能就一直发送。我尝试了 UDP,但我不断丢失重要的数据包,所以我转向了 tcp。我只是想知道是否有一种更智能的方式来接收 tcp 数据包,无论到客户端的链接是否断开了一段时间,它都只接收数据。就像 UDP 一样
  • @JMF 不,TCP 没有自动重新连接机制。您可以尝试将双方的超时时间增加到较高的值(参见 man 7 tcp)

标签: c tcp tcpclient


【解决方案1】:

必须发送一些数据。如果您不发送数据,则无法保证您可以检测到丢失的连接。您可以可靠地判断另一端是否仍然连接的唯一方法是您尝试向它发送一些数据并且它失败了。不尝试,你无法知道另一端是否在那里。

接下来,您必须检查read 的返回值。如果出现错误,您需要关闭connfd 并循环回到accept 调用以获取新连接。

【讨论】:

  • 谢谢大卫。我想那是最好的方法。我正在考虑使用计时器(因为我知道数据间隔进入,手动关闭和打开),但检查读取的返回更好。
  • 如果您知道在正常情况下您将至少每隔一段时间收到一次数据,您可以使用一个计时器,如果您打开了一个连接但没有接收到该连接,您可以使用该计时器来close有足够时间的数据。不过我不建议这样做,特别是如果您的代码在确定现有连接失败后只会尝试accept 新连接。对方可能在你之前很久就意识到它失去了连接,尝试重新连接,你的代码直到很久以后才会调用accept
  • 嗨,大卫。我刚刚在 while 函数中添加了 read 函数来检查传入的数据并且它起作用了。非常感谢。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2015-03-10
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-06-18
  • 2018-02-27
相关资源
最近更新 更多