IO多路复用eopll的练习

IO多路复用eopll的练习

#include <stdio.h>
#include <errno.h>
#include <sys/epoll.h>
#include <ctype.h>
#include <string.h>
#include <sys/types.h>
#include <stdlib.h>
#include <netinet/in.h>
#include <unistd.h>
#include <arpa/inet.h>
int main()
{
	// creat socket
	int lfd = socket(AF_INET, SOCK_STREAM, 0);
	if (lfd < 0) {
		perror("socket error");
		return -1;
	}

	//IO多路复用
	int opt = 1;
	setsockopt(lfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(int));
	
	// bind socket
	struct sockaddr_in serv;
	bzero(&serv, sizeof(serv));
	serv.sin_family = AF_INET;
	serv.sin_port = htons(8888);
	serv.sin_addr.s_addr = htonl(INADDR_ANY);
	int ret = bind(lfd,(const struct  sockaddr*) &serv, sizeof(serv));
	if (ret < 0) {
		perror("bind error");
		return -1;
	}

	//listenle
	listen(lfd, 128);
    
	// 创建一个epoll,得到一个文件描述符,epoll是由红黑树实现的,也可以理解为树的树根,往后往树上添加监听的socket,和删除socket都要通过根节点。
    // 1024最大连接个数
	int epfd = epoll_create(1024);
    // 创建接收节点的events
    struct epoll_event events[1024];
	struct epoll_event ev;
    // epoll_event表明监听事件
	ev.events = EPOLLIN;
    // epoll_event表明监听的socket
	ev.data.fd = lfd;
    // 监听lfd,如果有lfd响应说明有新的客户端连接
    epoll_ctl(epfd, EPOLL_CTL_ADD, lfd, &ev);
    
    
	int n, sockfd, i, cfd, nready;
    char buf[1024];
	
	
	while (1) {
		nready = epoll_wait(epfd, events, 1024, -1);
		if (nready < 0) {
			if (errno == EINTR) {
				continue;
			}
			break;
		}
        // 如果有事件响应有两种情况,一种是有新的客户端连接,一种是有客户端回复,有客户端连接就判断响应的是不是lfd就ok
		for (i = 0; i < nready; i++) {
			sockfd = events[i].data.fd;
			if (sockfd == lfd) {
				cfd = accept(lfd, NULL, NULL);
				ev.events = EPOLLIN;
				ev.data.fd = cfd;
				epoll_ctl(epfd, EPOLL_CTL_ADD, cfd, &ev);
				continue;
			}
			
			n = read(sockfd, buf, sizeof(buf));
			if (n <= 0) {
				close(sockfd);
				epoll_ctl(epfd, EPOLL_CTL_DEL, sockfd, NULL);
				perror("read error");
				continue;
			}
			printf("%dclient:%s\n", sockfd, buf);
			write(sockfd, buf, strlen(buf));
		}
	}

	close(lfd);
	close(epfd);

} 

epoll两种工作模式LT(水平触发),ET(边缘触发)

水平触发(默认模式):如果数据一次性没有读完缓冲区还有可读数据,则epoll_wait还会再次通知。

边缘触发:如果一次性数据没有读完,那么epoll_wait也不会去处理,知道下一次这个客户端再发来请求才会去处理。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值