多路复用并发模型  -- 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模型的更多相关文章

  1. Epoll模型详解

    Linux 2.6内核中提高网络I/O性能的新方法-epoll I/O多路复用技术在比较多的TCP网络服务器中有使用,即比较多的用到select函数. 1.为什么select落后    首先,在Lin ...

  2. 转一篇关于epoll模型的博文

    以前就看过这篇关于epoll文章,现在又翻出来看了一下,很久不看的知识真是容易忘啊. 原文出处: http://blog.163.com/huchengsz@126/blog/static/73483 ...

  3. 【转】select和epoll模型的差异

    http://www.cppblog.com/converse/archive/2008/10/12/63836.html epoll为什么这么快 epoll是多路复用IO(I/O Multiplex ...

  4. linux epoll模型

    原文:http://yjtjh.blog.51cto.com/1060831/294119 Linux I/O多路复用技术在比较多的TCP网络服务器中有使用,即比较多的用到select函数.Linux ...

  5. Linux网络服务器epoll模型的socket通讯的实现(一)

    准备写一个网络游戏的服务器的通讯模块,参考网上看到的一些代码,在linux下面实现一个多线程的epoll模型的socket通讯的代码,以下是第一部分多线程的切换代码: 1 #include <s ...

  6. (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 ...

  7. nginx中的epoll模型

    要了解epoll模型,就要一个一个知识点由浅至深地去探索. 1.IO复用技术 IO流请求操作系统内核,有串行处理和并行处理两种概念. 串行处理是前面一个操作处理地时候,后面的所有操作都需要等待.因此, ...

  8. select 和epoll模型区别

    1.select 和epoll模型区别 1.1.网络IO模型概述 通常来说,网络IO可以抽象成用户态和内核态之间的数据交换.一次网络数据读取操作(read),可以拆分成两个步骤:1)网卡驱动等待数据准 ...

  9. Epoll模型

    Epoll模型 相比于select,epoll最大的好处在于它不会随着监听fd数目的增长而降低效率.因为在内核中的select实现中,它是采用轮询来处理的,轮询的fd数目越多,自然耗时越多.并且,在l ...

随机推荐

  1. 关于字符串的简单dp

    看这道题题目叫做魔族密码多新奇的名字点开是道字符串的dp,思考然后想出lis其实但字符串之间的比对只有循环然后其实循环爆不了,太懒点开了题解发现有人使用stl——cstring的函数了方便多了,借鉴一 ...

  2. TFA(Trace File Analyzer)的安装与使用(ORACLE版本12C)

    TFA是Oracle从11.2版本开始推出的一种类似diagcollection的一个oracle 集群日志收集器,而且TFA比diagcollection集中和自动化的诊断信息收集能力更强大.TFA ...

  3. nginx最基本操作

    1.安装 yum install nginx 2.查看配置位置 nginx -t 3.查看nginx.conf,找到默认html配置路径 vi /etc/nginx/nginx.conf cd /us ...

  4. jQuery -- 监听input、textarea输入框值变化

    $('textarea').bind('input propertychange', function(){ if($(".textareachange").val() != &q ...

  5. MovieLens电影数据分析

    下载数据包 链接:https://grouplens.org/datasets/movielens/1m/ 解压: 四个文件分别是数据介绍,电影数据表,电影评分表,用户表 进行电影数据分析 进入ipy ...

  6. 网页制作中规范使用DIV+CSS命名规则,可以改善优化功效特别是团队合作时候可以提供合作制作效率,具体DIV CSS命名规则CSS命名大全内容如下:

    页头:header  如:#header{属性:属性值;}或.header{属性:属性值;},也许你需要了解class与id区别及用法登录条:loginBar         标志:logo      ...

  7. 10.5-uC/OS-III内部任务(时基任务OS-TickTask())

    几乎所有的实时系统都需要有一个能提供周期性时间的时间源,叫做时基周期或系统周期. uC/OS-III的时基周期处理程序封装在OS_TICK.C文件中. OS_TickTask()任务被uC/OS-II ...

  8. SSAS下玩转PowerShell

     操作SSAS数据库的方法有非常多,是否有一种能够方法能够通过脚本自己主动去做这些事呢,比方处理分区,创建备份以及监视SSAS的执行状况.   原文地址: http://www.mssqltips ...

  9. Python-Log-note.md

    #LOG - https://www.cnblogs.com/yyds/p/6901864.html - logging - logging模块提供模块级别的函数记录日志 - 包括四大组件 ## 1. ...

  10. Java 基础 面向对象之关键字内部类代码块修饰符

    final final概念 继承的出现提高了代码的复用性,并方便开发.但随之也有问题,有些类在描述完之后,不想被继承,或者有些类中的部分方法功能是固定的,不想让子类重写.可是当子类继承了这些特殊类之后 ...