首页
学习
活动
专区
圈层
工具
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

linux c 非阻塞输入

基础概念

在Linux环境下,C语言中的非阻塞输入指的是程序在执行输入操作时,不会因为等待用户输入而阻塞(暂停)程序的其他部分。这种模式允许程序在等待输入的同时继续执行其他任务,从而提高程序的效率和响应性。

相关优势

  1. 提高响应性:程序可以在等待用户输入的同时处理其他任务,使得程序更加灵活和响应迅速。
  2. 多任务处理:非阻塞输入使得编写能够同时处理多个任务的程序变得更加容易。
  3. 避免死锁:在某些情况下,阻塞输入可能导致程序陷入死锁状态,而非阻塞输入可以有效避免这种情况。

类型与应用场景

  • 轮询(Polling):程序定期检查输入设备是否有数据可读。适用于对实时性要求不高的场景。
  • 信号驱动(Signal-driven I/O):当输入设备有数据可读时,操作系统发送一个信号通知程序。适用于需要快速响应的场景。
  • 异步I/O:程序发起输入请求后,立即返回继续执行其他任务,操作系统在数据准备好后通知程序。适用于高并发场景。

示例代码

以下是一个使用fcntl函数将标准输入设置为非阻塞模式的简单示例:

代码语言:txt
复制
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>

int main() {
    int flags;

    // 获取当前文件描述符的状态标志
    if ((flags = fcntl(STDIN_FILENO, F_GETFL, 0)) == -1) {
        perror("fcntl F_GETFL");
        exit(EXIT_FAILURE);
    }

    // 设置文件描述符为非阻塞模式
    if (fcntl(STDIN_FILENO, F_SETFL, flags | O_NONBLOCK) == -1) {
        perror("fcntl F_SETFL");
        exit(EXIT_FAILURE);
    }

    char buffer[100];
    while (1) {
        ssize_t bytesRead = read(STDIN_FILENO, buffer, sizeof(buffer));
        if (bytesRead == -1) {
            if (errno == EAGAIN || errno == EWOULDBLOCK) {
                // 没有数据可读,继续执行其他任务
                printf("No data available, doing other tasks...\n");
                sleep(1);
                continue;
            } else {
                perror("read");
                exit(EXIT_FAILURE);
            }
        } else if (bytesRead > 0) {
            buffer[bytesRead] = '\0';
            printf("Read: %s", buffer);
        }
    }

    return 0;
}

遇到的问题及解决方法

问题:为什么设置为非阻塞模式后,read函数会立即返回而不是等待输入?

原因:在非阻塞模式下,read函数不会等待输入设备准备好数据。如果没有数据可读,它会立即返回并设置errnoEAGAINEWOULDBLOCK

解决方法:在调用read函数后,检查返回值和errno。如果返回值为-1且errnoEAGAINEWOULDBLOCK,则表示当前没有数据可读,程序可以继续执行其他任务。

问题:如何优雅地处理非阻塞输入?

解决方法:可以使用轮询、信号驱动或异步I/O等方式来处理非阻塞输入。轮询是最简单的方法,但可能会浪费CPU资源。信号驱动和异步I/O可以更高效地处理输入,但实现起来相对复杂。

总结

非阻塞输入在Linux C编程中是一个重要的概念,它可以提高程序的响应性和效率。通过合理使用非阻塞模式和相关技术,可以编写出更加灵活和高效的应用程序。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

Linux编程(阻塞和非阻塞IO)

Linux设备驱动中的阻塞和非阻塞I/0,简单来说就是对I/O操作的两种不同的方式,驱动程序可以灵活的支持用户空间对设备的这两种访问方式。...非阻塞操作:在不能进行设备操作时,并不挂起,它或者放弃,或者不停地查询,直到可以进行操作。...非阻塞应用程序通常使用select系统调用查询是否可以对设备进行无阻塞的访问最终会引发设备驱动中 poll 函数执行。...,没有输入则进程挂起睡眠 if(res == 1) { printf("%c/n",buf); } 非阻塞地读一个字符: char buf; fd = open("/dev/ttyS1",O_RDWR...=1); //串口上没有输入则返回,所以循环读取 printf("%c/n",buf); 阻塞操作常常用等待队列来实现,而非阻塞操作用轮询的方式来实现。

5.5K20

linux阻塞与非阻塞(connect连接超时)

