【linux】epoll----LT工作模式下实现多个客户机与服务机之间的通讯

(注:由于前面写过select和poll函数,在epoll函数中大部分代码会和钱前面两个的函数重复,所以在下面的代码中,不再注释
重复代码,只将与之不同的代码注释,读者可以通过“https://mp.csdn.net/postedit/83215953”或者“https://blog.csdn.net/xing1584114471/article/details/83343021”了解)

头文件以及宏定义:

#include <stdio.h>
#include <stdio.h>
#include <unistd.h>
#include <assert.h>
#include <string.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <sys/epoll.h>

#define MAXFD 10

子函数(将描述符fd添加到事件表上):

void epoll_add(int epfd, int fd)
{
	struct epoll_event ev;
	ev.events = EPOLLIN;
	ev.data.fd = fd;

	if(epoll_ctl(epfd, EPOLL_CTL_ADD, fd, &ev) == -1)
	{
		perror("epoll ctl error!");
	}
}

子函数(从事件表上讲描述符fd删除掉):

void epoll_del(int epfd, int fd)
{
	if(epoll_ctl(epfd, EPOLL_CTL_DEL, fd, NULL) == -1)
	{
		perror("epoll del error!");
	}
}

实现函数:

int main()
{
	//创建套接字
	int sockfd = socket(AF_INET, SOCK_STREAM, 0);
	assert(sockfd != -1);

	struct sockaddr_in saddr, caddr;
	memset(&saddr, 0, sizeof(saddr));
	saddr.sin_family = AF_INET;
	saddr.sin_port = htons(6000);
	saddr.sin_addr.s_addr = inet_addr("127.0.0.1");

	int res = bind(sockfd, (struct sockaddr*)&saddr, sizeof(saddr));
	assert(res != -1);

	listen(sockfd, 5);

	int epfd = epoll_creat(MAXFD);//创建内核时间表
	epoll_add(epfd, sockfd);//将第一个文件描述符sockfd添加到事件表中
	struct epoll_event events[MAXFD];//定义epoll_event类型的数组,该数组用来输出epoll_wait检测到的就绪事件

	while(1)
	{
		int n = epoll_wait(epfd, events, MAXFD, 5000);//用epoll_wait监视内和事件表上文件描述符,返回值n代表在事件表中前n个描述符就绪
		if( n == -1)
		{
			printf("epoll_wait error!\n");
			continue;
		}
		if( n == 0)
		{
			printf("time out!\n");
			continue;
		}
		else
		{
			int i = 0;
			for( ; i < n; i++)//epoll函数只需要遍历内核事件表中前n个文件描述符
			{
				int fd = events[i].data.fd;
				if(event[i].events & EPOLLIN)//再次判断epoll_event型的数组上是否读就绪
				{
					if(fd == sockfd) //判断是否为新客户机链接自己
					{
						int len = sizeof(caddr);
						int c = accept(fd, (struct sockaddr*)&saddr, &len);
						if( c < 0 )
						{
							printf("accept error!\n");
							continue;
						}
						printf("accept = %d\n",c);
						epoll_add(epfd, c);
					}
					else//判断是否为已连接的客户机有数据传来
					{
						char buff[128] = {0};
						int num = recv(fd, buff, 127, 0);
						if( num <= 0)
						{
							printf("one over!\n");
							epoll_del(epfd, fd);
							close(fd);
							continue;
						}
						printf("recv(%d) = %s\n",fd, buff);
						send(fd, "ok", 2, 0);
				     }
				   }
				}
			}
		}
	}
	exit(0);
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值