一、五种I/O模型
linux下的I/O模型分为五种模型:分别是阻塞式I/O、非阻塞式I/O、I/O多路复用(多路转接)、信号驱动I/O(SIGIO)、异步I/O。
这五种I/O由分为两类:一类是同步I/O包括了前面四种,还有一类是异步I/O。
我们把I/O是分成两步的,第一步是“等”,第二步就是数据的搬迁。而对于同步I/O和异步I/O来说它们之间的区别就是同步I/O的数据搬迁工作要自己来完成,而异步I/O的数据搬迁工作是别人来完成的。而阻塞I/O和非阻塞I/O的最大不同就是“等”的方式是不同的。

二、select函数介绍
select系统调用的用途是:在一段指定的时间内,监听用户感兴趣的文件描述符上的可读、可写、异常事件。
我们看一下select函数的原型:

我们可以看到select函数里面有五个参数,下面一次做一个说明:
1、nfds:指定被监听的文件描述符的总数,通常设置为select监听的所有文件描述符的中的最大值再加1(文件描述符是从0开始的)。
2、readfds、writefds、exceptfds这三个参数都是输入输出型参数,它们表示的是可读、可写、异常事件的文件描述符的集合。这三个参数做输入型参数的意思是:关心特定的文件描述符上的指定事件;这三个参数作为输出型参数的意思是:关心的事件发生了变化已经就绪。这三个参数的类型都是fd_set的指针类型的结构体,fd_set结构体仅包含一个整形数组,数组的每一位元素的每一位(bit)标记了一个文件描述符。fd_set所能包含的文件描述符的数量由FD_SETSIZE指定,这就限制了select处理的文件描述符的总量,这样select的服务就是有上限的。
我们在访问fd_set这个数据结构的时候还有几个宏来访问fd_set,集体如下:
FD_ZERO(fd_set *fdset) //清除fdset所有位
FD_SET(int fd, fd_set *fdset) //设置fdset位
FD_CLR(int fd, fd_set *fdset) //清除fdset位
int FD_ISSET(nt fd, fd_set *fdset)//检查fdset是否被设置
3、timeout:

用来设置select函数的超时时间,select给我们提供了而一个微秒级的计时方式,如果timeout的成员tv_sec和tv_usec的成员都为0,select就会立即返回。
4、返回值:
select成功时就会返回就绪(可读、可写、异常)的文件描述符的总数,在超时时间内没有任何文件描述符就绪返回0,如果select失败就会返回-1,设置errno为EINTR。

三、代码展示
select_server.c:

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<sys/select.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<fcntl.h>
#include<unistd.h>int fds[sizeof(fd_set)*8];static usage(const char* proc)
{printf("usage :%s [local_ip] [local_port]\n",proc);
}int startup(const char* ip,int port)
{int sock = socket(AF_INET,SOCK_STREAM,0);if(sock < 0){perror("socket");exit(2);}int opt = 1;setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt));struct sockaddr_in local;local.sin_family = AF_INET;local.sin_port = htons(port);local.sin_addr.s_addr = inet_addr(ip);//点分十进制化为四字节if(bind(sock,(struct sockaddr*)&local,sizeof(local)) < 0){perror("bind");exit(3);}if(listen(sock,10) < 0){perror("listen");exit(4);}return sock;
}int main(int argc,char* argv[])
{if(argc != 3){usage(argv[0]);return 1;}int listen_sock = startup(argv[1],atoi(argv[2]));printf("fd_set: %d\n",sizeof(fd_set)*8);int fds[sizeof(fd_set)];int nums = sizeof(fds)/sizeof(fds[0]);int i = 0;for(; i < nums; i++){fds[i] = -1;}fds[0] = listen_sock;int maxfd = -1;fd_set rfds;//读事件fd_set wfds;//写事件while(1){int maxfd = -1;struct timeval timeout = {2,0};FD_ZERO(&rfds);FD_ZERO(&wfds);i = 0;for(; i < nums;i++){if(fds[i] == -1){continue;}FD_SET(fds[i],&rfds);if(maxfd < fds[i]){maxfd = fds[i];}}switch(select(maxfd+1,&rfds,&wfds,NULL,&timeout)){                      case -1:   //select失败perror("select");break;case 0:    //超过时间没有任何描述符就绪printf("time out!\n");break;default:   //成功{//at least one fd ready!i = 0;for(; i < nums;i++){if(i == 0 && FD_ISSET(fds[i],&rfds))//listen_sock is ready, get connect{struct sockaddr_in client;socklen_t len = sizeof(client);int new_sock = accept(listen_sock,(struct sockaddr*)&client,&len);if(new_sock < 0){perror("accept");continue;}//accept只是获得了一个新的连接  但是不能保证读和写正常//如果对面一直不发数据  那么就会一直处于read里的等待情况  就会导致服务器的不能正常运行printf("get a new client: [%s:%d]\n",inet_ntoa(client.sin_addr),ntohs(client.sin_port));int j = 1;for(; j < nums ; j++){if(fds[j] == -1){break;}}if(j == nums)//已经达到所能监听的文件描述符的最大值 只能关闭{close(new_sock);}else{fds[j] = new_sock;}}else if(i != 0 && FD_ISSET(fds[i],&rfds))//normal fd is ready{char buf[1024];ssize_t s = read(fds[i],buf,sizeof(buf)-1);if( s > 0){buf[s] = 0;printf("client# %s\n",buf);FD_SET(fds[i],&wfds);}else if(s == 0){printf("client is quit!\n");close(fds[i]);fds[i] = -1;}else{perror("read");close(fds[i]);fds[i] = -1;}}if(i !=0 && FD_ISSET(fds[i],&wfds))//普通的写操作{const char* msg = "hello client!\n";ssize_t s = write(fds[i],msg,strlen(msg));if(s < 0){perror("write");}else{FD_CLR(fds[i],&wfds);}}}break;}}}close(listen_sock);return 0;
}

select_client.c:

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<sys/select.h>
#include<arpa/inet.h>
#include<fcntl.h>
#include<unistd.h>
#include<netinet/in.h>static usage(const char* proc)
{printf("usage: [client_ip] [client_port]%s\n",proc);
}int main(int argc,const char* argv[])
{if(argc != 3){usage(argv[0]);return 1;}int sock = socket(AF_INET,SOCK_STREAM,0);if(sock < 0){perror("socket");exit(2);}struct sockaddr_in server;server.sin_family = AF_INET;server.sin_port =htons(atoi(argv[2]));server.sin_addr.s_addr = inet_addr(argv[1]);if(connect(sock,(struct sockaddr*)&server,sizeof(server)) < 0){perror("connect");exit(3);}char  buf[1024];int fd;while(1){printf("please Enter#: ");fflush(stdout);fd = dup(1);ssize_t s = read(0,buf,sizeof(buf)-1);if(s > 0){buf[s-1] = 0;close(1);dup2(sock,1);printf("%s",buf);fflush(stdout);}else{perror("read");exit(4);}dup2(fd,1);ssize_t _s = read(sock,buf,sizeof(buf)-1);if(_s > 0){buf[_s] = 0;printf("server echo#: %s\n",buf);}}close(fd);close(sock);return 0;
}

四、运行结果
服务器端运行情况:

客户端的运行情况:

