文件描述符默认都是阻塞的IO
那么如何把一个fd设为非阻塞呢?需要用到fcntl函数:
参数解释:
①fd:需要操作的文件描述符
②根据cmd值得不同,决定后面追加的参数。
fcntl函数共有以下五种功能,分别对应五个cmd的值:
可以利用第三种功能,获取/设置文件状态标记,就可以将一个fd设为非阻塞,具体操作见代码:
#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>
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设为非阻塞的代码:
再运行代码:
此时就阻塞的等待我输入数据。
从上面的代码可以看到,当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;
}
运行结果: