小言_互联网的博客

linux网络编程学习笔记(6)——TCP连接状态的多种判断方法

437人阅读  评论(0)

    在TCP网络编程模型中,无论是客户端还是服务端,在网络编程的过程中都需要判断连接的对方网络状态是否正常。在linux系统中,有很多种方式可以判断连接的对方网络是否已经断开。

  • 通过错误码和信号判断
  • 通过select系统函数判断
  • 通过TCP_INFO套接字选项判断
  • 通过SO_KEEPALIVE套接字选项判断
  • 通过SO_RCVTIMEO/SO_SNDTIMEO判断

(一)通过错误码和信号判断

(1)写数据信号和错误码判断

    在写TCP连接数据的时候,如果对方连接已经正常断开,那么写数据端将会收到一个SIGPIPE信号,可以通过这个信号知道对方连接已经断开。该信号信号会终止当前进程,如果不在对方连接断开不退出进程,那么就应该注册信号函数。
    同时,如果对方连接已经正常断开,那么write写数据端将会返回写错误。返回的写长度为-1,此时的错误码为:32,对应错误值为EPIPE;因此可以写数据时write的返回值和错误码来判断对方连接是否已经断开了。

(2)读数据判断返回值

    如果当前是默认的阻塞模式读取,那么此时read读取返回的长度为0,错误码也是为0,其实表示读取成功。这里需要注意read 和recv接口的默认返回值是不一样的,使用recv接口也会返回EPIPE错误码。

client_tcp.c


  
  1. /************************************************************
  2. *Copyright (C),lcb0281at163.com lcb0281atgmail.com
  3. *FileName: 01_client_tcp.c
  4. *BlogAddr: caibiao-lee.blog.csdn.net
  5. *Description: TCP 客户端收发数据 
  6. *Date:     2020-01-04
  7. *Author:   Caibiao Lee
  8. *Version:  V1.0
  9. *Others:
  10.     通过read write 函数的返回值和错误码判断对方连接是否已经断开
  11. *History:
  12. ***********************************************************/
  13. #include <sys/uio.h>
  14. #include <string.h>
  15. #include <stdlib.h>
  16. #include <stdio.h>
  17. #include <stdlib.h>
  18. #include <strings.h>
  19. #include <sys/types.h>
  20. #include <sys/socket.h>
  21. #include <unistd.h>
  22. #include <arpa/inet.h>
  23. #include <signal.h>
  24. #include <errno.h>
  25. #include <netinet/ip.h>
  26. #include <netinet/tcp.h>
  27. #define SERVER_IP_ADDR        "192.168.1.111"
  28. #define PORT 8888    /* 侦听端口地址 */
  29. void sig_proccess(int signo)
  30. {
  31.      printf( "Catch a exit signal\n");
  32.      exit( 0);    
  33. }
  34. void sig_pipe(int sign)
  35. {
  36.      printf( "Catch a SIGPIPE signal\n");
  37.     
  38.      /* 释放资源 */    
  39. }
  40. void process_conn_client(int s32SocketFd)
  41. {
  42.      int size = 0;
  43.      char buffer[ 1024] = { 0};
  44.      char *sendData = "I am client";
  45.     
  46.      for(;;)
  47.     {
  48.         size = write(s32SocketFd, sendData, strlen(sendData)+ 1);    
  49.          if(size!= strlen(sendData)+ 1)
  50.         {
  51.              printf( "write data error size=%d errno=%d \n",size,errno);
  52.              //return ;
  53.         }
  54.         
  55.         size = read(s32SocketFd, buffer, 1024);
  56.          if(size<= 0)
  57.         {
  58.              printf( "read data error size=%d errno=%d \n",size,errno);
  59.              //return ;                
  60.         } else
  61.         {
  62.              printf( "recv Data: %s\n",buffer);
  63.         }
  64.         sleep( 1);
  65.     }    
  66. }
  67. int main(int argc, char *argv[])
  68. {
  69.      struct sockaddr_in server_addr;    
  70.      int l_s32SocketFd = 0;
  71.         
  72.     signal(SIGINT, sig_proccess);
  73.     signal(SIGPIPE, sig_pipe);
  74.     
  75.      /* 建立一个流式套接字 */
  76.     l_s32SocketFd = socket(AF_INET, SOCK_STREAM, 0);
  77.      if(l_s32SocketFd < 0)
  78.     { /* 出错 */
  79.          printf( "socket error\n");
  80.          return -1;    
  81.     }    
  82.     
  83.      /* 设置服务器地址 */
  84.     bzero(&server_addr, sizeof(server_addr));         /* 清0 */
  85.     server_addr.sin_family = AF_INET;                 /* 协议族 */
  86.     server_addr.sin_addr.s_addr = inet_addr(SERVER_IP_ADDR); /*服务器IP地址*/ /* 本地地址 */
  87.     server_addr.sin_port = htons(PORT);                 /* 服务器端口 */
  88.     
  89.      /* 连接服务器 */
  90.     connect(l_s32SocketFd, (struct sockaddr*)&server_addr, sizeof(struct sockaddr));
  91.     process_conn_client(l_s32SocketFd);     /* 客户端处理过程 */
  92.     
  93.     close(l_s32SocketFd);     /* 关闭连接 */
  94.     
  95.      return 0;
  96. }

