zoukankan      html  css  js  c++  java
  • Linux下的非阻塞IO(一)

    非阻塞IO是相对于传统的阻塞IO而言的。

    我们首先需要搞清楚,什么是阻塞IO。APUE指出,系统调用分为两类,低速系统调用和其他,其中低速系统调用是可能会使进程永远阻塞的一类系统调用。但是与磁盘IO有关的系统调用是个例外。

    我们以read和write为例,read函数读取stdin,如果是阻塞IO,那么:

    如果我们不输入数据,那么read函数会一直阻塞,一直到我们输入数据为止。

    如果是非阻塞IO,那么:

    如果存在数据,读取然后返回,如果没有输入,那么直接返回-1,errno置为EAGAIN

    我们用write做一个实验:

    #include <stdio.h>
    #include <stdlib.h>
    #include <string.h>
    #include <unistd.h>
    #include <sys/types.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    #include <sys/wait.h>
    #include <errno.h>
    #include <signal.h>
    
    char buf[500000];
    
    int main(int argc, const char *argv[])
    {
        int ntowrite, nwrite;
    
        ntowrite = read(STDIN_FILENO, buf, sizeof buf);
        fprintf(stderr, "read %d bytes
    ", ntowrite);
    
        activate_nonblock(STDOUT_FILENO, O_NONBLOCK);
    
        char *ptr = buf;
        int nleft = ntowrite; //剩余的字节数
        while(nleft > 0)
        {
            errno = 0;
            nwrite = write(STDOUT_FILENO, ptr, nleft);
            fprintf(stderr, "nwrite = %d, errno = %d
    ", nwrite, errno);
    
            if(nwrite > 0)
            {
                ptr += nwrite;
                nleft -= nwrite;
            }
        }
    
        deactivate_nonblock(STDOUT_FILENO);
    
        return 0;
    }

    该程序向标准输出写入500000个字节。

    如果使用:

    ./test < test.mkv > temp.file

    那么输出结果为:

    read 500000 bytes
    nwrite = 500000, errno = 0

    因为磁盘IO的速度较快,所以一次就可以写入,下面我们使用终端:

    ./test < test.mkv  2> stderr.txt

    这行命令将500000的内容打印到屏幕上,同时将fprintf记录的信息通过标准错误流写入stderr.txt。

    我们查看stderr.txt文件:

    read 500000 bytes
    nwrite = 12708, errno = 0
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = 11687, errno = 0
    nwrite = -1, errno = 11
    …………………………………………..

    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = -1, errno = 11
    nwrite = 1786, errno = 0

    采用命令统计了一下,总计read次数为15247次,其中返回-1次数为15203次,说明成功读取次数为44次。

    上面例子中,这种采用非阻塞IO的方式称为“轮询”,显然这是一种低效的方式,非阻塞IO通常与IO复用模型结合使用。

    另外,将fd设置为阻塞和非阻塞的函数代码如下:

    void activate_nonblock(int fd)
    {
        int ret;
        int flags = fcntl(fd, F_GETFL);
        if (flags == -1)
            ERR_EXIT("fcntl");
        flags |= O_NONBLOCK;
        ret = fcntl(fd, F_SETFL, flags);
        if (ret == -1)
            ERR_EXIT("fcntl");
    }
    
    
    void deactivate_nonblock(int fd)
    {
        int ret;
        int flags = fcntl(fd, F_GETFL);
        if (flags == -1)
            ERR_EXIT("fcntl");
        flags &= ~O_NONBLOCK;
        ret = fcntl(fd, F_SETFL, flags);
        if (ret == -1)
            ERR_EXIT("fcntl");
    }

    未完待续。

  • 相关阅读:
    网络资源(4)
    网络资源(3)
    网络资源(2)
    网络资源(1)
    OCP读书笔记(27)
    OCP读书笔记(26)
    OCP读书笔记(25)
    OCP读书笔记(24)
    OCP读书笔记(23)
    OCP读书笔记(22)
  • 原文地址:https://www.cnblogs.com/inevermore/p/4045344.html
Copyright © 2011-2022 走看看