多路转接模型之poll
poll系统调用和select类似。也是在指定时间内轮询一定数量的文件描写叙述符,以測试当中是否有就绪者。poll和select效率差点儿相同,仅仅是其使用接口相对简单些,poll不在局限于1024个文件描写叙述符。poll监听事件和触发事件分开,event表示监听事件。revents表示触发的事件。
相比select不用每一次都须要又一次设置监听事件。
- #include <poll.h>
- int poll(struct pollfd *fds, nfds_t nfds, int timeout);
- //第一个參数是struct pollfd数组
- struct pollfd
- {
- int fd; /* file descriptor */你要监控文件描写叙述符
- short events; /* requested events */ 监听文件描写叙述符上的事件 传入參数由用户设置
- short revents; /* returned events */监控文件描写叙述符事件返回值 传出參数由内核设置
- };
- POLLIN普通或带外优先数据可读,即POLLRDNORM | POLLRDBAND
- POLLRDNORM-数据可读
- POLLRDBAND-优先级带数据可读
- POLLPRI 高优先级可读数据
- POLLOUT普通或带外数据可写
- POLLWRNORM-数据可写
- POLLWRBAND-优先级带数据可写
- POLLERR 错误发生
- POLLHUP 发生挂起
- POLLNVAL 描写叙述字不是一个打开的文件
- 第二个參数。指结构体数组长度。
- timeout 毫秒级等待
- -1:堵塞等,#define INFTIM -1 Linux中未定义此宏
- 0:马上返回。不堵塞进程
- >0:等待指定毫秒数,如当前系统时间精度不够毫秒,向上取值
poll server端实例:
- #include<stdio.h>
- #include<string.h>
- #include<poll.h>
- #include <sys/un.h>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <netinet/in.h>
- #include <arpa/inet.h>
- #include<errno.h>
- #define OPEN_MAX 1024
- int create_listen(int port)
- {
- int listen_st,on;
- struct sockaddr_in s_addr;
- listen_st =socket(AF_INET,SOCK_STREAM,0);
- if(listen_st==-1)
- {
- perror("socket error ");
- return -1;
- }
- if(setsockopt(listen_st,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on))==-1)
- {
- perror("setsockopt error");
- return -1;
- }
- s_addr.sin_port=htons(port);
- s_addr.sin_family=AF_INET;
- s_addr.sin_addr.s_addr=htonl(INADDR_ANY);
- if(bind(listen_st,(struct sockaddr*)&s_addr,sizeof(struct sockaddr_in))==-1)
- {
- perror("bind error");
- return -1;
- }
- if (listen(listen_st, 5) == -1) // 设置文件描写叙述符具有监听的功能
- {
- perror("listen error");
- return -1;
- }
- return listen_st;
- }
- int run_server(int port)
- {
- int i,maxi,listen_st,conn_st,sockaddr_len;
- int nready;
- struct pollfd client[OPEN_MAX];
- char buf[1024];
- struct sockaddr_in c_addr;
- listen_st=create_listen(port);
- if(listen_st==-1)
- {
- return -1;
- }
- for(i=1;i<OPEN_MAX;i++)
- {
- client[i].fd=-1;
- }
- client[0].fd=listen_st;
- client[0].events=POLLIN;
- maxi=0;
- while(1)
- {
- nready = poll(client,maxi+1,-1);//poll 堵塞
- if(nready<0)
- {
- perror("poll error");
- break;
- }
- if((client[0].revents&POLLIN))//检測listen_st
- {
- sockaddr_len=sizeof(c_addr);
- conn_st=accept(listen_st,(struct sockaddr *)&c_addr,&sockaddr_len);
- printf("received form %s at port:%d \n",inet_ntoa(c_addr.sin_addr),ntohs(c_addr.sin_port));
- for(i=0;i<OPEN_MAX;i++)
- {
- if(client[i].fd<0)
- {
- client[i].fd=conn_st;
- client[i].events=POLLIN;
- break;
- }
- }
- if(i==OPEN_MAX)
- {
- printf("too many client \n");
- close(conn_st);
- }else
- {
- if(i>maxi) //记录最大下标
- {
- maxi=i;
- }
- }
- if(--nready==0) continue;
- }
- for(i=1;i<=maxi;i++)
- {
- if((conn_st=client[i].fd)<0)
- {
- continue;
- }
- if(client[i].revents&POLLIN)
- {
- memset(buf,0,sizeof(buf));
- int rv=read(conn_st,buf,sizeof(buf));
- if(rv<0)
- {
- if(errno==ECONNRESET)/* 当收到RST标志时*/
- //这样的错误是因为客户端发过FIN ACk掉线了客服端进程已经结束了 服务端再发FIN 客户端会发送RST
- {
- printf("client aborted connection \n");
- close(conn_st);
- client[i].fd=-1;
- }
- }
- else if(rv==0)
- {
- printf("close client \n");
- close(conn_st);
- client[i].fd=-1;
- }
- else
- {
- printf("recv from client:%s \n",buf);
- write(conn_st,buf,strlen(buf));
- }
- if (--nready == 0) break; //就绪个数减一
- }
- }
- }
- close(listen_st);
- return 0;
- }
- int main(int argc,char *argv[])
- {
- if(argc<2)
- {
- printf("usage:%s port \n",argv[0]);
- return 0;
- }
- int port=atoi(argv[1]);
- if(port==0)
- {
- printf("port error \n");
- return 0;
- }
- printf("start server \n");
- run_server(port);
- return 0;
- }
多路转接模型之poll的更多相关文章
- I/O多路转接模型
body, table{font-family: 微软雅黑; font-size: 13.5pt} table{border-collapse: collapse; border: solid gra ...
- IO多路转接select和poll
select IO多路复用的设置方法与信号的屏蔽有点相似: 信号屏蔽需要先设定一个信号集, 初始化信号集, 添加需要屏蔽的信号, 然后用sigprocmask设置 IO多路转接需要先设定一个文件描述符 ...
- UNIX环境高级编程——I/O多路转接(select、pselect和poll)
I/O多路转接:先构造一张有关描述符的列表,然后调用一个函数,直到这些描述符中的一个已准备好进行I/O时,该函数才返回.在返回时,它告诉进程哪些描述符已准备好可以进行I/O. poll.pselect ...
- 【Nginx】I/O多路转接之select、poll、epoll
当需要读两个以上的I/O的时候,如果使用阻塞式的I/O,那么可能长时间的阻塞在一个描述符上面,另外的描述符虽然有数据但是不能读出来,这样实时性不能满足要求,大概的解决方案有以下几种: 1.使用多进程或 ...
- 高级I/O之I/O多路转接——pool、select
当从一个描述符读,然后又写到另一个描述符时,可以在下列形式的循环中使用阻塞I/O: ) if (write(STDOUT_FILENO, buf, n) != n) err_sys("wri ...
- I/O多路转接 --- UNIX环境高级编程
I/O多路转接技术:先构造一张有关描述符的列表,然后调用一个函数,知道这些描述符中的一个已准备好进行I/O时,给函数才返回.在返回时,它告诉进程哪些描述符已准备好可以进行I/O. poll.selec ...
- Linux下I/O多路转接之epoll(绝对经典)
epoll 关于Linux下I/O多路转接之epoll函数,什么返回值,什么参数,我不想再多的解释,您不想移驾,我给你移来: http://blog.csdn.net/colder2008/artic ...
- Linux下I/O多路转接之select --fd_set
fd_set 你终于还是来了,能看到这个标题进来的,我想,你一定是和我遇到了一样的问题,一样的疑惑,接下来几个小时,我一定竭尽全力,写出我想说的,希望也正是你所需要的: 关于Linux下I/O多路转接 ...
- select函数与I/O多路转接
select函数与I/O多路转接 相作大家都写过读写IO操作的代码,例如从socket中读取数据可以使用如下的代码: while( (n = read(socketfd, buf, BUFSIZE) ...
随机推荐
- AIX RAC 安装失败完全卸载
1,删除软件安装目录 rm -rf /u01/app 2,删除以下目录内容 rm -rf/tmp/.oracle rm -rf/tmp/* rm -rf/tmp/ora* rm -rf/var/tmp ...
- CreateWindowEx详解
语法: HWND CreateWindowEx( DWORD dwExStyle, LPCTSTR lpClassName, LPCTSTR lpWindowName, DWORD dwStyle, ...
- WM
#define WM_NULL 0x0000 #define WM_CREATE 0x0001 #define WM_DESTROY 0x0002 #define WM_MOVE 0x0003 #de ...
- java_StringBuffer、StringBuilder
StringBuffer和StringBuider是可变的字符串,使用方法 相同,StringBuffer是线程安全的,StringBuider是线程不安全的 public class StringT ...
- Java 调用存储过程 返回结果集
这里使用Oracle数据库的thin连接. 下面是存储过程SQL 1 createorreplaceprocedure proc3(stid in student.stuid%type, stname ...
- [NOI2005]聪聪与可可
题目大意:有小a和小b,其中一个人到处乱走,每次走一步:另一个人抄近路逼近,每次1-2步.求期望路程. 整解:跑1000遍最短路/bfs,求两两距离,然后找从x逼近y第一步去哪,最后期望dp收场. d ...
- <Linux> 下安装和卸载JDK
安装 下载jdk https://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html 在local ...
- linux arping-通过发送ARP协议报文测试网络
博主推荐:更多网络测试相关命令关注 网络测试 收藏linux命令大全 arping命令是用于发送arp请求到一个相邻主机的工具,arping使用arp数据包,通过ping命令检查设备上的硬件地址.能 ...
- 树莓派-3 启用root
默认是user: pi, password: raspberry 通过如下设置root密码并启用 pi@raspberrypi:~ $ sudo passwd root Enter new UNIX ...
- ruby 第五次作业 part 1(分类、排序)
movies_controller.rb class MoviesController < ApplicationController def movie_params params.requi ...