pthread_create仅在套接字上有数据时才可用

时间:2014-04-07 12:52:00

标签: c sockets select-function

我有服务器正在向客户端发送udp数据报,并在数据包丢失时从客户端接收NACK数据报。我想创建将处理每个NACK数据包的线程,但我想只有在我从客户端接收内容时才创建线程。为此,我想使用select,如果套接字上有东西,则创建新线程并调用recvfrom fucntion。我定义了timeval结构并用0填充它,因为我不想等待,我想在服务器发送时选择“侦听套接字”,但是选择总是返回0 ....有没有任何解决方案可能与不同的想法如何发送和接收并行?

int main (void) {

    struct timeval tv;
    tv.tv_sec = 0;
    tv.tv_usec = 0;

    fd_set readfds;
    FD_ZERO(&readfds);

    // Create the socket
    if((sd = socket (AF_INET, SOCK_DGRAM, 0)) < 0) {
        printf("Server socket could not be created : %s\n",strerror(errno));
        return 0;
    }


    // Non-blocking socket
    fcntl(sd,F_SETFL,O_NONBLOCK);

    FD_ZERO(&readfds);
    FD_SET(sd,&readfds);

    //Allow multiple applications to receive datagrams that are destined to the same local port number.
    if(setsockopt(sd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)) == -1){ 
        printf("Setsockopt error: %s", strerror(errno));
        return 1;      
    }  

    // Initialize the group sockaddr structure with a multicast address of 226.1.1.1 and port 5555. 
    memset((char *) &multiaddr, 0, sizeof(multiaddr));
    multiaddr.sin_family = AF_INET;
    multiaddr.sin_addr.s_addr = inet_addr(IP_GROUP);
    multiaddr.sin_port = htons(PORT_NUM);  

    // Fill the structure udp_srv with informations like multicast address,port number and socket
    strcpy(udp_srv.peer_address,inet_ntoa(multiaddr.sin_addr));
    udp_srv.peer_port = multiaddr.sin_port;
    udp_srv.udp_sd = sd;

    if((fd = open("text.txt",O_RDONLY , 0777))== -1) {
        printf("Error while opening txt file!\n",strerror(errno));
        return 1;       
    } else {         
        while(1) {            
            sleep(1);    
            if((numRead = read(fd,tmp,512)) != 0) {               
                tmp[numRead]='\0';
                strcpy(pack.buffer,tmp);
                //function for sending data until EOF !     
                sent = data_sent(udp_srv,&pack);      
            } else {               
                sent = last_data_sent(udp_srv,&pack);       
                close(sd);
                exit(1);
            }         
            rv = select(sd + 1, &readfds, NULL, NULL, &tv);          

            if(rv > 0) {

                // Create a new thread and call recvfrom in nack_processing function !
                if(pthread_create(&cln_thread, NULL, nack_processing, (void *) &arg)){
                    printf("Pthrad create errorr\n",strerror(errno));
                    continue;
                }
            }     
        }
    }
    return 0;
}

2 个答案:

答案 0 :(得分:0)

很少有建议。

if(rv == 1)更改为if(rv > 0)并提供一些时间

struct timeval tv;
tv.tv_sec = 5;


fd_set readfds;
rv = select(sd + 1, &readfds, NULL, NULL, &tv);
rv = select(sd + 1, &readfds, (fd_set *)0, (fd_set *)0, &tv);

尝试使用

答案 1 :(得分:0)

至少在Linux下,您需要重新 - 在每次调用之前将struct timeval参数初始化为select(),因为它可能已被之前调用{{{{}}修改1}}。

select()