非阻塞connect详情介绍可以参见文章:https://blog.csdn.net/qq_41453285/article/details/89890429 一、非阻塞connect概述 man手册...解析文档,非阻塞connect如何使用: ①当我们将sock设置为非阻塞之后,使用connect去连接服务端,即使服务端开启了,connect系统调用也不会连接成功,connect而是以失败告终,并返回错误...②但是非阻塞connect返回的错误是有讲究的: 如果非阻塞connect返回的错误是EINPROGRESS,代表不是connect系统调用出错了,而是connect可能会在后面才会建立完整地连接(...,进一步来等待非阻塞connect客户端与服务端建立完整地连接,在等待的过程中,如果非阻塞connect建立成功了,客户端的sock_fd就会变成可写的(这个在本人的IO复用文章中介绍过,见下图) ④当非阻塞...于是在后面的select中等待非阻塞connect建立成功并且客户端fd变为可写的。

6.5K10
  • c# 非阻塞算法_c# – 了解非阻塞线程同步和Thread.MemoryBarrier

    if (_complete) { Thread.MemoryBarrier(); // Barrier 4 Console.WriteLine (_answer); } } } 我们讨论了是否有线程阻塞正在进行...另一方面,完全围栏只应该禁用指令重新排序和缓存,它的声音不符合线程阻塞的条件,(与锁定不同的是,它清除该线程等待其他人在继续之前释放锁定,并在此期间被阻止)时间) 关于那个线程’阻止状态’.我说的不是线程是否被置于阻塞状态...解决方法: 指令花费时间执行的事实并不意味着线程被阻止.当一个线程被特定地置于阻塞状态时被阻塞,而MemoryBarrier()不会这样做....标签:c,net,multithreading 来源: https://codeday.me/bug/20190521/1147704.html 发布者:全栈程序员栈长,转载请注明出处:https://

    43510

    【Linux】深入 Linux 进程等待机制:阻塞与非阻塞的奥秘

    (查看进程是否是正常退出) WEXISTATUS(status):如果WIFEXIED非0,提取子进程退出码。...如果任意时刻调用wait/waitpid,子进程存在且正常运行,则可能阻塞。 如果不存在该进程,则立即出错放回。...else { //father int status = 0; pid_t ret = 0; do { ret = waitpid(-1,&status,WNOHANG);//非阻塞等待...3.解释堵塞与非堵塞 阻塞场景:打电话等朋友接听 你拨打朋友的电话,直到朋友接通之前你什么都做不了。这就像阻塞调用,你必须等着事情完成。...非阻塞场景:发消息等待回复 你给朋友发了个消息,等他们回你。你不用一直盯着手机看,而是可以去做别的事情,等收到消息后再查看。这就像非阻塞调用,你不需要等着完成才能做其他事情。

    13510

    同步、异步、阻塞、非阻塞

    阻塞与非阻塞   应用进程请求I/O操作时,如果数据未准备好,如果请求立即返回就是非阻塞,不立即返回就是阻塞。简单说就是做一件事如果不能立即获得返回,需要等待,就是阻塞,否则就可以理解为非阻塞。...阻塞 阻塞调用是指调用结果返回之前,当前线程会被挂起。函数只有在得到结果之后才会返回。有人也许会把阻塞调用和同步调用等同起来,实际上它们是不同的。...非阻塞 非阻塞和阻塞的概念相对应,指在不能立刻得到结果之前,该函数不会阻塞当前线程,而会立刻返回。...同步/异步与阻塞/非阻塞的组合 同步阻塞形式: 等待执行结果是一直等待,执行时线程挂起(未对fd 设置O_NONBLOCK 标志位的read/write 操作) 同步非阻塞形式:等待执行结果是一直等待,...异步非阻塞形式:在处理消息是不等待,在执行消息是也不等待。

    3K40

    从linux源码看socket的阻塞和非阻塞 顶

    从linux源码看socket的阻塞和非阻塞 笔者一直觉得如果能知道从应用到框架再到操作系统的每一处代码,是一件Exciting的事情。 大部分高性能网络框架采用的是非阻塞模式。...笔者这次就从linux源码的角度来阐述socket阻塞(block)和非阻塞(non_block)的区别。 本文源码均来自采用Linux-2.6.24内核版本。...一个TCP非阻塞client端简单的例子 如果我们要产生一个非阻塞的socket,在C语言中如下代码所示: // 创建socket int sock_fd = socket(AF_INET, SOCK_STREAM...\非阻塞状态 我们用fcntl修改socket的阻塞\非阻塞状态。...阻塞后什么时候恢复运行呢 情况1:有对应的网络数据到来 首先我们看下网络分组到来的内核路径,网卡发起中断后调用netif_rx将事件挂入CPU的等待队列,并唤起软中断(soft_irq),再通过linux

    3.6K20

    同步异步,阻塞非阻塞

    什么是阻塞和非阻塞 阻塞和非阻塞是针对于进程在访问数据的时候,根据IO操作的就绪状态来采取的不同方式,阻塞方式下读取或者写入函数将一直等待,而非阻塞方式下,读取或者写入函数会立即返回一个状态值。...阻塞与非阻塞:针对函数(程序)运行的方式,在IO未就绪时,是等待就绪还是直接返回(执行别的操作)。...阻塞与非阻塞的区别: 阻塞是程序在调用系统函数IO时,在系统执行系统调用时由CPU通过轮询等方式来实现数据的IO。 非阻塞是在程序级别通过轮询/信号/事件的机制,去查看IO数据是否就绪。...可以是阻塞或非阻塞,阻塞则一直在等待内核/应用程序把IO数据准备好,非阻塞则是直接返回内核/应用程序是否已经准备好数据。 应用程序框架:同步或异步。...IO多路复用,同步,异步,阻塞和非阻塞 区别 关于异步,同步,阻塞与非阻塞 解读I/O多路复用技术

    3.2K60

    Java IO(3)非阻塞式输入输出(NIO)

    在上篇《Java IO(2)阻塞式输入输出(BIO)》的末尾谈到了什么是阻塞式输入输出,通过Socket编程对其有了大致了解。...现在再重新回顾梳理一下,对于只有一个“客户端”和一个“服务器端”来讲,服务器端需要阻塞式接收客户端的请求,这里的阻塞式表示服务器端的应用代码会被挂起直到客户端有请求过来,在高并发的应用场景有多个客户端发起连接下非阻塞式...接下来对于非阻塞式输入输出(NIO)的学习以及理解首先从它的三个基础概念讲起。 Channel(通道)   在NIO中,你需要忘掉“流”这个概念,取而代之的是“通道”。...5.2 非阻塞式网络编程(NIO Socket)   在通常情况下,对于网络编程用的比较多的还是阻塞式。非阻塞式在应用程序中并不是特别常见,但它在Tomcat等Web服务器中却很常见。...要使用Selector选择器,免不了大致会经过以下几个流程:创建Selector选择器;将Channel通道修改为非阻塞模式(只有Socket才能修改为非阻塞模式,FileChannel不能修改),并将通道注册至

    99980

    阻塞 & 非阻塞 | 同步 & 异步

    这里讲的都是基于IO的 阻塞、非阻塞、同步、异步 ---- 一个典型的IO操作包括了两个阶段,数据准备和数据读写。比如说现在要使用 recv 执行一个读操作,数据就绪就是远端是否有数据可读。...当IO工作在阻塞状态下的时候,如果数据没有就绪,recv就会阻塞当前线程;如果说IO工作在非阻塞状态下,会立即返回。...返回值-1的话,就说明连接出现问题,连接异常;如果返回值是0,且errno是EAGIN的话,就说明这是一个正常的非阻塞,返回数据未就绪状态。...一个同步IO接口的示例: char buf[1024]; int sz = recv(sockfd,buf,1024,0); //阻塞:一直在这儿死等 //非阻塞:时不时的回来问一下 if(sz>0)...---- 五种IO模型 阻塞: 非阻塞: 多路IO复用 信号驱动: 这里就完全放飞自我了 异步: ---- Reactor反应堆模型 One loop per thread

    2.9K10

    同步、异步、阻塞、非阻塞

    同步、异步、阻塞和非阻塞(网络编程) 同步 所谓同步,就是在发出一个功能调用时,在没有得到结果之前,该调用就不返回。 按照这个定义,其实绝大多数函数都是同步调用(例如sin, isdigit等)。...非阻塞 非阻塞和阻塞的概念相对应,指在不能立刻得到结果之前,该函数不会阻塞当前线程,而会立刻返回。...阻塞对象上可以有非阻塞的调用方式,我们可以通过一定的API去轮询状态,在适当的时候调用阻塞函数,就可以避免阻塞。而对于非阻塞对象,调用特殊的函数也可以进入阻塞调用。...同步阻塞:小明一直盯着下载进度条,到 100% 的时候就完成。 同步非阻塞:小明提交下载任务后就去干别的,每过一段时间就去瞄一眼进度条,看到 100% 就完成。...阻塞/非阻塞则是小明的等待方式,或者说 API 调用者的等待方式。 在不同的场景下,同步/异步、阻塞/非阻塞的四种组合都有应用。

    2.2K50

    阻塞与非阻塞的区别verilog_如何理解阻塞和非阻塞

    简单点说: 阻塞就是干不完不准回来, 非阻塞就是你先干,我现看看有其他事没有,完了告诉我一声 我们拿最常用的send和recv两个函数来说吧… 比如你调用send函数发送一定的Byte,在系统内部...:对于阻塞模式的socket send函数将不返回直到系统缓冲区有足够的空间把你要发送的数据Copy过去以后才返回,而对于非阻塞的socket来说send会立即返回WSAEWOULDDBLOCK告诉调用者说...:耗费着系统资源….对于非阻塞模式的socket该函数会马上返回,然后告诉你:WSAEWOULDDBLOCK—“现在没有数据,回头在来看看” 扩展: 在进行网络编程时,我们常常见到同步、异步、阻塞和非阻塞四种调用方式...非阻塞 非阻塞和阻塞的概念相对应,指在不能立刻得到结果之前,该函数不会阻塞当前线程,而会立刻返回。...阻塞对象上可以有非阻塞的调用方式,我们可以通过一定的API去轮询状态,在适当的时候调用阻塞函数,就可以避免阻塞。而对于非阻塞对象,调用特殊的函数也可以进入阻塞调用。

    2.3K20

    阻塞和非阻塞的实现

    我们可能都已经听过阻塞非阻塞的概念,本文以tcp中的connect系统调用为例子(基于1.12.13内核,新版的原理类似,但是过程就很复杂了,有时间再分析),分析阻塞和非阻塞是什么并且看他是如何实现的。...sock->state = SS_CONNECTED; // 返回成功 return(0); } 我们看到connect函数首先会调用tcp层的函数发送一个sync包,然后根据socket的属性(阻塞非阻塞...,可以通过setsocketopt设置)做下一步处理,如果是非阻塞,那么就比较简单,直接返回给应用层。...这也是非阻塞+事件驱动架构中的做法。因为这种架构下通常是单进程的,要避免阻塞进程,那么返回后什么时候才能知道连接成功呢?...以上就是进程阻塞和非阻塞的原理。

    2.2K20

    IO - 同步,异步,阻塞,非阻塞

    同步(synchronous) IO和异步(asynchronous) IO,阻塞(blocking) IO和非阻塞(non-blocking)IO分别是什么,到底有什么区别?...本文讨论的背景是Linux环境下的network IO。...blocking IO 在linux中,默认情况下所有的socket都是blocking,一个典型的读操作流程大概是这样: ?...最后,再举几个不是很恰当的例子来说明这四个IO Model: 有A,B,C,D四个人在钓鱼: A用的是最老式的鱼竿,所以呢,得一直守着,等到鱼上钩了再拉杆; B的鱼竿有个功能,能够显示是否有鱼上钩,所以呢...,B就和旁边的MM聊天,隔会再看看有没有鱼上钩,有的话就迅速拉杆; C用的鱼竿和B差不多,但他想了一个好办法,就是同时放好几根鱼竿,然后守在旁边,一旦有显示说鱼上钩了,它就将对应的鱼竿拉起来; D是个有钱人

    2.2K10

    同步异步阻塞非阻塞详解

    附c++代码 同步和异步 同步就是一个调用方发出请求开始,就一直处于等待状态,等待请求结果返回后才能继续执行其他任务。比如说调用一个函数,等待函数结果返回,这叫同步。...阻塞和非阻塞 对于阻塞和非阻塞,其实更关心的是进程的状态,如果函数返回结果之前,主进程被挂起,也就是处于阻塞状态,那这时候整个过程是阻塞的;如果结果返回之前,主进程状态是非阻塞的,那整个过程是非阻塞。...这时候关心的是你是否在等待结果,有没有不管这件事 阻塞与非阻塞: 你打电话问书店老板有没有《分布式系统》这本书,你如果是阻塞式调用,你会一直把自己“挂起”,直到得到这本书有没有的结果,如果是非阻塞式调用...在这里阻塞与非阻塞与是否同步异步无关。跟老板通过什么方式回答你结果无关。...这时候关心的是你是否立即有答复,你是否离开了书店 c++ socket阻塞与非阻塞代码 阻塞模式下的socket读取 int main(){ #创建和绑定socket,creat和bind函数

    2K50

    嵌入式Linux:阻塞式IO与非阻塞式IO

    在Linux系统中,I/O操作可以分为两种模式:阻塞式I/O和非阻塞式I/O。 这两种模式决定了进程在执行I/O操作时的行为方式,以及CPU资源的利用效率。...2 非阻塞式 I/O (Non-blocking I/O) 非阻塞I/O则是在执行I/O操作时,不管资源是否可用,操作系统都不会让进程进入阻塞状态,而是立即返回控制权。...非阻塞式 I/O 优点: 提高响应性,特别适用于需要实时交互的场景。 适合高并发服务器程序,如网络服务器,使用非阻塞I/O可以高效处理大量请求。...以鼠标输入设备文件为例,Linux中鼠标对应的设备文件通常位于/dev/input/目录下,命名为mouseX(X为序号)或eventX。...编写一个测试程序,使用阻塞式I/O读取鼠标的输入: int main() { int fd = open("/dev/input/event3", O_RDONLY); if (fd =

    8800
    领券