非阻塞IO

文件描述符默认都是阻塞的IO
那么如何把一个fd设为非阻塞呢?需要用到fcntl函数:

非阻塞IO_第1张图片

参数解释:
①fd:需要操作的文件描述符
②根据cmd值得不同,决定后面追加的参数。
fcntl函数共有以下五种功能,分别对应五个cmd的值:
非阻塞IO_第2张图片

可以利用第三种功能,获取/设置文件状态标记,就可以将一个fd设为非阻塞,具体操作见代码:

#include 
#include 
#include 

int main()
{
    int f1 = fcntl(0,F_GETFL);//先将当前fd的属性取出记录在f1(位图)
    if(f1 < 0){
        perror("fcntl");
        return 1;
    }
    fcntl(0,F_SETFL,f1|O_NONBLOCK);//然后设置fd的属性,把O_NONBLOCK的对应位设为1,则变为非阻塞

    //代码走到这里,标准输入0已经变为非阻塞

    char buf[1024];
    read(0,buf,sizeof(buf)-1);

    return 0;
}

看运行结果:
这里写图片描述
当从标准输入读数据时,直接就返回了。

如果注释掉把0设为非阻塞的代码:
非阻塞IO_第3张图片
再运行代码:
这里写图片描述
此时就阻塞的等待我输入数据。

从上面的代码可以看到,当0被设为非阻塞时,则不会阻塞等待直接返回;
所以接下来我们需要轮询读取数据:

非阻塞IO调用:
1.不会阻塞的等待;
2.条件不满足时直接以出错形式返回;
3.错误码errno被设为eagain(11号)。

void SetNoBlock(int fd){
    int f1 = fcntl(0,F_GETFL);
    if(f1 < 0){
        perror("fcntl");
        return;
    }
    fcntl(0,F_SETFL,f1|O_NONBLOCK);
}

int main()
{
    SetNoBlock(0);//把标准输入0设为非阻塞

    while(1){//轮询
        char buf[1024] = {0};
        ssize_t s = read(0,buf,sizeof(buf)-1);
        if(s < 0){//当没有数据读入时,三秒一次报错
            perror("read");
            sleep(3);
            continue;
        }
        printf("> %s\n",buf);
    }

    return 0;
}

运行结果:

非阻塞IO_第4张图片

你可能感兴趣的:(IO)