鱼C论坛

 找回密码
 立即注册
查看: 1363|回复: 5

[已解决]scanf 的 缓存机制是什么

[复制链接]
发表于 2020-10-7 07:51:25 | 显示全部楼层 |阅读模式
10鱼币
本帖最后由 ArmandXiao 于 2020-10-7 16:20 编辑

提供一行数字(用-1结束)
例子:
1 2 3 4 5 -1

scanf("%d", &num);
循环读取数字, 是没有问题的

但是!!!
一但改成
scanf("%d ", &num);
* %d 后面多一个空格

就变成整行读取了??[不确定]
最后的赋值给 num 的也是一个数 不是一行数
很疑惑
最佳答案
2020-10-7 07:51:26
我记得以前论坛里出现过这个问题,好像不用回车,输入非法字符也行,比如-1 a。也可以终止scanf

最佳答案

查看完整内容

我记得以前论坛里出现过这个问题,好像不用回车,输入非法字符也行,比如-1 a。也可以终止scanf
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2020-10-7 07:51:26 From FishC Mobile | 显示全部楼层    本楼为最佳答案   
我记得以前论坛里出现过这个问题,好像不用回车,输入非法字符也行,比如-1 a。也可以终止scanf

评分

参与人数 1荣誉 +3 鱼币 +3 收起 理由
ArmandXiao + 3 + 3 谢谢 我知道如何修改 知识这个缓冲区域不太.

查看全部评分

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2020-10-7 07:57:48 | 显示全部楼层
本帖最后由 ArmandXiao 于 2020-10-7 07:59 编辑
#include <stdio.h>
int main()
{
    int number = 0;
    int sum = 0;
    int count = 0;
    int line = 0;
    
    printf("Enter number of lines:\n"); 
    scanf("%d",&line);
    
    while (line > 0){
        printf("Enter line %d (end with -1)\n", line--);
        while (number != -1){
            scanf("%d ",&number);                               // 问题出在这行 你们可以改一下尝试一下 本来预期是 一个 -1 结束 现在 必须输入两个 -1
            scanf("scanf = %d\n", number);
            printf("scanf = %d\n", number);
            if(number == -1){
                printf("Average = %.2f\n", (double)sum/count);
                sum = count = number = 0;
                break;
            }
            sum += number;
            count++;
        }
    }
    
    return 0;
}
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2020-10-7 08:36:25 | 显示全部楼层
本帖最后由 风过无痕1989 于 2020-10-7 08:37 编辑

键盘输入 scanf() 函数是严格按照你在引号内的格式从缓冲区读取字符。一般情况下引号之间只写格式化要求,格式化内容要和输入内容一致,数与数、字符与字符之间可以用空格隔开,也可以用逗号隔开,否则就会出错。除了格式化操作符以外,不加任何其他字符,连空格都不要加。

scanf () 在你输入一个数后,它是不会立即显示的,要等再接收到一个非(空格、制表符、回车)的输入scanf语句才结束。而且,最后输入的那个非(空格、制表符、回车)的东西是不会被这个scanf读进来的,而是留在输入流里。

以下是百度百科的内容:
scanf()是C语言中的一个输出函数。与printf函数一样,都被声明在头文件stdio.h里,因此在使用scanf函数时要加上#include <stdio.h>。(在有一些实现中,printf函数与scanf函数在使用时可以不使用预编译命令#include <stdio.h>。)它是格式输入函数,即按用户指定的格式从键盘上把数据输入到指定的变量之中。

函数原型

        int scanf(const char * restrict format,...);
函数 scanf() 是从标准输入流stdin (标准输入设备,一般指向键盘)中读内容的通用子程序,可以说明的格式读入多个字符,并保存在对应地址的变量中。
函数的第一个参数是格式字符串,它指定了输入的格式,并按照格式说明符解析输入对应位置的信息并存储于可变参数列表中对应的指针所指位置。每一个指针要求非空,并且与字符串中的格式符一一顺次对应。
返回值

scanf函数返回成功读入的数据项数,读入数据时遇到了“文件结束”则返回EOF。
如:
        scanf("%d %d",&a,&b);
函数返回值为int型。如果a和b都被成功读入,那么scanf的返回值就是2;
如果只有a被成功读入,返回值为1;
如果a和b都未被成功读入,返回值为0;
如果遇到错误或遇到end of file,返回值为EOF。end of file为Ctrl+z 或者Ctrl+d。
例:使用scanf函数输入数据。

