行客 发表于 2019-1-26 13:37:28

使用scanf从键盘输入数据一种奇特情况的解释

本帖最后由 行客 于 2019-1-26 17:21 编辑

请大家看以下代码
#include "stdio.h"

int main(int argc, char* argv[])
{
    int a, b=666;
    char string;
    printf("b=%d\n", b);
    scanf("%d", &a);
    scanf("%d", &b);
    //scanf("%s", &string);
    scanf("%s", string);//string前加不加&都是可以正常获取到键盘值的
    printf("a=%d, b=%d, string=%s\n", a, b, string);

        return 0;
}

如果我们键盘输入
1 2 www.fishc.com
然后回车,控制台会输出什么?

我们可以看到结果为:

b=666
1 2 www.fishc.com
a=1, b=2, string=www.fishc.com

这个没有问题。
但是如果我们键盘输入:
1 www.fishc.com bbs.fishc.com
然后回车,控制台会输出什么?

我们可以看到,输出结果为:
b=666
1 www.fishc.com bbs.fishc.com
a=1, b=666, string=www.fishc.com

那么问题来了,为什么会是这样的输出?

我们知道,scanf() 是从键盘读取数据,带有行缓冲区的。
程序执行到第一个 scanf() 时等待用户输入,我们从键盘输入数据“1 www.fishc.com bbs.fishc.com”,然后回车:
scanf() 匹配到 1,赋值给变量a,同时将内部的位置指针移动到 1 后面;
到了第二个 scanf(),缓冲区中有数据,会直接读取。此时缓冲区中的内容为 www.fishc.com,即使忽略开头的空格也不是 scanf() 想要的整数,所以匹配失败了,不会给变量 b 赋新值,所以b中依旧是默认值666;
匹配失败则意味着内部的位置指针不会移动,所以当第三个scanf()执行时,由于变量string需要的是一个字符串,此时缓冲区中的内容仍然是www.fishc.com,所以变量string就被赋值为www.fishc.com。

jackz007 发表于 2019-1-26 14:00:49

    楼主,你这句是不是有问题?

scanf("%s", &string);

    难道不应该改成这样吗?

scanf("%s", string);

行客 发表于 2019-1-26 14:05:38

jackz007 发表于 2019-1-26 14:00
楼主,你这句是不是有问题?




你可以试一下,这两个获取的地址是一样的。

行客 发表于 2019-1-26 14:10:56

jackz007 发表于 2019-1-26 14:00
楼主,你这句是不是有问题?




一维数组中,只有使用“&数组名”时,才是取数组首地址;直接打印数组名或者&数组名都是取得数组首元素地址

二维数组中:
"&数组名"表示这个数组的首地址。
“数组名”与"&数组名”表示的是数组的首元素地址,要注意的是,这个首元素地址是“第一个元素数组”的首地址。

jackz007 发表于 2019-1-26 14:26:22

本帖最后由 jackz007 于 2019-1-26 14:27 编辑

行客 发表于 2019-1-26 14:10
一维数组中,只有使用“&数组名”时,才是取数组首地址;直接打印数组名或者&数组名都是取得数组首元 ...

    是一样的吗?试试下面的代码就知道是否一样了

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

main(void)
{
    char * p1 , * p2 , s[] = "I Love FishC" ;
    p1 = s                                  ;
    p2 = & s                              ; // 此句无法通过编译
    printf("p1 = 0x%08x\n" , p1)            ;
    printf("p2 = 0x%08x\n" , p2)            ;
}

行客 发表于 2019-1-26 17:17:28

本帖最后由 行客 于 2019-1-26 17:31 编辑

jackz007 发表于 2019-1-26 14:26
是一样的吗?试试下面的代码就知道是否一样了

没有说你的有错误,只是说两个获取的地址是一样的,可以等同操作。而且用scanf有一个隐式转换。

1 s与&s都表示地址,且值相等。
2 s与&s类型不一样,s的类型为char *,&s的类型为char (*),即指向一行有10个字符的指针。
3 在scanf("%s",&s);语句中,如果非要认为两个不同,那么,&s至少有一个隐含的强制类型转换,即相当于scanf("%s",(char *)&s)。

再有,很重要的一点,你新发的这个帖子中char *p2只是指向了首字符元素,就是指向一个char字符,自然不可以直接获得数组的地址&s。

还有,带&和不带&的两种方案,编译器编译后的汇编是完全相同的,没有根本性区别,我的意思是,是不是别执拗与这个。

10:       scanf("%s", &string);
0040D778   lea         eax,
0040D77B   push      eax
0040D77C   push      offset string "%s" (004227f8)
0040D781   call      scanf (0040f9b0)
0040D786   add         esp,8



10:       scanf("%s", string);
00401068   lea         eax,
0040106B   push      eax
0040106C   push      offset string "%s" (00425038)
00401071   call      scanf (004010d0)
00401076   add         esp,8

这个讨论偏离了帖子的主旨。。。
页: [1]
查看完整版本: 使用scanf从键盘输入数据一种奇特情况的解释