server_tcp.c


  
  1. /************************************************************
  2. *Copyright (C),lcb0281at163.com lcb0281atgmail.com
  3. *FileName: 01_server_tcp.c
  4. *BlogAddr: caibiao-lee.blog.csdn.net
  5. *Description: TCP 客户端收发数据 
  6. *Date:     2020-01-04
  7. *Author:   Caibiao Lee
  8. *Version:  V1.0
  9. *Others:
  10.     通过read write 函数的返回值和错误码判断对方连接是否已经断开
  11. *History:
  12. ***********************************************************/
  13. #include <sys/uio.h>
  14. #include <string.h>
  15. #include <stdlib.h>
  16. #include <stdio.h>
  17. #include <stdlib.h>
  18. #include <strings.h>
  19. #include <sys/types.h>
  20. #include <sys/socket.h>
  21. #include <unistd.h>
  22. #include <arpa/inet.h>
  23. #include <signal.h>
  24. #include <errno.h>
  25. #include <netinet/ip.h>
  26. #include <netinet/tcp.h>
  27. #define SERVER_IP_ADDR        "192.168.1.111"
  28. #define PORT 8888        /* 侦听端口地址 */
  29. #define BACKLOG 2        /* 侦听队列长度 */
  30. void sig_proccess(int signo)
  31. {
  32.      printf( "Catch a exit signal\n");
  33.      exit( 0);    
  34. }
  35. void sig_pipe(int sign)
  36. {
  37.      printf( "Catch a SIGPIPE signal\n");
  38.     
  39.      /* 释放资源 */    
  40. }
  41. /* 服务器对客户端的处理 */
  42. void process_conn_server(int s32SocketFd)
  43. {
  44.      int size = 0;
  45.      char buffer[ 1024];     /* 数据的缓冲区 */
  46.     
  47.      for(;;)
  48.     {    
  49.          /* 从套接字中读取数据放到缓冲区buffer中 */
  50.         size = read(s32SocketFd, buffer, 1024);    
  51.          if(size== 0)
  52.         { /* 没有数据 */
  53.              printf( "read size = %d, error %d \n",size,errno);
  54.              //return;    
  55.         } else if(size< 0)
  56.         {
  57.              printf( "read size = %d, error %d \n",size,errno);
  58.              //return ;
  59.         } else 
  60.         {
  61.              printf( "recv data:%s \n",buffer);
  62.             
  63.         }
  64.          memset(buffer, 0, sizeof(buffer));    
  65.          /* 构建响应字符,为接收到客户端字节的数量 */
  66.          strcpy(buffer, "I am server");
  67.         size = write(s32SocketFd, buffer, strlen(buffer)+ 1); /* 发给客户端 */
  68.          if(( strlen(buffer)+ 1)==size)
  69.         {
  70.         } else
  71.         {
  72.              printf( "write data error size = %d, errno=%d\n",size,errno);
  73.          //return ;
  74.         }
  75.         sleep( 1);
  76.     }    
  77. }
  78. int main(int argc, char *argv[])
  79. {
  80.      int l_s32ServerFd = -1;
  81.      int l_s32ClientrFd = -1;        
  82.      struct sockaddr_in server_addr; /* 服务器地址结构 */
  83.      struct sockaddr_in client_addr;     /* 客户端地址结构 */
  84.      int l_s32Ret = 0;     /* 返回值 */
  85.      pid_t pid;     /* 分叉的进行id */
  86.     
  87.     signal(SIGINT, sig_proccess);
  88.     signal(SIGPIPE, sig_pipe);
  89.     
  90.     
  91.      /* 建立一个流式套接字 */
  92.     l_s32ServerFd = socket(AF_INET, SOCK_STREAM, 0);
  93.      if(l_s32ServerFd < 0)
  94.     { /* 出错 */
  95.          printf( "socket error\n");
  96.          return -1;    
  97.     }    
  98.     
  99.      /* 设置服务器地址 */
  100.     bzero(&server_addr, sizeof(server_addr));     /* 清0 */
  101.     server_addr.sin_family = AF_INET;             /* 协议族 */
  102.     server_addr.sin_addr.s_addr = inet_addr(SERVER_IP_ADDR); /*服务器IP地址*/
  103.     server_addr.sin_port = htons(PORT);             /* 服务器端口 */
  104.     
  105.     
  106.     /*设置IP地址可以重复绑定*/
  107.      int l_s32UseAddr = 1;
  108.     if(setsockopt(l_s32ServerFd, SOL_SOCKET, SO_REUSEADDR, &l_s32UseAddr, sizeof( int)) < 0)
  109.     {
  110.         printf( "%s %d\tsetsockopt error! Error code: %d,Error message: %s\n"
  111.             __FUNCTION__, __LINE__, errno, strerror(errno));
  112.         return -2;
  113.     }
  114.      /* 绑定地址结构到套接字描述符 */
  115.     l_s32Ret = bind(l_s32ServerFd, (struct sockaddr*)&server_addr, sizeof(server_addr));
  116.      if(l_s32Ret < 0)
  117.     { /* 出错 */
  118.          printf( "bind error\n");
  119.          return -1;    
  120.     }
  121.     
  122.      /* 设置侦听 */
  123.     l_s32Ret = listen(l_s32ServerFd, BACKLOG);
  124.      if(l_s32Ret < 0)
  125.     { /* 出错 */
  126.          printf( "listen error\n");
  127.          return -1;    
  128.     }
  129.     
  130.      /* 主循环过程 */
  131.      for(;;)
  132.     {
  133.          int addrlen = sizeof(struct sockaddr);
  134.          /* 接收客户端连接 */
  135.         l_s32ClientrFd = accept(l_s32ServerFd, (struct sockaddr*)&client_addr, &addrlen);
  136.          if(l_s32ClientrFd < 0)
  137.         {     /* 出错 */
  138.              continue;     /* 结束本次循环 */
  139.         }    
  140.         
  141.          /* 建立一个新的进程处理到来的连接 */
  142.         pid = fork();         /* 分叉进程 */
  143.          if( pid == 0 )
  144.         {         /* 子进程中 */
  145.             close(l_s32ServerFd);         /* 在子进程中关闭服务器的侦听 */
  146.             process_conn_server(l_s32ClientrFd); /* 处理连接 */
  147.         } else
  148.         {
  149.             close(l_s32ClientrFd);         /* 在父进程中关闭客户端的连接 */
  150.         }
  151.     }
  152. }

