选择()在stdin上阻塞,不会超时

时间:2013-04-22 12:41:11

标签: c networking nonblocking

我正在编写一个程序,它接收UDP消息以及从用户那里获取输入,但是我的STDIN仍在阻止select。当我在选择程序运行正常之前我的FD_CLR stdin fd时,表明stdin套接字总是准备好从中读取数据。我尝试引入一个timeval电视来计时,但这似乎也不起作用。我应该在某个地方关闭套接字还是在不在的地方调用FD_CLR? 最终结果应该是非阻塞STDIN,但目前它阻止。 谢谢

int
wait_for_input(){
            fd_set fds;
            int maxfd, sd, err, n;
            struct sockaddr_in addr;
            char stdbuf[BUFLEN];
            unsigned char udpbuf[BUFLEN];

            //memset(stdbuf,0x0,sizeof(stdbuf));
            memset(stdbuf,0x0,sizeof(udpbuf));

            sd = socket(AF_INET, SOCK_DGRAM, 0);

            if(sd<0) {
            printf("Failed to Open UDP socket");
            }

            addr.sin_family = AF_INET;
            addr.sin_addr.s_addr = htonl(INADDR_ANY);
            addr.sin_port = htons(host_list[0]->port);
            err = bind(sd,(struct sockaddr *) &addr,sizeof(addr));

            if(err < 0){
                            printf("ERROR: Cant bind port");

            }

                            struct timeval tv;
            while(1){
                            FD_ZERO(&fds);
                            FD_SET(STDIN_FILENO,&fds);
                            FD_SET(sd,&fds);
                            tv.tv_sec = 1;
                            tv.tv_usec = 0;
                            fflush(stdout);
                            select(sd+1,&fds,NULL,NULL,&tv);

                            // If a UDP message arrives
                            if(FD_ISSET(sd,&fds)){
                                            n = recv(sd,udpbuf,sizeof(udpbuf),0);
                                            unpack(udpbuf);
                                            recompute_my_dv();
                                            fflush(stdout);

                             }
                            //If console data is entered.
                                if(FD_ISSET(STDIN_FILENO, &fds)){
                                            fgets(stdbuf,sizeof(stdbuf),stdin);
                                            parse(stdbuf);
                                            printf("server> ");
                                            fflush(stdout);
                                            FD_CLR(STDIN_FILENO,&fds);


                             }

                     }



 return 0;
 }

1 个答案:

答案 0 :(得分:0)

如果在该套接字的缓冲区上有消息,则FD_ISSET不返回1(或true),如果给定的文件描述符是文件描述符集的一部分,则返回true。这是一个问题。

接下来是你的循环应该开始(while循环)应该在select语句之前开始并循环回来。您应该捕获select返回的值。这是因为select返回在select(读,写或错误)的参数2中指定的缓冲区中的位数。因此,如果您将read指定为缓冲区,则需要读取select返回的多少位。

我认为这是两个主要问题。希望这可以帮助。