33网络通信之Epoll模型
多路复用并发模型 -- epoll
监控事件 events
EPOLLIN fd可读
EPOLLOUT fd可写
EPOLLPRI fd紧急数据可读
EPOLLERR fd发生错误
EPOLLHUP fd 被挂起
EPOLLONESHOT fd 只监控 1 次,监控完后自动删除
EPOLLLT epoll 工作模式,设置为 水平触发模式
EPOLLET epoll 工作模式,设置为 边缘触发模式
多路复用并发模型 -- epoll
epoll 工作模式
1)水平触发模式 (Level Triggered, 默认值)
事件发生时,应用程序可以不立即处理。
没有做处理,则下次epoll_wait 事件仍然被置位
2)边缘触发模式 (Edge Triggered)
事件发生时,应用程序必须处理
否则,这个事件会被丢弃
epoll工作在ET模式的时候,必须使用非阻塞套接口,以避免由于一个文件句柄的阻塞读/阻塞写操作把处理多个文件描述符的任务饿死
水平触发模式和边缘触发模式
水平触发监控的是状态: 有没有消息可读
边缘触发监控的是变换: 是不是由可读变成不可读,或者由不可读变成可读
#include<stdio.h>
#include<unistd.h>
#include<string.h> #include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h> #include <map> #include<sys/epoll.h>
#include<sys/time.h> #define SRV_PORT 0xabcd
#define DEAL_NUM 2
#define CONN_MAX 10000 int fd;
int nConn = 0;//num of connection
std::map<int, struct socket_in> g_fdmap; void epoll_process(int epfd,struct epoll_events,int cnt)
{
int i, iRet = 0;;
int newfd;
char szBuff[1000];
struct sockaddr_in addr;
socklen_t addrlen = sizeof(addr);
struct epoll_event ect; std::map<int struct socket_in>::iterator it; for (i = 0; i < cnt; ++i)
{
//Accept the message and process it
if (evts[i].data.fd == STDIN_FILENO)
{ }
//new client to connet
else if (evts[i].data.fd == fd)
{
newfd = accept(fd, (struct sockaddr*)&addr, &addrlen);
if (newfd < 0)
{
perror("Fail to accept!");
continue;;
}
if (nConn == CONN_MAX)
{
write(newfd, "Over limit!", 12);
printf("Over connect..\n");
close(newfd);
continue;
}
//normal operation
g_fdmap.insert(std::make_pair(newfd, addr)); //add newfd to epfd for monitor
evt.events = EPOLLIN;
evt.data.fd = newfd;
epoll_ctl(epfd, EPOLL_CTL_ADD, newfd, &evt); //reponse client
write(newfd, "Welcome", 8); printf("\rNew Connect from %s[%d]\n", inet_ntoa(addr.sin_addr), ntohs(addr.sin_port)); }
//recviver client message
else
{
memset(szBuff, 0, 1000); it = g_fdmap.find(evts[i].data.fd);
if (it == g_fdmap.end())
{
printf("\rUnknow client fd: %d\n", evts[i].data.fd);
continue;
} iRet=read(evts.data.fd, szBuff, 1000);
if (iRet < 0)
{
perror("Fail to read!");
continue;
}
else if (iRet == 0)
{
printf("Disconnct from %s[%d]\n", inet_ntoa((it->second()).sin_addr),
ntohs((it->second()).addr.sin_port)); //delete fd from epfd monitor
epoll_ctl(epfd, EPOLL_CTL_DEL,evts[i].data.fd,evts+i); //erase fa from map
g_fdmap.erase(it);
}
//normal operation
else
{
printf("\rRecv from %s[%d]:%s\n", inet_ntoa((it->second()).sin_addr,
ntohs((it->second()).addr.sin_port), szBuff);
} }
} return;
} void StarEpoll()
{
int iRet;
struct sockaddr_in addr;
socklen_t addrlen = sizeof(addr); fd = socket(PF_INET, SOCK_STREAM, 0);
if (fd < 0)
{
perror("Fail to socket!");
return;
} addr.sin_family = AF_INET;
addr.sin_addr.s_addr = htonl(INADDR_ANY);
addr.sin_port = htons(SRV_PORT); iRet = bind(fd, (struct sockaddr*)&addr, addrlen);
if (iRet)
{
perror("Fail to bind!");
close(fd);
return;
} iRet = listen(fd, 100);
if (iRet)
{
perror("Fail to listen!");
close(fd);
return;
} ///////////////init epollfd int epfd = epoll_create(1);
if (epfd < 0)
{
perror("Fail to epoll_create!");
close(fd);
return;
} struct epoll_event evt; //add stdin
evt.events = EPOLLIN;
evt.data.fd =STDIN_FILENO;
epoll_ctl(epfd, EPOLL_CTL_ADD, STDIN_FILENO); //add tcp server fd
evt.events = EPOLLIN;
evt.data.fd = fd;
epoll_ctl(epfd, EPOLL_CTL_ADD, &evt); /////////////////////// struct epoll_event evts[DEAL_NUM]; int cnt;
while (1)
{
cnt = epoll_wait(epfd, evts, DEAL_NUM, -1);
if (cnt < 0)
{
perror("Fail to epoll_wait!");
break;
}
else if (cnt == 0)
{
//timeout
continue;
}
else
{
epoll_process(epfd, evts, cnt);
}
} close(fd);
return;
} int main()
{
StarEpoll(); return 0;
}
33网络通信之Epoll模型的更多相关文章
- Epoll模型详解
Linux 2.6内核中提高网络I/O性能的新方法-epoll I/O多路复用技术在比较多的TCP网络服务器中有使用,即比较多的用到select函数. 1.为什么select落后 首先,在Lin ...
- 转一篇关于epoll模型的博文
以前就看过这篇关于epoll文章,现在又翻出来看了一下,很久不看的知识真是容易忘啊. 原文出处: http://blog.163.com/huchengsz@126/blog/static/73483 ...
- 【转】select和epoll模型的差异
http://www.cppblog.com/converse/archive/2008/10/12/63836.html epoll为什么这么快 epoll是多路复用IO(I/O Multiplex ...
- linux epoll模型
原文:http://yjtjh.blog.51cto.com/1060831/294119 Linux I/O多路复用技术在比较多的TCP网络服务器中有使用,即比较多的用到select函数.Linux ...
- Linux网络服务器epoll模型的socket通讯的实现(一)
准备写一个网络游戏的服务器的通讯模块,参考网上看到的一些代码,在linux下面实现一个多线程的epoll模型的socket通讯的代码,以下是第一部分多线程的切换代码: 1 #include <s ...
- (OK) Linux epoll模型—socket epoll server client chat
http://www.cnblogs.com/venow/archive/2012/11/30/2790031.html http://blog.csdn.net/denkensk/article/d ...
- nginx中的epoll模型
要了解epoll模型,就要一个一个知识点由浅至深地去探索. 1.IO复用技术 IO流请求操作系统内核,有串行处理和并行处理两种概念. 串行处理是前面一个操作处理地时候,后面的所有操作都需要等待.因此, ...
- select 和epoll模型区别
1.select 和epoll模型区别 1.1.网络IO模型概述 通常来说,网络IO可以抽象成用户态和内核态之间的数据交换.一次网络数据读取操作(read),可以拆分成两个步骤:1)网卡驱动等待数据准 ...
- Epoll模型
Epoll模型 相比于select,epoll最大的好处在于它不会随着监听fd数目的增长而降低效率.因为在内核中的select实现中,它是采用轮询来处理的,轮询的fd数目越多,自然耗时越多.并且,在l ...
随机推荐
- Delphi maskedit控件的掩码含义及用法方法
Delphi maskedit控件的掩码含义及用法方法 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 2 ...
- 转:Java 异常结构体系
原文地址:Java 异常结构体系 保存一份资料 前几天在参加网易和360公司的在线考试的时候,都出了一道关于java中异常类的多项选择题.这几天翻看了相关书籍和网上一些资料,结合自己的理解与思考,将自 ...
- SQL GROUP BY对多个字段进行分组
参考链接:http://stackoverflow.com/questions/2421388/using-group-by-on-multiple-columns 在平时的开发任务中我们经常会用到M ...
- 《HTTP - 概述》
晚上听着 木小雅<可能否>,让我这暴躁的一天得以回复一下. 学习是长久之事,不能急躁.应该系统的去学习一些东西. 1:HTTP 最严谨的叫法应该是 超文本转移协议,但大家一般都叫做 超文本 ...
- 【Python全栈-HTML】HTML如何做出分割线效果
参考: https://blog.csdn.net/weixin_39198406/article/details/78827671 一.普通效果 <hr> <hr align=ce ...
- oracle表空间的管理
1.创建表空间 CREATE TABLESPACE TBS_TR_DATA DATAFILE '/oradata/rTBS_TR_DATA_001.dbf' SIZE 64G EXTENT MANAG ...
- IQ调制原理
现代通信中,IQ调制基本上属于是标准配置,因为利用IQ调制可以做出所有的调制方式. 但是IQ调制到底是怎么工作的,为什么需要星座映射,成型滤波又是用来干嘛的.这个呢,讲通信原理的时候倒是都会泛泛的提到 ...
- IntelliJ配置SpringMVC提示“found:java.lang.String required:java.lang.String”
File->Invalid Cache&Restart 据说Android Studio也会出现这个问题,但是没遇到过 重启环境后,找不到Controller的问题也解决了
- int(1)和int(11)是否有区别?
MySQL类型关键字后面的括号内指定整数值的显示宽度(例如,INT(11)).该可选显示宽度规定用于显示宽度小于指定的列宽度的值时从左侧填满宽度.显示宽度并不限制可以在列内保存的值的范围,也不限制超过 ...
- sql语句优化(二)
1.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引. 建立索引好处 : 之前做的一个项目 , 一个查询10w多条的数据 ,需要20s ,后来加 ...