body, table{font-family: 微软雅黑; font-size: 13.5pt} table{border-collapse: collapse; border: solid gray; border-width: 2px 0 2px 0;} th{border: 1px solid gray; padding: 4px; background-color: #DDD;} td{border: 1px solid gray; padding: 4px;} tr:nth-child(2n){background-color: #f8f8f8;}

I/O多路转接模型:
    在这种模型下,如果请求的I/O操作阻塞,且它不是真正阻塞I/O,而是让其中的一个函数等待,在这期间,I/O还能进行其他操作。
#include <sys/select.h>
#include <sys/time.h>
int select(int maxfd, fd_set *readset, fd_set *writeset, fd_set *exceptionset, const struct timeval* timeout);   \\返回:就绪描述字的正数目,0——超时,-1——出错
参数解释:
maxfd            最大的文件描述符(其值应该为最大的文件描述符字 + 1)
readset          内核读操作的描述符字集合
writeset         内核写操作的描述符字集合
exceptionset     内核异常操作的描述符字集合
timeout          等待描述符就绪需要多少时间。NULL代表永远等下去,一个固定值代表等待固定时间,0代表根本不等待,检查描述字之后立即返回。

其中 readset、writeset、exceptionset 都是 fd_set 集合。该集合的相关操作如下:
void FD_ZERO(fd_set *fdset);               /* 将所有fd清零 */
void FD_SET(int fd, fd_set *fdset);        /* 增加一个fd */
void FD_CLR(int fd, fd_set *fdset);        /* 删除一个fd */
int FD_ISSET(int fd, fd_set *fdset);       /* 判断一个fd是否有设置 */

一般来说,在使用select函数之前,首先要使用FD_ZERO和FD_SET来初始化文件描述符集,在使用select函数时,可循环使用FD_ISSET测试描述符集,在执行完对相关文件描述符之后,使用FD_CLR来清除描述符集(也可以直接用FD_ZERO直接全部清零)。

select函数中的 timeout 是一个 struct timeval 类型的指针,该结构体如下:
struct timeval
{
        long tv_sec;         /* second */         //秒
        long tv_usec;        /* microsecond */    //微秒
};


流程:

fd_set readset;
FD_SET(0,&readset);
FD_SET(fdr,&readset);
int ret=select(fdr+1,&readset,NULL,NULL,NULL);
if(ret >0)
{
     if(FD_ISSET(0,&readset))
     {
     }
     if(FD_ISSET(fdr,&readset))
     {
     }
}

select 监控哪些描述符可以读

a.c b.c
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<sys/time.h>
#include<unistd.h>
#include<stdio.h>
#include<strings.h>
#include<string.h>
int main(int argc,char** argv)
{
        int fdw=open(argv[1],O_RDWR);
        int fdr=open(argv[2],O_RDWR);
        printf("fdw=%d fdr=%d\n",fdw,fdr);
        char buf[128]="";
        fd_set rdset;
        int ret;
        while(1)
        {
                FD_ZERO(&rdset);
                FD_SET(0,&rdset);
                FD_SET(fdr,&rdset);
                ret=select(fdr+1,&rdset,NULL,NULL,NULL);
                if(ret>0)
                {
                        if(FD_ISSET(0,&rdset))
                        {
                                bzero(buf,sizeof(buf));
                                read(0,buf,sizeof(buf));
                                write(fdw,buf,strlen(buf)-1);  //从控制终端读的时候,回车也会读进去,所以要-1去掉回车
                        }
                        if(FD_ISSET(fdr,&rdset))
                        {
                                bzero(buf,sizeof(buf));
                                read(fdr,buf,sizeof(buf));
                                printf("b:%s\n",buf);
                        }
                }
        }
        return 0;
}

#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<sys/time.h>
#include<unistd.h>
#include<stdio.h>
#include<strings.h>
#include<string.h>
int main(int argc,char** argv)
{
        int fdr=open(argv[1],O_RDWR);
        int fdw=open(argv[2],O_RDWR);
        printf("fdr=%d fdw=%d\n",fdr,fdw);
        char buf[128]="";
        fd_set rdset;
        int ret;
        while(1)
        {
                FD_ZERO(&rdset);
                FD_SET(0,&rdset);
                FD_SET(fdr,&rdset);
                ret=select(fdr+1,&rdset,NULL,NULL,NULL);
                if(ret>0)
                {
                        if(FD_ISSET(0,&rdset))
                        {
                                bzero(buf,sizeof(buf));
                                read(0,buf,sizeof(buf));
                                write(fdw,buf,strlen(buf)-1);
                        }
                        if(FD_ISSET(fdr,&rdset))
                        {
                                bzero(buf,sizeof(buf));
                                read(fdr,buf,sizeof(buf));
                                printf("a:%s\n",buf);
                        }
                }
        }
        return 0;
}

I/O多路转接模型的更多相关文章

  1. 多路转接模型之poll

    poll系统调用和select类似.也是在指定时间内轮询一定数量的文件描写叙述符,以測试当中是否有就绪者.poll和select效率差点儿相同,仅仅是其使用接口相对简单些,poll不在局限于1024个 ...

  2. Linux下I/O多路转接之select --fd_set

    fd_set 你终于还是来了,能看到这个标题进来的,我想,你一定是和我遇到了一样的问题,一样的疑惑,接下来几个小时,我一定竭尽全力,写出我想说的,希望也正是你所需要的: 关于Linux下I/O多路转接 ...

  3. 高级I/O之I/O多路转接——pool、select

    当从一个描述符读,然后又写到另一个描述符时,可以在下列形式的循环中使用阻塞I/O: ) if (write(STDOUT_FILENO, buf, n) != n) err_sys("wri ...

  4. IO多路转接select和poll

    select IO多路复用的设置方法与信号的屏蔽有点相似: 信号屏蔽需要先设定一个信号集, 初始化信号集, 添加需要屏蔽的信号, 然后用sigprocmask设置 IO多路转接需要先设定一个文件描述符 ...

  5. I/O多路转接 --- UNIX环境高级编程

    I/O多路转接技术:先构造一张有关描述符的列表,然后调用一个函数,知道这些描述符中的一个已准备好进行I/O时,给函数才返回.在返回时,它告诉进程哪些描述符已准备好可以进行I/O. poll.selec ...

  6. Linux下I/O多路转接之epoll(绝对经典)

    epoll 关于Linux下I/O多路转接之epoll函数,什么返回值,什么参数,我不想再多的解释,您不想移驾,我给你移来: http://blog.csdn.net/colder2008/artic ...

  7. UNIX环境高级编程——I/O多路转接(select、pselect和poll)

    I/O多路转接:先构造一张有关描述符的列表,然后调用一个函数,直到这些描述符中的一个已准备好进行I/O时,该函数才返回.在返回时,它告诉进程哪些描述符已准备好可以进行I/O. poll.pselect ...

  8. select函数与I/O多路转接

    select函数与I/O多路转接 相作大家都写过读写IO操作的代码,例如从socket中读取数据可以使用如下的代码: while( (n = read(socketfd, buf, BUFSIZE) ...

  9. I/O多路转接-epoll

    By francis_hao    Aug 5,2017   APUE讲多路转接的章节介绍了select.pselect和poll函数.而epoll是linux内核在2.5.44引入的.在glibc ...

随机推荐

  1. 一 : springmvc常用注解

    springmvc常用注解详解1.@Controller在SpringMVC 中,控制器Controller 负责处理由DispatcherServlet 分发的请求,它把用户请求的数据经过业务处理层 ...

  2. Android初学:Gradle 'HelloWorld' project refresh failed

    Gradle 'HelloWorld' project refresh failed Error:Failed to open zip file.Gradle's dependency cache m ...

  3. Oracle_SQL92_连接查询

    Oracle_SQL92_连接查询   笛卡儿积 --笛卡尔积 select * from emp;----14 select * from dept;----4 select * from emp, ...

  4. Linux包管理器

    按Linux系统分类 Redhat系列:Redhat(本身就是Centos).Centos.Fedora等,采用Dpkg包管理器 Debian系列:Debian.Ubuntu等,使用RPM包管理器 R ...

  5. 使用vue框架运行npm run dev 时报错解决

    使用使用vue框架运行npm run dev 时报错 如下: 原因: localhost:8080 有可能其他软件占用了,导致其他问题的出现 我们可以动态修改地址 解决: 进入项目文件的config文 ...

  6. 将自己的代码托管到github上

    这几天一直在做一个爬虫的小demo,代码基本写的差不多了,想着如何把他放在一个地方,如是乎注册了一个github账号,开始了自己的git之旅. 首先是下载git,这个我就不多说啦!到处都有推荐看看廖雪 ...

  7. Codeforces 900 E. Maximum Questions (DP,技巧)

    题目链接:900 E. Maximum Questions 题意: 给出一个长度为n只含有a和b还有'?'的串s,且'?'可以被任意替换为a或b.再给出一个字符串t (奇数位上为a,偶数位上为b,所以 ...

  8. WPF 简易新手引导

    这两天不忙,所以,做了一个简易的新手引导小Demo.因为,不是项目上应用,所以,做的很粗糙,也就是给需要的人,一个思路而已. 新手引导功能的话,就是告诉用户,页面上操作的顺序,第一步要做什么,第二步要 ...

  9. Hystrix-request collapsing(请求合并)

    介绍: Hystrix的请求合并就是把重复的请求批量的用一个HystrixCommand命令去执行,以减少通信消耗和线程数的占用.Hystrix的请求合并用到了HystrixCollapser这个抽象 ...

  10. python_爬百度百科词条

    如何爬取? 明确目标:爬取百度百科,定初始百度词条:python,初始URL:http://baike.baidu.com/item/Python,爬取数据量为1000条,值爬取简介,标题,和简介中u ...