网络编程 IO多路复用

IO多路复用

    定义:单线程或单进程同时监测若干个文件描述符是否可以执行IO操作的能力
    
    作用:
    应用程序通常需要处理来自多条事件流中的事件,比如我现在用的电脑,需要同时处理键盘鼠标的输入、中断信号等等事件,再比如web服务器如nginx,需要同时处理来来自N个客户端的事件。
    
    逻辑控制流在时间上的重叠叫做 并发
    
    而CPU单核在同一时刻只能做一件事情,一种解决办法是对CPU进行时分复用(多个事件流将CPU切割成多个时间片,不同事件流的时间片交替进行)。在计算机系统中,我们用线程或者进程来表示一条执行流,通过不同的线程或进程在操作系统内部的调度,来做到对CPU处理的时分复用。这样多个事件流就可以并发进行,不需要一个等待另一个太久,在用户看起来他们似乎就是并行在做一样。
    
    使用并发处理的成本:
    线程/进程创建成本
    CPU切换不同线程/进程成本 Context Switch
    多线程的资源竞争
    
    有没有一种可以在单线程/进程中处理多个事件流的方法呢?一种答案就是IO多路复用。

    因此IO多路复用解决的本质问题是在用更少的资源完成更多的事。
    
    IO模型

    1、阻塞IO  (对设备进行读写操作,如果对方没有发送消息,则处于阻塞)
    2、非阻塞IO  EAGAIN  忙等待 errno                                                                                                                                                                                                                                    
    3、信号驱动IO  SIGIO 用的相对少(了解)
    4、并行模型 进程,线程
    5, IO多路复用  select、poll、epoll


    
    1、阻塞IO ===》最常用 默认设置

      f5e0d548f1db40c2a4259007e92cc7b2.png

6b4b00e309704e49a1f135ceca161e6a.png    当34行打印完后要等待 现在屏幕输入内容才能进行打印所以 如果不对stdin进行数据输入 则会进入阻塞

2、非阻塞

非阻塞IO ===》在阻塞IO的基础上调整其为不再阻塞等待。
     在程序执行阶段调整文件的执行方式为非阻塞:
            ===》fcntl() ===>动态调整文件的阻塞属性

    #include <unistd.h>
    #include <fcntl.h>
    int fcntl(int fd, int cmd, ... /* arg */ );
    功能:修改指定文件的属性信息。
    参数:fd 要调整的文件描述符
          cmd 要调整的文件属性宏名称
          ... 可变长的属性值参数。
    返回值:成功  不一定,看cmd
            失败  -1;

    eg:修改文件的非阻塞属性:
        int flag ;
        flag  = fcntl(fd,F_GETFL,0);  ///获取fd文件的默认属性到flag变量中。
        flag  = flag | O_NONBLOCK;    ///将变量的值调整并添加非阻塞属性
        fcntl(fd,F_SETFL,flag);       ///将新属性flag设置到fd对应的文件生效。

        以上代码执行后的阻塞IO将变成非阻塞方式。

    c9d906bc2085464b84dce9906bd171b7.png
 d62c9d7cf95b48d4a94476629ac6d558.png 通过改变管道属性变为非阻塞模式,以及因为要对stdin进行操作所以也要对stdin -- 0 进行属性的改变 

在运行时 如果没有对屏幕进行输入也不影响上面的打印会一直在循环里,当对屏幕进行输入的时候也会直接打印

11520c5c6e944c658f1e35dce52a3518.png

ac7f8ebde9c742118137cd3360302fef.png

 3.信号驱动io

 
    文件描述符需要追加 O_ASYNC 标志。
    设备有io事件可以执行时,内核发送SIGIO信号。(当对面开始写,就给这边发来个信号,这边就开始读)
    
        1.追加标志
        int flag ;
        flag  = fcntl(fd,F_G ETFL,0);
        fcntl(fd,F_SETFL,flag | O_ASYNC);    

//flag=fcntl(fd,O_GETFL);

  flag=flag|O_ASYNC);

  fcntl(fd,F_SETFL,flag);

        2.设置信号接收者
        fcntl(fd,F_SETOWN,getpid());//常用设置
        3.对信号进行捕获
        signal(SIGIO,myhandle);//

2be71bed1b2649fba735fbfdfcf39927.png

