14-15 高并发服务器中epoll编程

1. 通过epoll实现高并发服务器模型

LT(level triggered)是缺省的工作方式,并且同时支持block和no-block socket.在这种做法中,内核告诉你一个文件描述符是否就绪了,然后你可以对这个就绪的fd进行IO操作。如果你不作任何操作,内核还是会继续通知你的,所以,这种模式编程出错误可能性要小一点。传统的select/poll都是这种模型的代表.

ET (edge-triggered)是高速工作方式,只支持no-block socket。如果一直不对这个fd作IO操作(从而导致它再次变成未就绪),内核不会发送更多的通知(only once)

#include "socket_includes.h"
#include <unistd.h>
#include <fcntl.h>
#include <sys/epoll.h>
#include <errno.h>

int mz_ipv4_tcp_create_socket(void)
{
	int listenfd, sockfd, opt = 1;
	struct sockaddr_in server, client;
	socklen_t len;
	int timep;
	int ret;

	listenfd = socket(AF_INET, SOCK_STREAM, 0);
	if(listenfd < 0){
		perror("Create socket fail.");
		return -1;
	} 

	if((ret = setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt))) < 0){
		perror("Error, set socket reuse addr failed");  
		return -1;
	}

	bzero(&server, sizeof(server));
	server.sin_family = AF_INET;
	server.sin_port   = htons(SERV_PORT);
	server.sin_addr.s_addr  = htonl(INADDR_ANY);
	
	len = sizeof(struct sockaddr);
	if(bind(listenfd, (struct sockaddr *)&server, len)<0){
			  perror("bind error.");
		return -1;
	}
	  
	listen(listenfd, MAX_LISTEN_QUE);

	return listenfd;
}

int mz_process_data(int sockfd)
{
	int bytes;
	char buf[100];
	char *s = buf;
	char flag = 1;
	int len;

	while(flag){
		bytes = recv(sockfd, s, 5, 0);
		if(bytes < 0){
			if(errno == EAGAIN){
				printf("no data.\n");
				break;
			}
			perror("recv err:");
			return -1;
		}
		if(bytes == 0){
			return -2;
		}

		if(bytes == 5){
			flag = 1;
		}else {
			flag = 0;
		}
		
		s += bytes;
		len += bytes;
		printf("bytes:%d\n", bytes);
		//sleep(2);
	}
	printf("buf:%s\n", buf);
	send(sockfd, buf, len, 0);
	
	return 0;
}

int main(int argc, char *argv[])
{
	int listenfd, sockfd;
	int epollfd, fds;
	struct epoll_event ev, events[MAX_EVENTS];
	int i, rv;
	struct sockaddr_in client;
	int len;

	len = sizeof(struct sockaddr_in);
	epollfd = epoll_create(MAX_EVENTS);
	if(epollfd < 0){
		perror("epoll_create err:");
		return -1;
	}

	listenfd = mz_ipv4_tcp_create_socket();

	fcntl(listenfd, F_SETFL, O_NONBLOCK);

	ev.data.fd = listenfd;
	ev.events = EPOLLIN;
	rv = epoll_ctl(epollfd, EPOLL_CTL_ADD, listenfd, &ev);
	if(rv < 0){
		perror("epoll_ctl err:");
		return -1;
	}

	while(1){
		//time_out
		fds = epoll_wait(epollfd, events, MAX_EVENTS, -1);
		if(fds < 0){
			perror("epoll_wait err:");
			return -1;
		}

		for(i = 0; i < fds; i++){
			if(events[i].data.fd == listenfd){
				sockfd = accept(listenfd, (struct sockaddr *)&client, &len);
				if(sockfd < 0){
					perror("accept err:");
					continue;
				}
				ev.data.fd = sockfd;
				ev.events = EPOLLIN | EPOLLET;
				epoll_ctl(epollfd, EPOLL_CTL_ADD, sockfd, &ev);
				continue;
			}else{
				rv = mz_process_data(events[i].data.fd);
				if(rv == -2){
					epoll_ctl(epollfd, EPOLL_CTL_DEL, events[i].data.fd, &ev);
					close(events[i].data.fd);
					continue;
				}
			}
		}
	}
}

2. 并发编程方法比较:

a. ache模型,Process Per Connection,简称PPC;TPC,Thread Per Connection模型

各做各的事情


b. select / poll 模型之select

能做的事情有限,FD_SETSIZE/1024,也可以修改

效率问题,线性扫描的方法,严重影响;

内核,用户空间内存拷贝


poll除了低一点,其他基本一致


c. epoll:select反过来就是epoll的好处了

能处理的链接数:#cat /proc/sys/fs/file-max //10W

效率问题:epoll只关注活跃的连接

内存拷贝:共享内存




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值