unix网络编程str_cli使用epoll实现

unix环境高级编程中也有这个函数,都是为了讲解IO多路转接。从本质上来看epoll就是一个改善了的select和poll,本质没发生任何变化,对于构建在poll,select和epoll上的框架使用者来说,没什么区别。而对框架设计者来说,使用epoll来替换select或者poll代价也不大。

epoll说明

epoll主要涉及到三个函数,这个要比select和poll复杂,从本质上来说是因为epoll把通知内核我想要观察的描述符和内核通知我哪些描述符发生了变化分成了两个独立的部分,而前述的select和poll都是合二为一的,这也是为什么epoll能做到性能优势。

1.通知内核我想要观察哪些描述符

epoll_create

epollcreate用于创建一个文件描述符,以后所有的操作都与此文件描述符有关。实际上你可以认为这个相当于一个文件,存放了进程要观察的所有文件描述符集合。

原型是

  1. int epoll_create(int size)

调用方式也很简单:

int epfd=epoll_create(10);

size 根据需要调整.

epoll_ctl 用于通知内核添加删除待观察文件描述符

epoll_ctl的使用比较复杂,主要是涉及到epoll_event 的操作.

epoll_event的定义如下:

  1. typedef union epoll_data {
  2. void *ptr;
  3. int fd;
  4. __uint32_t u32;
  5. __uint64_t u64;
  6. } epoll_data_t;
  7. //感兴趣的事件和被触发的事件
  8. struct epoll_event {
  9. __uint32_t events; /* Epoll events */
  10. epoll_data_t data; /* User data variable */
  11. };

epoll_ctl的原型如下:

int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);

  • epfd 就是刚刚用epoll_create返回的描述符
  • op 是对文件描述符集合的操作,主要有增,删,改,没有查是因为这个是使用者自己维护的

    + EPOLL_CTL_ADD 增

    + EPOLL_CTL_MOD 改

    + EPOLL_CTL_DEL 删
  • fd 待增删改的文件描述符
  • event 这个是最复杂的数据结构也是我们关注的核心.这里我们只使用到了文件描述符的操作,其他说真的,我也不晓得用来做什么.

    + events是要观察文件描述符的哪些事件,主要有

    - EPOLLIN 可读

    - EPOLLOUT 可写

    - EPOLLERR 发生错误

    - EPOLLET 设置为边沿出发Edge Triggered, 类似于数电中的概念 ,如果不指定默认是水平触发Level Triggered

    - EPOLLONESHOT 只观察一次事件,一旦发生,对应的fd就会被内核从集合中删除

    - EPOLLPRI tcp 带外数据,用的很少

    - EPOLLHUP 对应的文件描述符被挂断,终端关闭事件.

    + data中的fd 设置为和epoll_ctl参数列表中的fd一致即可.

经过以上处理,内核已经知道了和epfd关联的待观察文件描述符集以及上面的事件.

等待关心事件的发生.

epoll_wait就是用来等待上述事件发生的,原型:int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout), epfd和epoll_ctl是一样的,events是内核返回给进程哪个文件描述符发生了什么事情,具体用法参考epoll_ctl。maxevents指明了events数组 大小;timeout可以指定超时,单位为毫秒(0会立即返回,-1是永久阻塞)。

unix网络编程中的str_cli函数

这个函数同时监听标准输入和一个连接到服务器的socket,如果用户输入了什么内容就会发送到服务器,如果从服务器收到了任何内容就直接输出到终端.


  1. //
  2. // Created by bai on 2/2/16.
  3. //
  4. #include "../lib/unp.h"
  5. #include <sys/epoll.h>
  6. void str_cli(FILE *fp, int sockfd)
  7. {
  8. int efd;
  9. struct epoll_event event;
  10. struct epoll_event events[20];
  11. int i;
  12. int nfds;
  13. char buf[MAXLINE];
  14. int n;
  15. efd = epoll_create (10);
  16. if(efd<0){
  17. err_sys("epoll create failed");
  18. }
  19. event.data.fd=fileno(fp);
  20. event.events=EPOLLIN;
  21. epoll_ctl(efd,EPOLL_CTL_ADD,fileno(fp),&event);
  22. event.data.fd=sockfd;
  23. event.events=EPOLLIN;
  24. epoll_ctl(efd,EPOLL_CTL_ADD,sockfd,&event);
  25. for ( ; ; ) {
  26. nfds=epoll_wait(efd,events,sizeof(events)/sizeof(struct epoll_event),-1);
  27. if(nfds<0){
  28. err_sys("epoll wait error %d ",errno);
  29. }
  30. for(i=0;i<nfds;i++){
  31. if(events[i].data.fd==sockfd){
  32. if ( (n = Read(sockfd, buf, MAXLINE)) == 0) {
  33. err_quit("str_cli: server terminated prematurely");
  34. }
  35. Write(fileno(stdout), buf, n);
  36. }
  37. if(events[i].data.fd==fileno(fp)){
  38. if ( (n = Read(fileno(fp), buf, MAXLINE)) == 0) {
  39. close(sockfd);
  40. break;
  41. }
  42. Writen(sockfd, buf, n);
  43. }
  44. }
  45. }
  46. close(efd);
  47. }

