【问题标题】:Implementing poll() on a TCP server's read/write在 TCP 服务器的读/写上实现 poll()
【发布时间】:2013-11-02 16:31:47
【问题描述】:

我需要此服务器能够侦听并与客户端建立新连接,同时写入现有连接.. 即。异步非阻塞 i/o。我被告知要使用 poll(),但是在花费大量时间试图掌握套接字编程之后,我仍然不确定如何实现 poll() 函数。

int sockfd; 

int main(int argc, char *argv[])
{
 int newsockfd, portno; 
 socklen_t clilen;  
 char buffer[256];       
 struct sockaddr_in serv_addr, cli_addr;  
 int n;             

 if (argc < 2) {         
     fprintf(stderr,"ERROR, no port provided\n"); 
     exit(1);
 }

 sockfd = socket(AF_INET, SOCK_STREAM, 0);      
 if (sockfd < 0)                    
    error("ERROR opening socket");          

 bzero((char *) &serv_addr, sizeof(serv_addr));     

 portno = atoi(argv[1]);            
 serv_addr.sin_family = AF_INET;        
 serv_addr.sin_addr.s_addr = INADDR_ANY;    
 serv_addr.sin_port = htons(portno);    

 if (bind(sockfd, (struct sockaddr *) &serv_addr, 
          sizeof(serv_addr)) < 0) 
          error("ERROR on binding");        
 listen(sockfd,5);                  

 clilen = sizeof(cli_addr);     

 while(1){          
     newsockfd = accept(sockfd, 
                 (struct sockaddr *) &cli_addr, 
                 &clilen);              
     if (newsockfd < 0)                 
          error("ERROR on accept");

     // READ READ READ READ READ READ READ READ READ READ READ READ READ READ READ READ 
     bzero(buffer,256);
     n = read(newsockfd,buffer,255);    


     if (n < 0) error("ERROR reading from socket");
     printf("Here is the message: %s\n",buffer);

     // WRITE WRITE WRITE WRITE WRITE WRITE WRITE WRITE WRITE WRITE WRITE WRITE WRITE 
     n = write(newsockfd,"I got your message",18); 
     if (n < 0) error("ERROR writing to socket");
     close(newsockfd);  
 }

 return 0; 

}

我的理解是我需要构建这样的东西:

//  Set up array of file descriptors for polling
struct pollfd ufds[2];
ufds[0].fd = sockfd;    
ufds[0].events = POLLIN; 

ufds[1].fd = newsockfd;
ufds[1].events = POLLOUT;

并使用 poll(ufds,2,2000);在循环内检查 sockfd 或 newsockfd 是否有任何活动,在这种情况下,我使用适当的读取或写入。如果有人能给我一些指导,我将非常感激。

【问题讨论】:

  • 你读过“选择”吗?

标签: c sockets client-server polling


【解决方案1】:

内核将填充您的struct pollfd 数组的revents 字段中发生的事件。

来自手册页:

revents 字段是一个输出参数,由内核填充实际发生的事件。 revents 中返回的位可以包括任何在 events 中指定的位,或值 POLLERR、POLLHUP 或 POLLNVAL 之一。 (这三个位在 events 字段中是无意义的,只要对应条件为真,就会在 revents 字段中设置。)

如果您希望收到已接受连接的事件通知,则需要提前预留空间或为每个连接调整 struct pollfd 数组的大小。

您需要一些方法来区分监听套接字。您可以将其存储在数组的索引零中。

int i, n;

n = poll(ufds, num_fds_in_array, timeout_value);

/* errors or timeout? */
if (n < 1)
    ;

for (i = 0; i < num_fds_in_array; i++) {
    /* were there any events for this socket? */
    if (!ufds[i].revents)
        continue;

    /* is it our listening socket? */
    if (!i) {
         if (ufds[0].revents & POLLIN)
             /* call accept() and add the new socket to ufds */
         else
             /* error */

         continue;
    }

    /* is there incoming data on the socket? */
    if (ufds[i].revents & POLLIN)
        /* call recv() on the socket and decide what to do from there */
}

POLLOUT 标志用于在套接字上发送数据不会阻塞调用者时发出信号。

对于非阻塞 I/O,我会使用更强大的 API,因为它需要更多的簿记才能可靠地执行。请参阅下一段。

不幸的是,在使用poll 时,没有空间让每个连接的辅助数据存储状态。根据您的平台,有可用的替代方案,例如。 G。用于 Linux 的 epoll、用于 *BSD 的 kqueue 以及一些用于 Windows 的选项。如果要将poll 与上下文数据一起使用,则必须使用可以使用文件描述符或数组索引进行搜索的数据结构。

【讨论】:

  • 如果您能展示如何将此代码应用于 OPs 代码,我将非常高兴。我有完全相同的问题,我有正确阻止的 read() 命令,但我想要连接超时。我尝试了很多东西,但它不起作用。我不明白你的代码......什么是ufds?是newsockfd吗??
【解决方案2】:

你为什么不使用 libevent?它完全异步且非阻塞。 http://libevent.org/

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2021-07-27
    • 1970-01-01
    • 2011-11-01
    • 1970-01-01
    • 2012-06-15
    • 1970-01-01
    • 2012-09-22
    • 1970-01-01
    相关资源
    最近更新 更多