2016-08-29 8 views
4

UDP-сервер принимает пакеты с системным вызовом select. И я хочу получить последний пакет от каждого UDP-клиента. (Я также хочу слушать несколько клиентских пакетов UDP).Получить последний пакет на сервере udp - C

Коды моей простой UDP-сервер:

int main(void) { 
    int fd; 
    int port = 5678; 
    char buffer[1024]; 
    fd_set readfs; 
    socklen_t client_length; 
    struct timeval timeout_interval; 
    struct sockaddr_in6 server_addr; 
    struct sockaddr_in6 client_addr; 
    int result; 
    int recv; 
    char client_addr_ipv6[100]; 

    fd = socket(PF_INET6, SOCK_DGRAM, 0); 
    printf(" \e[1m \e[34m ---------------------------------------- \n-------------------- UDP SERVER --------------------\n \e[39m \e[0m \n"); 
    printf("Process: \e[34m %d \e[49m Port ..\n", port); 

    if (fd < 0) { 
     printf("ERR: fd < 0"); 
    } else { 
     memset(&server_addr, 0, sizeof(server_addr)); 
     server_addr.sin6_family = AF_INET6; 
     server_addr.sin6_addr = in6addr_any; 
     server_addr.sin6_port = htons(port); 
     memset(&client_addr, 0, sizeof(client_addr)); 
     client_addr.sin6_family = AF_INET6; 
     client_addr.sin6_addr = in6addr_any; 
     client_addr.sin6_port = htons(port); 

     if (bind(fd, (struct sockaddr *) &server_addr, sizeof(server_addr)) 
       >= 0) { 
      printf("\e[1m INFO: \e[0m \e[34m Bind success.. \e[39m\n"); 

     } else { 
      printf("Bind."); 
      return -1; 
     } 
     for (;;) { 
      FD_ZERO(&readfs); 
      FD_SET(fd, &readfs); 
      int max_fd = MAX(0, fd); 
      timeout_interval.tv_sec = 3; 
      timeout_interval.tv_usec = 50000000; 

      result = select(max_fd + 1, &readfs, NULL, NULL, &timeout_interval); 
      //printf("\n %d \t %d \n", result, fd); 
      if (result < 0) { 
       printf("ERR\n"); 
      } else if (result == 0) { 
       printf("\nTimeout\n"); 
      } else { 
       if (FD_ISSET(fd, &readfs)) { 
        client_length = sizeof(client_addr); 
        if ((recv = recvfrom(fd, buffer, sizeof(buffer), 0, 
            (struct sockaddr *) &client_addr, &client_length)) 
          < 0) { 
         printf("Recv-ERR!"); 
         break; 
        } 
        inet_ntop(AF_INET6, &(client_addr.sin6_addr), client_addr_ipv6, 100); 
        //printf("Client IP/Port : %s ",client_addr_ipv6); 

        printf("\n ------------------------------------------ \n"); 
        printf("\e[1m Data: \e[0m \e[32m %.*s \n Client IP/Port : \e[34m %s/%d \n\e[39m", recv, buffer,client_addr_ipv6,ntohs(client_addr.sin6_port)); 
       } 
      } 
     } 
    } 
} 
+1

Предполагая, что вы имеете в виду последний пакет, отправленный другим сервером, вы не можете определить, какой из них последний, поскольку UDP-пакеты неупорядочены. Вам нужно будет поместить некоторые данные в пакет самостоятельно, чтобы определить, какой из них был «последним». –

+0

Я имею в виду последний поступающий пакет от каждого udp-клиента. Итак, как это сделать, не ставя «последнее» определение в пакете. Могу ли я использовать код «это последний пакет от этого udp-клиента»? или как я могу это сделать? – Gkan

+0

Как вы ожидаете узнать, что пакет является «последним»? – dbush

ответ

1

Лучший способ справиться с этим, чтобы отправитель поставить порядковый номер в каждом пакете. Для каждого отправленного пакета порядковый номер увеличивается на 1.

На стороне приемника вы должны отслеживать счетчик последнего полученного вами пакета. Если следующий пакет, в который входит, имеет более высокий счетчик, он является последним. Если он не больше, это более старый пакет, и вы можете обрабатывать его соответствующим образом для своего приложения.

Кроме того, вы должны переименовать вас в recv переменной на что-то вроде recv_len. recv - это имя функции сокета, которая была бы замаскирована этим определением переменной.

+0

Имеют смысл, но самый старый пакет для меня не имеет значения, поэтому я хотел бы обойтись без подсчета или секвенсирования пакетов. Я имею в виду; Самый старый пакет не имеет значения, и я хотел бы использовать в процессе последний. Кроме того, у меня есть некоторый процесс в udp-сервере после получения пакета, тогда буфер прослушивает другие пакеты в одно и то же время. Поэтому, если мой процесс завершится, сервер продолжит, какой из них будет следующим пакетом (из буфера). Я хочу, чтобы, если конец процесса, буфер должен быть ясным (я должен проверить, какой клиент отправляет этот пакет), и он должен быть последним пакетом в буфере от каждого клиента. – Gkan

+0

@ GökhanAyrancıoğlu Невозможно очистить локальный буфер, чтобы получить только последний пакет в буфере, не прочитав их все. Если вы делаете два последовательных чтения, вы не можете сказать, был ли первый прочитанный вами пакет в буфере сам по себе, или если за ним стоит другой. Вам просто нужно продолжать читать их. И если пакеты выходят из строя, в пакете должно быть что-то, что вам скажут. – dbush

+0

Спасибо за ваш комментарий. На самом деле я должен написать размер пакета и формат в том же для каждого пакета. Поэтому я попытаюсь прочитать их все, как вы сказали. Затем я разберу буфер с таким размером, чтобы я мог найти таким образом, который является самым последним. Спасибо за помощь. – Gkan