unix网络编程str_cli使用epoll实现的更多相关文章

  1. unix网络编程 str_cli epoll 非阻塞版本

    unix网络编程 str_cli epoll 非阻塞版本 unix网络编程str_cli使用epoll实现讲了使用epoll配合阻塞io来实现str_cli,这个版本是配合非阻塞io. 可以看到采用非 ...

  2. UNIX网络编程学习指南--epoll函数

    epoll是select/poll的强化版,都是多路复用的函数,epoll有了很大的改进. epoll的功能 1.支持监听大数目的socket描述符 一个进程内,select能打开的fd是有限制的,有 ...

  3. UNIX网络编程——epoll 的accept , read, write(重要)

    在一个非阻塞的socket上调用read/write函数,返回EAGAIN或者EWOULDBLOCK(注:EAGAIN就是EWOULDBLOCK). 从字面上看,意思是: EAGAIN: 再试一次 E ...

  4. 《Unix 网络编程》14:高级 I/O 函数

    高级 I/O 函数 ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ...

  5. 《 UNIX网络编程》源码的使用

    学习编程这东西,看代码,改代码,运行代码这样才能学到实际东西!本书说在www.unpbook.com可以获取源码,不过打不开!所以google unpv13e.tar.gz 并在网络上找到了:源码:h ...

  6. 记录一次配置unix网络编程环境的过程和遇到的问题

    记录一次搭建unix网络编程环境过程中遇到的问题和总结 计算机环境虚拟机 linuxmint-18-xfce-64bit 1.打开unix网络编程.iso 把目录下的文件复制到某一目录,修改权限,可命 ...

  7. 【unix网络编程第三版】阅读笔记(五):I/O复用:select和poll函数

    本博文主要针对UNP一书中的第六章内容来聊聊I/O复用技术以及其在网络编程中的实现 1. I/O复用技术 I/O多路复用是指内核一旦发现进程指定的一个或者多个I/O条件准备就绪,它就通知该进程.I/O ...

  8. UNIX网络编程——并发服务器(TCP)

    在迭代服务器中,服务器只能处理一个客户端的请求,如何同时服务多个客户端呢?在未讲到select/poll/epoll等高级IO之前,比较老土的办法是使用fork来实现. 网络服务器通常用fork来同时 ...

  9. 《Unix 网络编程》05:TCP C/S 程序示例

    TCP客户/服务器程序示例 系列文章导航:<Unix 网络编程>笔记 目标 ECHO-Application 结构如下: graph LR; A[标准输入/输出] --fgets--> ...

随机推荐

  1. spring mvc helloworld 和表单功能、页面重定向

    Spring MVC Hello World 例子 这里有个很好的教程:https://www.cnblogs.com/wormday/p/8435617.html 下面的例子说明了如何使用 Spri ...

  2. oracle查看表空间和物理文件大小

    查看各表空间的使用情况 select a.tablespace_name,a.bytes/1024/1024 "Sum MB",(a.bytes-b.bytes)/1024/102 ...

  3. python学习笔记(三):文件操作和集合

    对文件的操作分三步: 1.打开文件获取文件的句柄,句柄就理解为这个文件 2.通过文件句柄操作文件 3.关闭文件. 文件基本操作: f = open('file.txt','r') #以只读方式打开一个 ...

  4. puclic 页面公共CSS样式

    body, div, dl, dt, dd, ul, ol, li, pre, form, fieldset, blockquote, h1, h2, h3, h4, h5, h6,p{ paddin ...

  5. MySQL备份还原之三使用xtrabackup

    1 xtrabackup安装 1)解压源码包 tar -xzvf percona-xtrabackup-2.1.7.tar.gz 2)安装perl环境(DBI/DBD) yum install per ...

  6. Sublime Text 套件介紹:Pretty JSON

    JSON,一個輕量級的資料交換語言,目前許多網站AJAX request的回應結果都是JSON格式   以下是一個標準的JSON格式   1 2 3 4 5 6 7 8 9 10 11 12 13 1 ...

  7. spark编译安装 spark 2.1.0 hadoop2.6.0-cdh5.7.0

    1.准备: centos 6.5 jdk 1.7 Java SE安装包下载地址:http://www.oracle.com/technetwork/java/javase/downloads/java ...

  8. Linux常用基本命令 1

    useradd 创建用户. password 修改密码. date 查看时间 man date 帮助文档.f往后翻 b往前翻 q退出.软修改 man hwclock 修改硬件时钟, cal 查看日历 ...

  9. Zend Studio 10汉化方法

    选择Help菜单->Install New Software...在Work with框中复制此地址:http://download.eclipse.org/technology/babel/u ...

  10. c++ (proxy)代理模式

    假设我们有几个具有相似的窗体,都包含关闭窗体(closeButton)和按钮单击事件(ClickButton)我们在处理时,不想直接操作每个窗体,可以请求代理. #include<iostrea ...