I/O多路复用select服务器相关推荐

  1. LinuxI/O多路复用转接服务器——select模型实现

    LinuxI/O多路复用转接服务器--select模型实现 select函数 函数原型 参数和返回值 fd_set结构体 位操作函数 select实现实现I/O多路复用服务器 实现流程 程序实现 服务 ...

  2. java nio原理 epoll_多路复用 Select Poll Epoll 的实现原理(BIO与NIO)

    BIO blocking阻塞的意思,当我们在后端开发使用的时候,accetp 事件会阻塞主线程. 当accept事件执行的时候,客户的会和服务建立一个socket 连接.一般后端就会开启一个线程执行后 ...

  3. socket 编程篇六之IPO多路复用-select poll epoll

    http://blog.csdn.net/woxiaohahaa/article/details/51498951 文章参考自:http://blog.csdn.net/tennysonsky/art ...

  4. LinuxI/O多路复用转接服务器——epoll模型实现

    LinuxI/O多路复用转接服务器--epoll模型实现 epoll函数 epoll函数组 epoll_create函数 epoll_ctl函数 epoll_wait函数 epoll实现实现I/O多路 ...

  5. I/O多路转换之select与select服务器

    select高效的原因 首先要知道一个概念,一次I/O分两个部分(①等待数据就绪 ②进行I/O),减少等的比重,增加I/O的比重就可以达到高效服务器的目的.select工作原理就是这个,同时监控多个文 ...

  6. LinuxI/O多路复用转接服务器——poll模型实现

    LinuxI/O多路复用转接服务器--poll模型实现 poll函数 函数原型 参数和返回值 poll实现实现I/O多路复用服务器 实现流程 程序实现 服务端程序 客户端程序 运行结果 poll优缺点 ...

  7. 多路复用select和epoll的区别(转)

    先说下本文框架,先是问题引出,然后概括两个机制的区别和联系,最后介绍每个接口的用法 一.问题引出 联系区别 问题的引出,当需要读两个以上的I/O的时候,如果使用阻塞式的I/O,那么可能长时间的阻塞在一 ...

  8. python3 异步 非阻塞 IO多路复用 select poll epoll 使用

    有许多封装好的异步非阻塞IO多路复用框架,底层在linux基于最新的epoll实现,为了更好的使用,了解其底层原理还是有必要的. 下面记录下分别基于Select/Poll/Epoll的echo ser ...

  9. linux I/O--I/O多路复用--select总结(三)

    1.基本概念 IO多路复用是指内核一旦发现进程指定的一个或者多个IO条件准备读取,它就通知该进程.IO多路复用适用如下场合: 当客户处理多个描述字时(一般是交互式输入和网络套接口),必须使用I/O复用 ...

  10. I/O多路复用——select

    系统提供select函数来实现多路复用I/O模型,select系统调用是用来让我们的程序监视多个文件句柄的状态变化的.程序会停在select这里等待,直到被监视的文件句柄有一个或多个发生了状态改变. ...

最新文章

  1. 提高PHP运行速度的小技巧
  2. ComponentOne WebChart for ASP.NET基础
  3. linux环境: shell初始化文件, for TCSH, CSH
  4. java基础File的简单使用记录
  5. 史上最全企业数据产品选型对比(含数仓、报表、BI、中台、数据治理)
  6. Oracle使用IN 不能超过1000问题
  7. Spring按类型自动装配注入数组、集合、Map
  8. eclipse 配色方案
  9. Adobe Flex初记
  10. OSPF实验:OSPF认证
  11. unity linux桌面环境,现在仍然可以在Ubuntu 20.04上安装Unity桌面环境
  12. 未捕获异常string was not recognized_给你代码:PHP7中的异常与错误处理
  13. pyspark若不能运行,需指定支持版本的java:os.environ[‘JAVA_HOME‘] = ‘F:\Java15.0.2‘
  14. uva10246- Asterix and Obelix
  15. Android仿人人客户端(v5.7.1)——新鲜事之完整篇
  16. 宠物动物有什么计算机专业岗位,宠物医院前台岗位职责
  17. cocos creator 加载微信云端图片
  18. 2021GPLT L1-8 乘法口诀数列(JAVA语言)
  19. 智慧家安监控系统——用Java + html、css、js实现
  20. 英语口语六级模拟测试软件,2017英语六级口语考试试题及答案_英语六级口语试题及答案 doc完整版...

热门文章

  1. python用嵌套if结构开发一个输入(input)快递价格的计算器
  2. hdmi接口线_HDMI高清线注意事项
  3. android图片颜色识别器,颜色识别器APP
  4. 13 Python函数进阶
  5. CSDN搬家到博客园
  6. 洛谷P3332 K大数查询
  7. 报价单常见问题及高效拯救你的报价实用手册
  8. Lookup 组件用法全解
  9. 新路由3详细刷机教程
  10. python 单因子方差分析_假设检验之F检验-方差分析