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\n", 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\n", 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");

}

 

未完待续。

你可能感兴趣的:(linux)