非阻塞socket与epoll的ET模式结合使用注意事项

当非阻塞socket与epoll的ET模式结合使用进行编程时需要注意的事项:
当客户端与服务端建立连接数据通信完成后,即服务端收到客户端的数据后,再向客户端发送数据,当发送完数据后立马断开连接。客户端epoll_wait和read的处理:

while(1)
{
     nfds = epoll_wait(epfd_rcv, events, EVENTSIZE , -1);// get the sockfd ready
        if (nfds < 0){
                        if(errno == EINTR)
                               continue;
                        else{
                                 fprintf(stderr, "epoll_wait err:%s\n", strerror(errno));
                                 return NULL;
                        }
        }
        for (i=0; i < nfds; i++){
            if(events[i].events & EPOLLIN){
                if ( (sockfd = events[i].data.fd) < 0) {
                    continue;
                }
                num2=1;
                while(num2>0)
                {
                       num2 = read(sockfd, recv_buf, g_receive_package_size - data_buf_r[sockfd]);

                        if (num2 < 0) {        //有对errno的处理
                             if ( (errno != EINTR) && (errno != EAGAIN) ) {
                                  if (close(sockfd) == 0) {
                                     ...........
                                  }
                             }
                      } else if (num2 == 0) {  //无对errno的处理,当num2=0 时系统没有给errno赋新值,而errno是全局变量会保存上一次系统赋给它的值     
                                 if (close(sockfd) == 0) {
                                   .............
                                 }
                     }else{
                            .............
                            ............
                     }
                }
       }
}

因为是非阻塞socket和ET模式,如果不进行循环read的话,服务器发送完数据后立即close掉连接,那么客户端能read到服务器发送来的数据但是捕捉不到服务器close连接的事件,而客户端的tcp/ip协议栈收到了服务器发来的FIN包,但由于应用程序中没有捕获到服务器close连接的事件,也就是没有read返回0,因此没有向服务器发送FIN包,导致客户端大量的CLOSE_WAIT.


http://blog.163.com/wangdong8213@126/blog/static/45852366201282633349139/


你可能感兴趣的:(非阻塞socket与epoll的ET模式结合使用注意事项)