非阻塞io(设置方式),fcntl(介绍,使用)

目录

非阻塞式io

一般

通用方法 -- fcntl()

介绍

函数原型

cmd

应用

示例

普通read

加入fcntl()


非阻塞式io

一般

我们之前使用过的recv()函数中,有一个flags参数,它可以设置io方式为非阻塞方式

其他函数也有,但每次使用都需要设置,未免有点麻烦,我们有更通用的方法

通用方法 -- fcntl()

介绍

实际上,操作文件时都是使用文件描述符fd

  • 每个fd都指向一个文件对象,对象中是有关于文件的属性的,其中就包括flags标志位 

如果直接在底层设置了非阻塞方式,所有文件操作都会是非阻塞的

  • 比如这里的fcntl,只要设置了就不用管了
  • 或是在调用open打开文件时就设置上O_NONBLOCK,原理也是一样的

函数原型

cmd

  • cmd值不同,后面追加的可变参数也不同
  • (这里说的状态标记就是前面提到的flags)

应用

实现一个将指定文件设置成非阻塞式io的函数:

示例

普通read
#include <iostream>
#include <sys/fcntl.h>
#include <unistd.h>
#include <stdio.h>

using namespace std;
int main()
{
    cout << "enter:";
    fflush(stdout);
    char buff[100];
    int ret = read(0, buff, sizeof(buff) - 1);
    if (ret > 0)
    {
        buff[ret - 1] = 0;
        cout << "echo: " << buff << endl;
    }
    else if (ret == 0)
    {
        cout << "end" << endl;
    }
    else
    {
        cout << "error" << endl;
    }
    return 0;
}

若我们不主动输入,就会一直阻塞在read函数里

加入fcntl()
#include <iostream>
#include <sys/fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>

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

using namespace std;
int main()
{
    cout << "enter:";
    fflush(stdout);
    set_no_block(0);
    char buff[100];
    while (true)
    {
        int ret = read(0, buff, sizeof(buff) - 1);
        if (ret > 0)
        {
            buff[ret - 1] = 0;
            cout << "echo: " << buff << endl;
        }
        else if (ret == 0)
        {
            cout << "end" << endl;
        }
        else
        {
            printf("error,errno : %d,strerror: %s\n", errno, strerror(errno));
            sleep(1);
        }
    }

    return 0;
}

F_GETFL -- 获取标志位

  • 返回值返回当前标志位(位图)

F_SETFL -- 设置标志位,一般是新增一个选项

  • 要将原标志位和新增选项进行位运算,将位运算的结果填入参数

设置好后,发现在没有数据时,走了返回值<0的情况

  • 也正常,毕竟既没有数据可读,也没有读到文件尾

如果打印出错误码和错误信息:

  • 错误码是11,表示临时资源不可用

当然,只要我们输入了数据,还是可以正常被上层读走的

那么,设置成非阻塞后,真的出错/没有数据可读都会走else,如何区分?

  • 通过错误码errno区分
  • 既然没有数据时会被设置为11,那就用这个来区分
  • 当然,如果判断语句里直接写11,未免有点挫,EWOULDBLOCK= EAGAIN=11,用这个来代替11

修改后:

#include <iostream>
#include <sys/fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>

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

using namespace std;
int main()
{
    cout << "enter:";
    fflush(stdout);
    set_no_block(0);
    char buff[100];
    while (true)
    {
        int ret = read(0, buff, sizeof(buff) - 1);
        if (ret > 0)
        {
            buff[ret - 1] = 0;
            cout << "echo: " << buff << endl;
        }
        else if (ret == 0)
        {
            cout << "end" << endl;
        }
        else
        {
            // printf("error,errno : %d,strerror: %s\n", errno, strerror(errno));
            if (errno == EWOULDBLOCK)
            {
                cout << "no data,try again" << endl;
                //如果没有就绪,我们可以在判断成功后做其他事情
            }
            else
            {
                cout << "read error" << endl;
            }
            sleep(1);
        }
    }

    return 0;
}

  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
非阻塞 I/O 操作是指在进行输入/输出操作时,不会阻塞程序的执行。在传统的阻塞 I/O 操作中,当程序调用读取或写入数据的函数时,如果数据还没有准备好或无法立即写入,程序会一直等待直到操作完成才能继续执行其他任务。这种等待会导致程序阻塞,无法同时处理其他任务或响应其他事件。 而非阻塞 I/O 操作则允许程序在进行输入/输出操作时立即返回,而不需要等待操作完成。如果数据还没有准备好或无法立即写入,非阻塞操作会立即返回一个错误码或特定的状态码,让程序可以继续执行其他任务或检查其他事件。 使用非阻塞 I/O 可以实现异步的 I/O 处理,使程序能够同时处理多个 I/O 操作或响应其他事件,提高程序的并发性和响应性能。在非阻塞模式下,程序可以定期轮询或使用回调函数等方式来检查 I/O 操作的状态,当数据准备好或可以写入时再进行实际的读取或写入操作。 需要注意的是,非阻塞 I/O 操作需要程序对返回的状态进行处理,并根据情况决定下一步的操作。这可能涉及到循环轮询、状态机设计、事件驱动等技术。同时,非阻塞 I/O 操作也可能会增加代码的复杂性,因为需要处理异步的操作和状态管理。 非阻塞 I/O 在网络编程中特别常见,可以用于同时处理多个网络连接或请求,提高系统的可扩展性和性能。在操作系统中,通常会提供相应的机制或函数来实现非阻塞 I/O,如使用 `fcntl()` 函数设置文件描述符的非阻塞属性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值