我们在创建socket成功后会有个listenfd,listenfd = socket(AF_INET, SOCK_STREAM, 0)
然后会把这个fd加入epoll wait队列中,网上很多没有经过验证的代码是这样写的:
ev.data.fd = listenfd;
ev.events=EPOLLIN|EPOLLET;
epoll_ctl(epfd,EPOLL_CTL_ADD,listenfd,&ev);
这样会导致服务器在并发处理客户端连接时,丢失部分连接,或者说丢失epoll事件
表现为:采用netstat查看网络,会看到Recv-Q大于0,但是程序执行不到accept代码段。
这个也和具体内核有关系,但是最好代码方面不要这么写。
正确的写法为:ev.events=EPOLLIN; 这里不用指定EPOLLET
补充:EPOLLET,意思是状态翻转的时候才触发事件。在高并发的情况下,如果先后来了两个客户端的连接事件,会出现后者的连接在服务器端的epoll发现不了新连接的情况。但是客户端显示成功,且数据可以正确发送到服务器,在服务器端用netstat可以看到接受队列里面有数据。
2、如果采用et模式,读的时候要读完数据
3、要处理发送,接收函数的返回值和错误码。如:read,recv,send,write,errno
write如果返回-1,可能是写缓冲满,需要等待out事件再写入
本文深入探讨了服务器端socket配置时避免在listenfd上使用ET模式的必要性,解释了该错误配置可能导致的问题,并提供了正确的epoll事件配置方式。同时,文章还强调了在使用ET模式时读取数据必须完全读完的重要性,以及处理socket发送和接收函数返回值和错误码的必要性。
1090

被折叠的 条评论
为什么被折叠?



