使用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。
楼主,你这句是不是有问题?
scanf("%s", &string);
难道不应该改成这样吗?
scanf("%s", string); jackz007 发表于 2019-1-26 14:00
楼主,你这句是不是有问题?
你可以试一下,这两个获取的地址是一样的。 jackz007 发表于 2019-1-26 14:00
楼主,你这句是不是有问题?
一维数组中,只有使用“&数组名”时,才是取数组首地址;直接打印数组名或者&数组名都是取得数组首元素地址
二维数组中:
"&数组名"表示这个数组的首地址。
“数组名”与"&数组名”表示的是数组的首元素地址,要注意的是,这个首元素地址是“第一个元素数组”的首地址。 本帖最后由 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: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]