(二)通过select系统函数判断

    select实际是IO复用的一个接口,它可以同时检测多个连接是否有数据可读写操作,并且可以设置检测的超时时间。
    在点对点的连接中如果select超时,它返回值为0;

  • 当出现异常的时候,返回-1,如果对方断开可能收到104的错误码,也就是ECONNRESET,表示连接被重置
  • 当select返回1,表示正常,如果read此时返回的值为0,表示对方连接已经断开。

  
  1. /******************************************************** 
  2. Function:     process_conn_server    
  3. Description: 服务器对客户端的处理
  4. Input:    s32SocketFd :服务端接收到客户端连接的ID;
  5. OutPut: none
  6. Return: 0: success,none 0:error
  7. Others: 通过select判断客户端的连接状态
  8. Author: Caibiao Lee
  9. Date:    2020-01-04
  10. *********************************************************/
  11. void process_conn_server(int s32SocketFd)
  12. {
  13.      int size = 0;
  14.      int l_s32Ret = 0;
  15.      char buffer[ 1024];     /* 数据的缓冲区 */
  16.     fd_set l_stReadfd;
  17.     struct timeval l_stTimeout={ 0};
  18.     
  19.      for(;;)
  20.     {    
  21.         l_stTimeout.tv_sec= 0;
  22.         l_stTimeout.tv_usec= 10000;
  23.         FD_ZERO(&l_stReadfd);
  24.         FD_SET(s32SocketFd ,&l_stReadfd);
  25.         l_s32Ret = select(s32SocketFd+ 1, &l_stReadfd, NULL, NULL, &l_stTimeout);
  26.         if (l_s32Ret<= 0)
  27.         {
  28.              printf( "select error l_s32Ret=%d errno=%d\n",l_s32Ret,errno);
  29.             usleep( 100000);
  30.         }
  31.         else if(FD_ISSET(s32SocketFd,&l_stReadfd))
  32.         {
  33.              printf( "l_s32Ret = %d \n",l_s32Ret);
  34.              /* 从套接字中读取数据放到缓冲区buffer中 */
  35.             size = read(s32SocketFd, buffer, 1024);    
  36.              if(size== 0)
  37.             { /* 没有数据 */
  38.                  printf( "read size = %d, error %d \n",size,errno);
  39.              //return;    
  40.             } else if(size< 0)
  41.             {
  42.                  printf( "read size = %d, error %d \n",size,errno);
  43.              //return ;
  44.             } else 
  45.             {
  46.                  printf( "recv data:%s \n",buffer);
  47.             }
  48.         }
  49.          memset(buffer, 0, sizeof(buffer));    
  50.          /* 构建响应字符,为接收到客户端字节的数量 */
  51.          strcpy(buffer, "I am server");
  52.         size = write(s32SocketFd, buffer, strlen(buffer)+ 1); /* 发给客户端 */
  53.          if(( strlen(buffer)+ 1)==size)
  54.         {
  55.         } else
  56.         {
  57.              printf( "write data error size = %d, errno=%d\n",size,errno);
  58.          //return ;
  59.         }
  60.         sleep( 1);
  61.     }    
  62. }

