1、server端的fd不需要设置et模式
我们在创建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事件再写入