EPOLL 判断客户端连接关闭

本文介绍了三种通过EPOLL判断TCP客户端连接关闭的方法:1) 当TCP的recv函数返回0时;2) 注册EPOLLERR事件并接收关闭信号;3) 当recv/send返回-1且错误不为EWOULDBLOCK或EINTR时主动关闭连接。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1. TCP recv返回0, 说明对方关闭

2. 注册EPOLLERR, 收到事件是关闭

3. recv/send 返回-1时, 如果错误不是EWOULDBLOCK或者EINTR, 也主动关闭连接。 

#include <sys/epoll.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>


volatile int exitNow=0;


int main()
{
	int epfd;
	int lisSocket;
	char simpleGlobBuffer[1000];
	struct epoll_event epEvent,recEvent[10];


	if( (epfd=epoll_create(10))==-1 )
	{
		return 1;
	}


	if( (lisSocket=socket(AF_INET,SOCK_STREAM,0))==-1 )
	{
		return 2;
	}


	int flag;


	if(fcntl(lisSocket,F_GETFL,&flag)==-1)
	{
		return 6;
	}


	flag|=O_NONBLOCK;


	if(fcntl(lisSocket,F_SETFL,flag)==-1)
	{
		return 7;
	}


	sockaddr_in lisAddr;


	lisAddr.sin_family=AF_INET;
	lisAddr.sin_port=htons(12000);
	lisAddr.sin_addr.s_addr=htonl(INADDR_ANY);


	int reUseOn=1;


	if(setsockopt(lisSocket,SOL_SOCKET,SO_REUSEADDR,&reUseOn,sizeof(int))==-1)
	{
		return 3;
	}


	if(bind(lisSocket,(struct sockaddr*)&lisAddr,sizeof(lisAddr))==-1)
	{
		return 4;
	}


	if(listen(lisSocket,SOMAXCONN)==-1)
	{
		return 5;
	}


	epEvent.data.fd=lisSocket;
	epEvent.events=EPOLLIN|EPOLLET|EPOLLERR;


	if(epoll_ctl(epfd,EPOLL_CTL_ADD,lisSocket,&epEvent)==-1)
	{
		return 20;
	}


	sockaddr_in clientAddr;
	socklen_t addrLen=sizeof(clientAddr);


	while(!exitNow)
	{
		int nRec=epoll_wait(epfd,recEvent,10,-1);


		if(nRec==-1)
		{
			if(errno==EINTR)
			{
				continue;
			}


			return 8;
		}
		else
		{
			for(int i=0;i<nRec;++i)
			{
				if(recEvent[i].data.fd==lisSocket)
				{
					if(recEvent[i].events&EPOLLERR)
					{
						return 10;
					}


					int clifd;


					while( true )
					{
						clifd=accept(lisSocket,(struct sockaddr*)&clientAddr,&addrLen);


						if(clifd==-1)
						{
							if(errno==EAGAIN)
							{
								break;
							}
							else if(errno==EINTR)
							{
								continue;
							}
							
							return 10;
						}


						epEvent.data.fd=clifd;
						epEvent.events=EPOLLIN|EPOLLET|EPOLLERR|EPOLLRDHUP;
						
						if(fcntl(clifd,F_GETFL,&flag)==-1)
						{
							return 13;
						}


						flag|=O_NONBLOCK;


						if(fcntl(clifd,F_SETFL,flag)==-1)
						{
							return 14;
						}


						if(epoll_ctl(epfd,EPOLL_CTL_ADD,clifd,&epEvent)==-1)
						{
							return 9;
						}
					}
				}
				else
				{
					if(recEvent[i].events&EPOLLERR || recEvent[i].events&EPOLLRDHUP)
					{
						epoll_ctl(epfd,EPOLL_CTL_DEL,recEvent[i].data.fd,NULL);
						close(recEvent[i].data.fd);
						printf("client close , because of err \n");
						continue;
					}
					
					int nRecv;


					while(true)
					{
						nRecv=recv(recEvent[i].data.fd,simpleGlobBuffer,1000,0);
						
						if(nRecv==0)
						{
							epoll_ctl(epfd,EPOLL_CTL_DEL,recEvent[i].data.fd,NULL);
							close(recEvent[i].data.fd);
							printf("client close , normal close \n");
							break;
						}
						else if(nRecv==-1)
						{
							if(errno==EAGAIN)
							{
								break;
							}
							else if(errno==EINTR)
							{
								continue;
							}
							else
							{
								return 12;
							}
						}


						if(write(STDOUT_FILENO,simpleGlobBuffer,nRecv)!=nRecv)
						{
							return 11;
						}
					}
				}
			}
		}
	}


	close(lisSocket);
	
	return 0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值