(三)通过TCP_INFO套接字选项判断

    通过getsockopt函数可以获取TCP连接的连接状态,当状态为ESTABLISHED的时候表示该连接正常。TCP的其它状态还有:

  • CLOSED:表示初始状态。对服务端和C客户端双方都一样。
  • LISTEN:表示监听状态。服务端调用了listen函数,可以开始accept连接了。
  • SYN_SENT:表示客户端已经发送了SYN报文。当客户端调用connect函数发起连接时,首先发SYN给服务端,然后自己进入SYN_SENT状态,并等待服务端发送ACK+SYN。
  • SYN_RCVD:表示服务端收到客户端发送SYN报文。服务端收到这个报文后,进入SYN_RCVD状态,然后发送ACK+SYN给客户端。
  • ESTABLISHED:表示连接已经建立成功了。服务端发送完ACK+SYN后进入该状态,客户端收到ACK后也进入该状态。
  • FIN_WAIT_1:表示主动关闭连接。无论哪方调用close函数发送FIN报文都会进入这个这个状态。
  • FIN_WAIT_2:表示被动关闭方同意关闭连接。主动关闭连接方收到被动关闭方返回的ACK后,会进入该状态。
  • TIME_WAIT:表示收到对方的FIN报文并发送了ACK报文,就等2MSL后即可回到CLOSED状态了。如果FIN_WAIT_1状态下,收到对方同时带FIN标志和ACK标志的报文时,可以直接进入TIME_WAIT状态,而无须经过FIN_WAIT_2状态。
  • CLOSING:表示双方同时关闭连接。如果双方几乎同时调用close函数,那么会出现双方同时发送FIN报文的情况,此时就会出现CLOSING状态,表示双方都在关闭连接。
  • CLOSE_WAIT:表示被动关闭方等待关闭。当收到对方调用close函数发送的FIN报文时,回应对方ACK报文,此时进入CLOSE_WAIT状态。
  • LAST_ACK:表示被动关闭方发送FIN报文后,等待对方的ACK报文状态,当收到ACK后进入CLOSED状态。

