在终端上编译会发出指针警告和奇怪的符号
Compilation on Terminal gives off pointer warning and strange symbols
我正在尝试从我的终端编译 运行 一个简单的 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����������<�����$�������<���������������Ä������������__cstring�������__TEXT����������`������������`�����������������������������__unwind_info���__TEXT����������l�����H�������l������������������������������__eh_frame������__TEXT����������∏�����@�������∏��������������������������������Ë���__DATA��������������������������������������������������__nl_symbol_ptr�__DATA���������������������������������������������������__la_symbol_ptr�__DATA����������������������������������������������������H���__LINKEDIT������� ������������� ������`��������������������"��Ä0���� ����� ������������� �� ���@ ��0���������∏ ����� !��@���
我真的不知道发生了什么。我已经做了很长时间了,我知道这不应该那么难,但我真的不知道出了什么问题。有人知道会发生什么吗?
这里有两个主要错误:
- 您尝试打印变量
a
、b
和 c
的地址。
- 您调用了未定义的行为,但没有初始化您的
反 0.
你想打印整数,所以改变这个:
printf("%d %d %d", &a,&b,&c);
对此:
printf("%d %d %d", a, b, c);
我也加了一些空格,不过那只是为了美化代码,提高可读性,不影响执行。
您不想打印地址!这就是你收到警告的原因。在这里阅读更多:What is the difference between the * and the & operators in c programming?
正如 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
、b
和c
作为计数器!所以你 必须 将它们初始化为 0,否则你会开始将你的值添加到垃圾值中,这会调用 Undefined Behavior!
所以请改变这个:
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$
gsamaras 提到的点是正确的,你应该使用:
printf("%d %d %d", a, b, c);
您使用的语法用于
scanf(传入需要填充数据的变量地址)
在定义时也初始化变量。
关于编译:
名称 -o 指定编译后生成的输出文件,因此它会覆盖您的源代码(即 73.c)。
如果省略 -o,默认输出文件生成为 a.out。因此,对于编译,您可以使用以下和 运行 程序作为 ./a.out:
gcc 73.c
man gcc 引用:
-o file
Place output in file file. This applies to whatever sort of output is being produced, whether it be an executable file, an object file, an assembler
file or preprocessed C code.
If -o is not specified, the default is to put an executable file in a.out, the object file for source.suffix in source.o, its assembler file in source.s,
a precompiled header file in source.suffix.gch, and all preprocessed C source on standard output.
我正在尝试从我的终端编译 运行 一个简单的 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����������<�����$�������<���������������Ä������������__cstring�������__TEXT����������`������������`�����������������������������__unwind_info���__TEXT����������l�����H�������l������������������������������__eh_frame������__TEXT����������∏�����@�������∏��������������������������������Ë���__DATA��������������������������������������������������__nl_symbol_ptr�__DATA���������������������������������������������������__la_symbol_ptr�__DATA����������������������������������������������������H���__LINKEDIT������� ������������� ������`��������������������"��Ä0���� ����� ������������� �� ���@ ��0���������∏ ����� !��@���
我真的不知道发生了什么。我已经做了很长时间了,我知道这不应该那么难,但我真的不知道出了什么问题。有人知道会发生什么吗?
这里有两个主要错误:
- 您尝试打印变量
a
、b
和c
的地址。 - 您调用了未定义的行为,但没有初始化您的 反 0.
你想打印整数,所以改变这个:
printf("%d %d %d", &a,&b,&c);
对此:
printf("%d %d %d", a, b, c);
我也加了一些空格,不过那只是为了美化代码,提高可读性,不影响执行。
您不想打印地址!这就是你收到警告的原因。在这里阅读更多:What is the difference between the * and the & operators in c programming?
正如 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
、b
和c
作为计数器!所以你 必须 将它们初始化为 0,否则你会开始将你的值添加到垃圾值中,这会调用 Undefined Behavior!
所以请改变这个:
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$
gsamaras 提到的点是正确的,你应该使用:
printf("%d %d %d", a, b, c);
您使用的语法用于
scanf(传入需要填充数据的变量地址)
在定义时也初始化变量。
关于编译: 名称 -o 指定编译后生成的输出文件,因此它会覆盖您的源代码(即 73.c)。 如果省略 -o,默认输出文件生成为 a.out。因此,对于编译,您可以使用以下和 运行 程序作为 ./a.out:
gcc 73.c
man gcc 引用:
-o file Place output in file file. This applies to whatever sort of output is being produced, whether it be an executable file, an object file, an assembler file or preprocessed C code.
If -o is not specified, the default is to put an executable file in a.out, the object file for source.suffix in source.o, its assembler file in source.s, a precompiled header file in source.suffix.gch, and all preprocessed C source on standard output.