一个简单(奇怪)的 C 语言问题

48 天前
 WilliamColton

使用的 IDE 是 CLion ,编译器是其自带的 MinGW 11.0 w64

下面的代码使用 chatgpt 加了注释,主要的问题是:

在直接给定输入

8

1 9 2 6 0 8 1 7

时,结果为 0.

而再次给定输入

8

1 9 2 6 90 8 1 7

时,结果还是 0.

而手动一个一个输入则正常显示 1.

虽然描述的很离奇,但是确实是这样,即:第一次的结果会影响第二次的结果,使之变成第一次的结果

我问了我们教 C 语言的老师,连他也不知道怎么回事,说可能是 CLion 的问题,但是这个答案并不能令我信服,

故来寻求各位 V 友帮助

求各位 V 友解答 QAQ

代码如下:

#include <stdio.h>

int main() {
    int n;
    scanf("%d", &n); // 从标准输入中读取一个整数,存储到变量 n 中

    int min;
    scanf("%d", &min); // 假设输入的第一个数为最小值,存储到变量 min 中
    int num;
    for (int i = 1; i < n; i++) { // 循环读取剩余的 n-1 个整数
        scanf("%d", &num); // 从标准输入中读取一个整数,存储到变量 num 中
        printf("%d\n", num); // 将读取的整数打印到标准输出,以换行符结束

        if (num < min) { // 检查当前读取的整数是否比最小值小
            min = num; // 更新最小值为当前读取的整数
        }
    }
    printf("%d", min); // 打印最小值到标准输出

    return 0; // 返回 0 表示程序正常结束
}

2651 次点击
所在节点    C
31 条回复
NessajCN
48 天前
没有复现,建议检查自身环境
PTLin
48 天前
这种问题自己打个断点调试一下比来这里问更快的。
ben666
48 天前
遇到问题
1. 单步调试
2. printf 打印调试

另外,注意一下 C 代码风格与规范,参考:
1. linux kernel 规范: https://www.kernel.org/doc/html/v4.10/process/coding-style.html
2. nginx 规范: https://nginx.org/en/docs/dev/development_guide.html#code_style

也可以参考 https://github.com/baidu/dperf/ 的代码风格
GeruzoniAnsasu
48 天前
奇怪问题通常是蠢原因。 比如开了两个窗口(真实经历)
WilliamColton
48 天前
@PTLin #2
@ben666 #3
@GeruzoniAnsasu #4
没有开两个窗口
同时调试了,第二次运行 scanf 收到的是第一次的结果。。。
WilliamColton
48 天前

可以看见明明是 0 ,却读入了 5a (即 90 )
mightybruce
48 天前
代码结果我没有复现, 给你一点提示
scanf() 并不是直接让用户从键盘输入数据,而是先检查缓冲区,处理缓冲区中的数据

scanf() 的这些特性都是有章可循的,其根源就是行缓冲区。

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

如果没有,就等待用户输入。用户从键盘输入的每个字符都会暂时保存到缓冲区,直到按下回车键,产生换行符\n ,输入结束,scanf() 再从缓冲区中读取数据,赋值给变量。
如果有数据,那就看是否符合控制字符串的规则

如果能够匹配整个控制字符串,那最好了,直接从缓冲区中读取就可以了,就不用等待用户输入了。
如果缓冲区中剩余的所有数据只能匹配前半部分控制字符串,那就等待用户输入剩下的数据。
smdbh
48 天前
scanf 是回车有效吧,如果是一行输入,就是 ```scanf("%d%d%d %d%d%d%d%d", &min[0],&min[1], .... &min[7], );``` 前后都是 8 个
bugcoder
48 天前
你在 return 一句前面加上 fflush(stdin); 把输入缓存清空一下。 一般 scanf 出问题都是缓存的问题。
GenericT
48 天前
Clion 有个运行配置有个“重定向输入自”,你是不是在里面放东西了
WilliamColton
48 天前
@bugcoder #9 无效

@GenericT #10 没有啊
aa514758835
48 天前
把中间文件删干净重新编译把,我有时候也会遇到明明是 false 但是走了 true 分支的情况
GenericT
48 天前
建议把整个 workspace 打包上传,不然没法判断
yolee599
48 天前
@smdbh #8 正解,按照 OP 给出的代码,应该这样输入:
8\n
1\n
9\n
2\n
6\n
0\n
8\n
1\n
7\n

而不是这样:
8\n
1 9 2 6 0 8 1 7\n
PTLin
48 天前
在纯净的终端里用 mingw 的 gcc/clang 编译然后运行,然后复现你的问题,整理成流程最后再来问吧。要不现在这样也没人可以复现,问也没意义。
sbldehanhan
48 天前
程序两次运行之间没有任何联系。所以,第一次影响第二次是不可能发生的。遇到问题先从自己身上找原因,相信计算机比人靠谱。
tool2d
48 天前
用 mingw gcc 编译了一下,没办法复现,结果倒是对的。
4
1 2 3 4
2
3
4
1
CEBBCAT
48 天前
尝试使用输入重定向来解决,我怀疑你复制粘贴的字符掺杂了不可见字符之类的。

例如,echo '8\n\n1 2 3\n' > t; cat t | od -xa; cat t | ./a.exe

已经会用调试器了啊,很好,这些奇奇怪怪的问题可以追查,相信最后要不发现哭笑不得的错误,要不然就是对计算机有更多了解。或者也可以切换到其他平台比如 Linux macOS 。

另外,论坛里面有很多半吊子的,网上也是,有的时候别太信。小马过河,尽量规避因为别人的鼠目寸光给自己带来的误导
MoYi123
48 天前
如果要在 clion 的 terminal 里输入, 推荐把 Emulate terminal in the output console 打开
araraloren
48 天前
@yolee599 `scanf` can handle the input correctly, you don't need enter line by line.

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/1026058

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX