Gcc 4.8.2默认编译并运行可变长度数组

时间:2015-01-10 20:08:03

标签: c gcc variable-length-array

在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,但它的工作原理。有人可以向我解释为什么会这样吗?

3 个答案:

答案 0 :(得分:2)

除C99标准外,GCC还允许可变长度数组作为扩展名:

6.19 Arrays of Variable Length

  

ISO C99允许使用可变长度自动数组,作为扩展,GCC在C90模式和C ++中接受它们。

AFAIK,GCC 4.8.2默认情况下以-std=gnu90模式编译代码。使用选项-std=c89对其进行编译,您将看到大量警告和错误。

答案 1 :(得分:1)

因此GCC支持两种不同版本的C89。它支持c89gnu89。后者意味着启用了许多gcc扩展。

GCC 4.8.2的默认标准语言为gnu90,与gnu89相同。 [see gcc documentation]

让我们看一下使用这些不同语言时得到的不同警告/错误:

GNU89和GNU90

[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] 

C89

[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这样的开关是必须的。