读端:   

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <signal.h>
int fd;
void handle(int num)
{
    char buf[256]={0};
    read(fd,buf,sizeof(buf));
    printf("fifo %s\n",buf);

}
int main(int argc, char *argv[])
{
    signal( SIGIO,handle);
    int ret = mkfifo("myfifo",0666);    
    if(-1 == ret)
    {
        //如果是管道文件已存在错误,让程序继续运行
        if(EEXIST== errno)
        {
        
        }else 
        {
            perror("mkfifo");
            exit(1);
        }
    }

    fd = open("myfifo",O_RDONLY);
    if(-1 == fd)
    {
        perror("open");
        exit(1);
    }
    int flag = fcntl(fd,F_GETFL);
    fcntl(fd,F_SETFL ,flag|O_ASYNC);

    fcntl(fd,F_SETOWN, getpid() );
    while(1)
    {
        char buf[256]={0};
        fgets(buf,sizeof(buf),stdin);
        printf("terminal:%s\n",buf);
    }
    close(fd);
    //remove("myfifo");

    return 0;
}

写端如上题所述;


    4.并发 


        1.进程
        2.线程
 IO 多路复用 ===》并发服务器 ===》TCP协议    

读端;

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <pthread.h>

void * th1(void* arg)
{

    int fd = *(int*)arg;
    while(1)
    {
        char buf[256]={0};
        read(fd,buf,sizeof(buf));
        printf("fifo %s\n",buf);
    }

    return NULL;
}
void* th2(void* arg)
{
    while(1)
    {
        char buf[256]={0};
        fgets(buf,sizeof(buf),stdin);
        printf("terminal:%s\n",buf);
    }
}
int main(int argc, char *argv[])
{
    int ret = mkfifo("myfifo",0666);    
    if(-1 == ret)
    {
        //如果是管道文件已存在错误,让程序继续运行
        if(EEXIST== errno)
        {
        
        }else 
        {
            perror("mkfifo");
            exit(1);
        }
    }

    int fd = open("myfifo",O_RDONLY);
    if(-1 == fd)
    {
        perror("open");
        exit(1);
    }
    pthread_t tid1,tid2;
    pthread_create(&tid1,NULL,th1,&fd);
    pthread_create(&tid2,NULL,th2,NULL);
    pthread_join(tid1,NULL);
    pthread_join(tid2,NULL);
    close(fd);
    //remove("myfifo");

    return 0;
}

写段都一样
      、

5 IO多路复用 

select循环服务器 ===> 用select函数来动态检测有数据流动的文件描述符


    #include <sys/select.h>

    #include <sys/time.h>
    #include <sys/types.h>
    #include <unistd.h>

    int select(int nfds, fd_set *readfds, fd_set *writefds,
                fd_set *exceptfds,
                struct timeval *timeout);
    功能:完成指定描述符集合中有效描述符的动态检测。
          该函数具有阻塞等待功能,在函数执行完毕后
          目标测试集合中将只保留最后有数据的描述符。

    参数:nfds 描述符的上限值,一般是链接后描述符的最大值+1;
          readfds 只读描述符集
          writefds 只写描述符集
          exceptfds 异常描述符集
          以上三个参数都是 fd_set * 的描述符集合类型
          timeout  检测超时 如果是NULL表示一直检测不超时 。

    返回值:超时 0
            失败  -1
            成功 >0


        为了配合select函数执行,有如下宏函数:
        void FD_CLR(int fd, fd_set *set);
        功能:将指定的set集合中编号为fd的描述符号删除。

        int  FD_ISSET(int fd, fd_set *set);
        功能:判断值为fd的描述符是否在set集合中,
              如果在则返回真,否则返回假。

        void FD_SET(int fd, fd_set *set);
        功能:将指定的fd描述符,添加到set集合中。

        void FD_ZERO(fd_set *set);
        功能:将指定的set集合中所有描述符删除。

ef975b523013417dbb22c34429c6b42d.png

 读端:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <sys/select.h>

