1

我正在尝试从我的终端编译和运行一个简单的 C 程序。这是我的代码:

#include <stdio.h>
//integers that are 3n, 3n + 1, and 3n + 2

int main(){
  int n,i,a,b,c;
  scanf("%d", &n);
  for (;n>0;n--) {
    scanf("%d", &i);
    if(n%3==0){a+=1;}
    if(n%3==1){b+=1;}
    if(n%3==2){c+=1;}
  }
  printf("%d %d %d", &a,&b,&c);
  return 0;
}

编译一次后,它会生成以下警告:

warning: format specifies type 'int' but the argument has type 'int *' [-Wformat]

我不明白为什么它会生成该警告,因为我没有分配任何指针变量。为什么会生成它?

此外,在编译了几次之后,我的 c 文件变成了符号。我不明白为什么会这样。我的文件名是 73.c,这就是我在终端上编译它的方式:

gcc 73.c -o 73.c

之后,我的文件被转换成这样的东西(这行和行):

œ˙Ì˛����Ä��������Ö� ��������H���__PAGEZERO��������������������������������������������������������(��__TEXT����������������������������������������������������__text����������__TEXT����������0�����˝�������0���������������Ä������������__stubs���������__TEXT����������.������������.��������������Ä�����������__stub_helper���__TEXT����������<�����$�������&lt;���������������Ä������������__cstring�������__TEXT����������`������������`�����������������������������__unwind_info���__TEXT����������l�����H�������l������������������������������__eh_frame������__TEXT����������∏�����@�������∏��������������������������������Ë���__DATA��������������������������������������������������__nl_symbol_ptr�__DATA���������������������������������������������������__la_symbol_ptr�__DATA����������������������������������������������������H���__LINKEDIT������� ������������� ������`��������������������"��Ä0���� ����� �������������  �� ���@ ��0���������∏ ����� !��@���

我真的不知道发生了什么。我已经做了很长时间了,我知道这不应该那么难,但我真的不知道出了什么问题。有人知道会发生什么吗?

4

2 回答 2

4

这里有两个主要问题:

  1. 您尝试打印变量的地址ab并且c
  2. 您调用Undefined Behavior,但未将计数器初始化为 0。

你想打印整数,所以改变这个:

printf("%d %d %d", &a,&b,&c);

对此:

printf("%d %d %d", a, b, c);

我还加了一些空格,不过那只是为了美化代码,让代码更具可读性,不会影响执行。

您不想打印地址!这就是你收到警告的原因。在此处阅读更多内容:c 编程中的 * 和 & 运算符有什么区别?


正如 achelper 所提到的,您没有按照应有的方式编译程序(但这不是代码的问题),但这是您应该习惯的。我将演示我为编译程序所做的工作(我也喜欢使用 -Wall标志,它启用所有警告):

C02QT2UBFVH6-lm:~ gsamaras$ nano main.c
C02QT2UBFVH6-lm:~ gsamaras$ gcc main.c -Wall -o main
main.c:13:22: warning: format specifies type 'int' but the argument has type
      'int *' [-Wformat]
  printf("%d %d %d", &a,&b,&c);
          ~~         ^~
main.c:13:25: warning: format specifies type 'int' but the argument has type
      'int *' [-Wformat]
  printf("%d %d %d", &a,&b,&c);
             ~~         ^~
main.c:13:28: warning: format specifies type 'int' but the argument has type
      'int *' [-Wformat]
  printf("%d %d %d", &a,&b,&c);
                ~~         ^~
main.c:11:16: warning: variable 'c' is uninitialized when used here
      [-Wuninitialized]
    if(n%3==2){c+=1;}
               ^
main.c:5:16: note: initialize the variable 'c' to silence this warning
  int n,i,a,b,c;
               ^
                = 0
main.c:10:16: warning: variable 'b' is uninitialized when used here
      [-Wuninitialized]
    if(n%3==1){b+=1;}
               ^
main.c:5:14: note: initialize the variable 'b' to silence this warning
  int n,i,a,b,c;
             ^
              = 0
main.c:9:16: warning: variable 'a' is uninitialized when used here
      [-Wuninitialized]
    if(n%3==0){a+=1;}
               ^
main.c:5:12: note: initialize the variable 'a' to silence this warning
  int n,i,a,b,c;
           ^
            = 0
6 warnings generated.
C02QT2UBFVH6-lm:~ gsamaras$ ./main
2
1
2
1461300256 1461300252 1461300248C02QT2UBFVH6-lm:~ gsamaras$ 

一般来说,当有疑问时,将警告视为错误,所以让我们一起调试该代码,它会很有趣。专业提示:从第一个警告开始并解决它。在那之后,下一个警告可能是相关的,因此没有你真正阅读它们就消失了!

应用我的建议后,与您在问题中描述的内容相关的所有警告都消失了。只剩下未初始化的变量。

这很简单,只需将变量初始化为某个初始值,例如 -1,如果这当然有意义的话。

但是,在您的情况下,您使用a,bc作为计数器!因此,您必须将它们初始化为 0,否则您将开始将值添加到垃圾值中,这会调用未定义的行为

所以一定要改变这个:

int n,i,a,b,c;

对此:

int n, i, a = 0, b = 0, c = 0;

没有计数器的初始化,我现在在我的机器上

C02QT2UBFVH6-lm:~ gsamaras$ ./main
2
1
2
1505029184 1 1C02QT2UBFVH6-lm:~ gsamaras$

你看这printf()给了我一些垃圾,而在将我的计数器初始化为 0 之后,我得到了:

C02QT2UBFVH6-lm:~ gsamaras$ ./main
2
1
2
0 1 1C02QT2UBFVH6-lm:~ gsamaras$ 
于 2016-08-10T19:06:46.177 回答
2

gsamaras 提到的观点是正确的,你应该使用:

 printf("%d %d %d", a, b, c);

您使用的语法用于

  • scanf(需要填充数据的变量的传递地址。)

  • 定义时还要初始化变量。

关于编译: 名称 -o 指定编译后生成的输出文件,因此它会覆盖您的源代码(即 73.c)。如果省略 -o,则默认输出文件生成为 a.out。因此,对于编译,您可以使用以下命令并将程序作为 ./a.out 运行:

gcc 73.c

男人 gcc 报价

-o file 将输出放在文件文件中。这适用于正在生成的任何类型的输出,无论是可执行文件、目标文件、汇编文件还是预处理的 C 代码。

如果没有指定-o,默认是在a.out中放一个可执行文件,在source.o中source.suffix的目标文件,在source.s中它的汇编文件,在source.suffix.gch中的一个预编译头文件,以及标准输出上的所有预处理 C 源代码。

于 2016-08-10T19:53:59.477 回答