#include <stdio.h>
int main(void)
{
    int a,b,c;
    printf("Give me the value of a,b,c seperated with whitespaces:\n");
    scanf("%d%d%d",&a,&b,&c);
    printf("a=%d,b=%d,c=%d\n",a,b,c);
    return 0;
}
&a,&b,&c中的&是寻址操作符,&a表示对象a在内存中的地址 ,是一个右值。变量a,b,c的地址是在编译阶段分配的(存储顺序由编译器决定)。
这里注意:如果scanf中%d是连着写的如“%d%d%d”,在输入数据时,数据之间不可以用逗号分隔,只能用空白字符(空格或tab键或者回车键)分隔——“2 (空格)3(tab) 4” 或 “2(tab)3(回车)4”等。若是“%d,%d,%d”,则在输入数据时需要加“,”,如“2,3,4”。
格式指令说明

format指向的字符串包含的格式指令由以下字符序列组成 :
*
表示读入的数据将被舍弃。带有*的格式指令不对应可变参数列表中的任何数据。
域宽
以一个非零的十进制整数形式出现。表示该格式指令最多读入的字符数。
格式说明符
c 读入域宽指定的数目个字符组成的字符序列(后面不会加上空字节),如果省略宽度则读入单字符。如%c或%1c读入单字符,%2c读入两个字符(后面不会加上空字节),以此类推。
s读入一个的字符序列,后面会加上空字节,遇到空白字符(\t \r \n 空格等)完成读取。
d 读入可选有符号(可选有符号表示输入时可以带符号也可以不带符号,不带符号则视为非负)十进制整数。输入格式应该像strtol函数的base实参为10调用时识别的字符序列一样。
u 读入无符号符号十进制整数。输入格式应该像strtol函数的base实参为10调用时识别的字符序列一样。
i 读入可选有符号整数。输入格式应该像strtol函数的base实参为0调用时识别的字符序列一样。
a,e,f,g,A,E,F,G 读入可选有符号浮点数,输入格式应该像strtod函数识别的字符序列一样。
o 读入可选有符号八进制整数。输入格式应该像strtoul函数的base实参为8调用时识别的字符序列一样。
x,X读入可选有符号十六进制整数。输入格式应该像strtoul函数的base实参为16调用时识别的字符序列一样。
p 读入一个指针值。读入的字符序列应该与fprintf的%p产生的字符序列形式相同。
n 不读入任何字符,而是把到该位置已读入的字符数存储到与之对应的int*指向的位置。本转换说明符如果带有*或者带有域宽信息(如:%*n或%3n等),则后果是未定义的。
扫描字符集合
% 读入% 符号(百分号)
无效的转换说明符将引起未定义的行为。
长度修饰符
hh与d, i, o, u, x, X, or n配合使用,表示对应一个signed char或unsigned char数据。
h与d, i, o, u, x, X, or n配合使用,表示对应一个short int或unsigned short int数据。
l 与d, i, o, u, x, X, or n配合使用,表示对应一个long int或unsigned long int数据;与a, A, e, E, f, F, g, or G&#63272;&#63272;配合使用表示对应一个double数据;与c,s,[配合使用表示对应wchar_t数据。
ll与d, i, o, u, x, X, or n配合使用,表示对应一个long long int或unsigned long long int数据。
j与d, i, o, u, x, X, or n配合使用,表示对应一个intmax_t或uintmax_t数据。
z与d, i, o, u, x, X, or n配合使用,表示对应一个size_t数据(或与size_t对应的有符号整型数据)。
t与d, i, o, u, x, X, or n配合使用,表示对应一个ptrdiff_t数据(或与ptrdiff_t对应的无符号整型数据)。
L 与a, A, e, E, f, F, g, or G&#63272;&#63272;配合使用,表示对应一个long double数据。
如果长度修饰符与格式说明符不匹配则引起未定义的行为。

评分

参与人数 1荣誉 +3 鱼币 +3 收起 理由
ArmandXiao + 3 + 3 谢谢 我知道如何修改 知识这个缓冲区域不太.

查看全部评分

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2020-10-7 16:46:38 | 显示全部楼层

我也是醉了,我第一句话就告诉你“键盘输入 scanf() 函数是严格按照你在引号内的格式从缓冲区读取字符。一般情况下引号之间只写格式化要求,格式化内容要和输入内容一致,数与数、字符与字符之间可以用空格隔开,也可以用逗号隔开,否则就会出错。除了格式化操作符以外,不加任何其他字符,连空格都不要加。”

还说问题没有解决,你有没有认真去看我的回复?

评分

参与人数 1鱼币 +5 收起 理由
ArmandXiao + 5 谢谢你 :)

查看全部评分

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2020-10-7 19:25:22 | 显示全部楼层
风过无痕1989 发表于 2020-10-7 16:46
我也是醉了,我第一句话就告诉你“键盘输入 scanf() 函数是严格按照你在引号内的格式从缓冲区读取字符。 ...

我其实早就知道了解决的方法 只是想了解下 buffer的机制 而且我也给你评分了。。 我把最优答案给了那个鱼油因为他回复的理由更贴切我了解的东西 不用再纠结了 我已经找到解决的方法了 谢谢你 :)
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-1-12 23:08

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表