int main(int argc, char *argv[])
{
    int ret = mkfifo("myfifo",0666);    
    if(-1 == ret)
    {
        //如果是管道文件已存在错误,让程序继续运行
        if(EEXIST== errno)
        {
        
        }else 
        {
            perror("mkfifo");
            exit(1);
        }
    }

    int fd = open("myfifo",O_RDONLY);
    if(-1 == fd)
    {
        perror("open");
        exit(1);
    }
    //1 create set 
    fd_set rd_set,tmp_set;
    //2 add fd 
    FD_ZERO(&rd_set);
    FD_ZERO(&tmp_set);
    FD_SET(fd,&tmp_set);
    FD_SET(0,&tmp_set);
    struct timeval tv;
    while(1)
    {
        tv.tv_sec  =2;
        tv.tv_usec = 0;
        //4 clean flag
        rd_set = tmp_set;
        //3 wait event 
       int sel_ret =  select(fd+1,&rd_set,NULL,NULL,&tv);
       if(sel_ret>0)
       {
           char buf[256]={0};
           if(FD_ISSET(fd,&rd_set))
           {
               read(fd,buf,sizeof(buf));
               printf("fifo %s\n",buf);
           }
           if(FD_ISSET(0,&rd_set))
           {
               bzero(buf,sizeof(buf));
               fgets(buf,sizeof(buf),stdin);
               printf("terminal:%s\n",buf);
           }
       }
       else if(0 == sel_ret)
       {
        printf("time out\n");
       }
       else 
       {
            perror("select");
            exit(1);
       }
    }
    close(fd);
    //remove("myfifo");

    return 0;

写端:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
int main(int argc, char *argv[])
{
    int ret = mkfifo("myfifo",0666);    
    if(-1 == ret)
    {
        //如果是管道文件已存在错误,让程序继续运行
        if(EEXIST== errno)
        {

        }else 
        {
            perror("mkfifo");
            exit(1);
        }
    }
    //open 会阻塞,等到另一端读段打开,解除阻塞
    int fd = open("myfifo",O_WRONLY);
    if(-1 == fd)
    {
        perror("open");
        exit(1);
    }

    while(1)
    {
        char buf[256]="hello,fifo,test";
        write(fd,buf,strlen(buf));
        sleep(3);
    }
    close(fd);

    return 0;
}


      6  select poll epoll的区别

1. select
int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);
void FD_CLR(int fd, fd_set *set);
int FD_ISSET(int fd, fd_set *set);
void FD_SET(int fd, fd_set *set);
void FD_ZERO(fd_set *set);

select的调用一般要注意几点:
① readfds等是指针结果参数,会被函数修改,所以一般会另外定义一个allread_fdset,保持全部要监听读的句柄,将它的拷贝传递给select函数,返回可读的句柄集合,类型fdset支持赋值运算符=;
② 要注意计算nfds,当新增监听句柄时比较容易修改,当减少监听句柄时较麻烦些,如果要精确修改需要遍历或者采用最大堆等数据结构维护这个句柄集,以方便的找到第二大的句柄,或者干脆在减少监听句柄时不管nfds;
③ timeout如果为NULL表示阻塞等,如果timeout指向的时间为0,表示非阻塞,否则表示select的超时时间;
④ select返回-1表示错误,返回0表示超时时间到没有监听到的事件发生,返回正数表示监听到的所有事件数(包括可读,可写,异常),通常在处理事件时 会利用这个返回值来提高效率,避免不必要的事件触发检查。(比如总共只有一个事件,已经在可读集合中处理了一个事件,则可写和异常就没必要再去遍历句柄集 判断是否发生事件了);
⑤ Linux的实现中select返回时会将timeout修改为剩余时间,所以重复使用timeout需要注意。

select的缺点在于:
① 由于描述符集合set的限制,每个set最多只能监听FD_SETSIZE(在Linux上是1024)个句柄(不同机器可能不一样);
② 返回的可读集合是个fdset类型,需要对所有的监听读句柄一一进行FD_ISSET的测试来判断是否可读;
③ nfds的存在就是为了解决select的效率问题(select遍历nfds个文件描述符,判断每个描述符是否是自己关心的,对关心的描述符判断是否发生事件)。但是解决不彻底,比如如果只监听0和1000两个句柄,select需要遍历1001个句柄来检查事件。


3. epoll
int epoll_create(int size);
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout);

epoll 解决了select和poll的几个性能上的缺陷:①不限制监听的描述符个数(poll也是),只受进程打开描述符总数的限制;②监听性能不随着监听描述 符数的增加而增加,是O(1)的,不再是轮询描述符来探测事件,而是由描述符主动上报事件;③使用共享内存的方式,不在用户和内核之间反复传递监听的描述 符信息;④返回参数中就是触发事件的列表,不用再遍历输入事件表查询各个事件是否被触发。
epoll显著提高性能的前提是:监听大量描述符,并且每次触发事件的描述符文件非常少。
epoll的另外区别是:①epoll创建了描述符,记得close;②支持水平触发和边沿触发。

        
        
        
        
        

  • 27
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值