C的输入输出深入解析

输入输出(IO)是指用户和程序交互的方式,输入是指获取用户从键盘上输入的数据,输出是指将数据显示到屏幕上

缓冲区

缓冲区是内存空间的一部分,在内存中预留了一部分空间,用来暂时存储输入输出的数据

为什么会有缓冲区这个东西?

硬件速度远低于CPU,输入设备的响应时间通常是毫秒级别,而CPU处理指令的速度是纳秒级别

如果每次I/O操作都等待硬件完成,CPU会长时间空闲,导致资源浪费。若程序直接与硬件交互,每次I/O操作都会导致程序阻塞等待。有了缓冲区后,函数printf和scanf会将数据写入缓冲区,程序会继续执行下去,无需等待I/O口完成操作。根据缓冲区的缓冲策略,取出数据并刷新缓冲区

这样一来缓冲区既协同了硬件与CPU之间的速度,又提高了CPU 的效率

使用printf完整的输出流程

数据-》用户空间缓冲区-》缓冲区刷新-》标准库调用系统函数write()-》内核缓冲区-》操作系统内核调用显卡驱动设备-》驱动程序将数据写入显卡显存-》从显卡读取数据 刷新屏幕显示内容

缓冲策略

根据缓冲区刷新的时机可以将缓冲区分为全缓冲、行缓冲、不带缓冲

全缓冲

这种情况下,只有当数据填满缓冲区才能进行实际的输入输出操作,缓冲区的大小都有限制的,比如4K 、2K 、1K,数据量达到最大值才能从缓冲区取出数据并刷新

行缓冲

这种情况下,在输入输出的过程中只有遇到换行符才能从缓冲区取出数据,scanf就是带有行缓冲的

不带缓冲

这种情况下,对于输入输出的数据没有缓冲的地方,直接进行输入输出的操作。windows下的printf

总结:对于输入操作,缓冲区的存在是必要的

从缓冲区的角度分析输入函数scanf()的几个现象

#include <stdio.h>
int main()
{
    int a = 0, b = 0, c = 1, d = 2;
    scanf("%d", &a);  //输入整数并赋值给变量a
    scanf("%d", &b);  //输入整数并赋值给变量b
    printf("a+b=%d\n", a+b);  //计算a+b的值并输出
    scanf("%d %d", &c, &d);  //输入两个整数并分别赋值给c、d
    printf("c*d=%d\n", c*d);  //计算c*d的值并输出

    return 0;
}

从键盘上正确输入

12↙
60↙
a+b=72
10 23↙
c*d=230

连续输入

现在我们改变输入方式,四个变量的值连续输入,结果也能正确。说明连续输入的数据存放在缓冲区,scanf连续读取给对应的变量

12 60 10 23↙
a+b=72
c*d=230

多输入数据

结果也是正确

12 60 10 23 99↙
a+b=72
c*d=230

少输入数据

输入3个整数,前两个scanf把前两个整数读取了,执行到第三个scanf时,缺少一个数据,所以我们还是要继续输入,

12 60 10↙
a+b=72
23↙
c*d=230

数据类型不符合

前两个数据读取正常。执行到第三个scanf时,要求输入两个十进制整数,a10不符合,所以只能读取失败。说明scanf遇到不符合类型的数据,并不会等待用户输入

12 60 a10
a+b=72
c*d=2

总结:遇到scanf()函数时,程序会先检查输入缓冲区是否有数据。

  • 如果没有,就等待用户输入。用户输入的数据都会暂时保存到缓冲区(包括\n和空格)
  • 如果有数据,要么匹配成功,要么读取失败

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值