功能代码如下:


  
  1. /******************************************************** 
  2. Function:     check_tcp_alive    
  3. Description: 通过TCP_INFO查询网络状态
  4. Input:    s32SocketFd :服务端接收到客户端连接的ID;
  5. OutPut: none
  6. Return: 0: success,none 0:error
  7. Others: 
  8. Author: Caibiao Lee
  9. Date:    2020-01-04
  10. *********************************************************/
  11. int check_tcp_alive(int s32SocketFd)
  12. {
  13.      while( 1)
  14.     {
  15.          printf( "alive  s32SocketFd = %d \n",s32SocketFd);
  16.          if(s32SocketFd> 0)
  17.         {
  18.              struct tcp_info info;
  19.              int len = sizeof(info);
  20.             getsockopt(s32SocketFd, IPPROTO_TCP, TCP_INFO, &info, ( socklen_t *)&len);
  21.             
  22.              printf( "info.tcpi_state = %d\n",info.tcpi_state);
  23.              if(info.tcpi_state == TCP_ESTABLISHED)
  24.             {
  25.                  printf( "connect ok \r\n");
  26.                  //return 0;
  27.             }
  28.              else
  29.             {
  30.                  printf( "connect error\r\n");
  31.                  //return -1;
  32.             }
  33.         }
  34.         sleep( 1);
  35.          printf( "\n\n");
  36.     }
  37. }

(四)通过SO_KEEPALIVE套接字选项判断

    选项SO_KEEPALIVE用于设置TCP连接的保持,当设置此项后,连接会测试连接的状态。这个选项用于可能长时间没有数据交流的连接,通常在服务器端进行设置。
    当设置SO_KEEPALIVE选项后,如果在两个小时内没有数据通信时,TCP会自动发送一个活动探测数据报文,对方必须对此进行响应,通常有如下3种情况。

  1. TCP的连接正常,发送一个ACK响应,这个过程应用层是不知道的。再过两个小时,又会再发送一个。
  2. 对方发送RST响应,对方在2个小时内进行了重启或者崩溃。之前的连接己经失效,套接字收到一个ECONNRESET错误,之前的套接字关闭。
  3. 如果对方没有任何响应,则本机会发送另外8个活动探测报文,时间的间隔为75s,当第一个活动报文发送11分15秒后仍然没有收到对方的任何响应,则放弃探测,套接字错误类型设置为ETIMEOUT,并关闭套接字连接。如果收到一个ICMP控制报文响应,此时套接字也关闭,这种情况通常收到的是一个主机不可达的ICMP报文,此时套接字错误类型设置为EHOSTUNREACH,并关闭套接字连接。

    SO_KEEPALIVE的使用场景主要是在可能发送长时间无数据响应的TCP连接,例如Telnet会话,经常会出现打开一个telnet客户端后,长时间不用的情况,这需要服务器或 者客户端有一个探测机制知道对方是否仍然活动。根据探测结果服务器会释放己经失效的客户端,保证服务器资源的有效性,例如有的telnet客户端没有按照正常步骤进行关闭。

    网上有不少资料介绍不推荐使用SO_KEEPALIVE来判断网络连接是否断开,具体原因没有去追踪,这里不再介绍它的使用。

