在C编程中,我遇到了一种情况,我不小心初始化了一个可变大小的数组并且工作正常。我做了一些研究,显然可变长度数组可以从C99编译中获得。显然,GCC 4.8.2的默认编译选项是C98。
以下是我用来测试的代码:
#include "stdio.h"
#include "stdlib.h" // rand(), srand()
#include "time.h"
void printArray(const char* c) {
// impossible to get size of c because it returns pointer size
int array[sizeof(c)/sizeof(char)];
int i;
for(i=0; i<(sizeof(c)/sizeof(char))-1; i++) {
int fill=-1;
if(c[i]=='a')
fill = 0;
else if(c[i]=='b')
fill = 1;
array[i]=fill;
}
printf("contents of array in binary: \n");
for(i=0; i<(sizeof(c)/sizeof(char))-1; i++) {
printf("%d, ", array[i]);
}
printf("\n");
}
void printRandomArray() {
srand(time(NULL));
// variable length array is possible using C99
int array[rand()%10];
int i;
printf("contents of random array: \n");
for(i=0; i<(sizeof(array)/sizeof(int)); i++) {
array[i]=rand()%10;
printf("%d, ", array[i]);
}
printf("\n");
}
int main(int argc, char* argv[]) {
char c[]="abbabbabbaababababababb";
printArray(c);
printRandomArray();
return 1;
}
printRandomArray()不应该工作,因为我使用GCC 4.8.2默认编译,这是C98,但它的工作原理。有人可以向我解释为什么会这样吗?
答案 0 :(得分:2)
除C99标准外,GCC还允许可变长度数组作为扩展名:
ISO C99允许使用可变长度自动数组,作为扩展,GCC在C90模式和C ++中接受它们。
AFAIK,GCC 4.8.2默认情况下以-std=gnu90
模式编译代码。使用选项-std=c89
对其进行编译,您将看到大量警告和错误。
答案 1 :(得分:1)
因此GCC支持两种不同版本的C89。它支持c89
和gnu89
。后者意味着启用了许多gcc扩展。
GCC 4.8.2的默认标准语言为gnu90
,与gnu89
相同。 [see gcc documentation]
让我们看一下使用这些不同语言时得到的不同警告/错误:
[2:10pm][wlynch@apple /tmp] /opt/gcc/4.8.2/bin/gcc asd.c
[2:10pm][wlynch@apple /tmp] /opt/gcc/4.8.2/bin/gcc -std=gnu89 asd.c
[2:10pm][wlynch@apple /tmp] /opt/gcc/4.8.2/bin/gcc -std=gnu90 asd.c
[2:10pm][wlynch@apple /tmp]
[2:10pm][wlynch@apple /tmp] /opt/gcc/4.8.2/bin/gcc -std=c89 asd.c
asd.c:2:21: warning: extra tokens at end of #include directive [enabled by default]
#include "stdlib.h" // rand(), srand()
^
asd.c: In function ‘printArray’:
asd.c:6:5: error: expected expression before ‘/’ token
// impossible to get size of c because it returns pointer size
^
asd.c:15:9: error: ‘array’ undeclared (first use in this function)
array[i]=fill;
^
asd.c:15:9: note: each undeclared identifier is reported only once for each function it appears in
asd.c: In function ‘printRandomArray’:
asd.c:26:5: error: expected expression before ‘/’ token
// variable length array is possible using C99
^
asd.c:30:24: error: ‘array’ undeclared (first use in this function)
for(i=0; i<(sizeof(array)/sizeof(int)); i++) {
^
[2:10pm][wlynch@apple /tmp] /opt/gcc/4.8.2/bin/gcc -std=gnu89 asd.c
[2:10pm][wlynch@apple /tmp]
如果我们修复了这些错误,然后使用-pedantic
进行编译,我们将会看到您要查找的诊断信息:
[2:28pm][wlynch@apple /tmp] /opt/gcc/4.8.2/bin/gcc -std=c89 -pedantic asd.c
asd.c: In function ‘printRandomArray’:
asd.c:27:5: warning: ISO C90 forbids variable length array ‘array’ [-Wvla]
int array[rand()%10];
^
asd.c:27:5: warning: ISO C90 forbids mixed declarations and code [-Wpedantic]
答案 2 :(得分:0)
首先,没有C98这样的东西。它是C89 / 90,然后是C95(通常合并到C89 / 90),然后是C99,依此类推。
其次,GCC编译器在其默认模式下根本不实现任何标准C.它正在编译的语言称为GNU C.默认情况下它支持VLA并没有什么不寻常之处。您必须手动配置GCC才能使其符合任何C标准。像-pedantic
或-pedantic-errors
这样的开关是必须的。