【问题标题】:scanf behavior of passing address of a pointer(address variable)传递指针地址的scanf行为(地址变量)
【发布时间】:2016-06-20 00:34:58
【问题描述】:

我知道这是使用 scanf() 的错误方法。但我希望 C 语言专家回答这个问题:

#include <stdio.h>
int main(int argc, char *argv[])
{
    int *a;
    printf("please input one number:\n");
    scanf("%d", &a);
    printf("your input is %d\n", a);
    return 0;
}

输出:

please input one number:
30
your input is 30

由于*a已经存在,执行时不会带来段错误。 scanf() 神奇地将正确的数据填充到 *a 中,将 (int *) 填充到 (int) 中;那么 printf( ) 就会正确输出。

我认为这段代码可以工作,因为 sizeof(int*) 和 sizeof(int) 在我的系统上是相同的值。所以我也会检查我系统上的一些类型大小:

sizeof(short) return 2
sizeof(short*) return 4
sizeof(int) return 4
sizeof(int*) return 4
sizeof(long) return 4
sizeof(long*) return 4
sizeof(float) return 4
sizeof(float*) return 4
sizeof(double) return 8
sizeof(double*) return 4
sizeof(char) return 1
sizeof(char*) return 4

似乎这段代码在浮动时也可以工作,但事实并非如此。我尝试的代码在这里:

#include <stdio.h>
int main(int argc, char *argv[])
{
    float *a;
    printf("please input one number:\n");
    scanf("%f", &a);
    printf("your input is %f\n", a);
    return 0;
}

输出:

please input one number:
3.5
your input is -0.034267

scanf() 应该可以工作,因为我在调用 scanf() 时使用了 %f。因为我仍然给它一个 4 字节的内存地址。

我想不通。了解整个 scanf() 事情的人能否解释为什么这两个代码(几乎相同)的工作方式不同?

【问题讨论】:

  • 我认为 float 在作为可变长度参数之一传递时被提升为 double
  • 您期望结果是什么,实际结果是什么?地址应该是一个整数,因此使用 %f 将其读取为浮点数似乎不正确。
  • 您调用了未定义的行为,它发生起作用了。
  • a 已经是一个地址 -> scanf("%d", a);%d 应该收到值 *a,而不是地址。
  • @AleksandarMakragić 这是使用scanf()printf() 的常用方法,但不适用于本实验。由于a没有初始化,很有可能会出现Segmentation Fault。

标签: c scanf


【解决方案1】:

通常float 类型的数据作为可变长度参数传递,正如@zch 所说,被提升为doubleprintf() 期望它。

在这种情况下,传递的是float*,它不会神奇地变成double

这就是printf() 得到错误数据的原因——数据和大小都是错误的。

【讨论】:

  • 对。一个 4 字节的 float * 被传递,但 printf 尝试读取一个 8 字节的双精度,以便将其打印为 %f
  • 谢谢,你是对的。我搜索整个论坛得到这个线程:stackoverflow.com/questions/210590/…
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2020-12-30
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2016-08-25
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多