(五)通过SO_RCVTIMEO/SO_SNDTIMEO判断

    这个是通过套接字的SO_RCVTIMEO、SO_SNDTIMEO来设置收发数据超时。对于前面的前面的几种判断方式,都是基于对方正常网络断开后,主机才能够正常的判断到网络状态。如果连接的某一方突然断电,主机并不能知道对方设备突然断电,通过TCP_INFO查询到的也是网络正常,但实际情况是这是网络连接已经断开了。
    这时,可以使用收发数据超时来判断:
    如果设置的时间没有收到数据,read时会返回-1,同时有错误码EAGAIN产生,这时是可以判断出对连接已经断开了。
    这种方式的确定就是,如果设定的一段时间没有收发数据,就会被判断为超时断开连接。


  
  1. /******************************************************** 
  2. Function:     process_conn_server    
  3. Description: 通过设置收发操作判断对方连接已经断开了
  4. Input:    s32SocketFd :服务端接收到客户端连接的ID;
  5. OutPut: none
  6. Return: 0: success,none 0:error
  7. Others: 
  8. Author: Caibiao Lee
  9. Date:    2020-01-04
  10. *********************************************************/
  11. void process_conn_server(int s32SocketFd)
  12. {
  13.      int size = 0;
  14.      char buffer[ 1024];     /* 数据的缓冲区 */
  15.      int optlen = -1;     /* 整型的选项类型值 */
  16.      int l_s32Ret = 0;
  17.      /* 设置发送和接收超时时间 */
  18.      struct timeval tv;
  19.     tv.tv_sec = 10;     /* 1秒 */
  20.     tv.tv_usec = 200000; /* 200ms */
  21.     optlen = sizeof(tv);
  22.     l_s32Ret = setsockopt(s32SocketFd, SOL_SOCKET, SO_RCVTIMEO, &tv, optlen); /* 设置接收超时时间 */
  23.      if(l_s32Ret == -1){ /* 设置接收超时时间失败 */
  24.          printf( "设置接收超时时间失败\n");            
  25.     }
  26.     
  27.     l_s32Ret = setsockopt(s32SocketFd, SOL_SOCKET, SO_SNDTIMEO, &tv, optlen); /* 设置发送超时时间 */
  28.      if(l_s32Ret == -1){
  29.          printf( "设置发送超时时间失败\n");            
  30.     }
  31.     
  32.      for(;;)
  33.     {    
  34.          /* 从套接字中读取数据放到缓冲区buffer中 */
  35.         size = read(s32SocketFd, buffer, 1024);    
  36.          if(size== 0)
  37.         { /* 没有数据 */
  38.              printf( "read size = %d, error %d \n",size,errno);
  39.              //return;    
  40.         } else if(size< 0)
  41.         {
  42.              printf( "read size = %d, error %d \n",size,errno);
  43.              //return ;
  44.         } else 
  45.         {
  46.              printf( "recv data:%s \n",buffer);
  47.             
  48.         }
  49.          memset(buffer, 0, sizeof(buffer));    
  50.          /* 构建响应字符,为接收到客户端字节的数量 */
  51.          strcpy(buffer, "I am server");
  52.         size = write(s32SocketFd, buffer, strlen(buffer)+ 1); /* 发给客户端 */
  53.          if(( strlen(buffer)+ 1)==size)
  54.         {
  55.         } else
  56.         {
  57.              printf( "write data error size = %d, errno=%d\n",size,errno);
  58.          //return ;
  59.         }
  60.         sleep( 1);
  61.     }    
  62. }  

(六)自定义通信心跳判断

    在一些比较重要的命令收发链接中,一般是客户端和服务端会建立心跳机制,心跳时间间隔根据不同的业务需求而不同。当约定的时间段内没有收到心跳数据包,就可以判断对方是否已经断开了连接。
    这种方式非常简单,对于嵌入式设备而言,主要的缺点是心跳会耗费流量,同时会增加一点点系统负载,并且不适合并发连接的情况。

    以上就是现在比较常用的判断网络连接的方法。 如有错误,欢迎指出!

 

 

 

 

 


转载:https://blog.csdn.net/li_wen01/article/details/103923339
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场