Typedef适用于结构但不适用于枚举,仅适用于C++

mon*_*nts 7 c c++ enums gcc typedef

我有以下代码:

test_header.h:

typedef enum _test_enum test_enum;
enum _test_enum
{
    red,
    green,
    blue
};

typedef struct _test_struct test_struct;
struct _test_struct
{
    int s1;
    int s2;
    int s3;
};
Run Code Online (Sandbox Code Playgroud)

test.c的:

#include <test_header.h>
#include <stdio.h>

int main()
{
    test_struct s;
    s.s1=1;
    s.s2=2;
    s.s3=3;
    printf("Hello %d %d %d\n", s.s1, s.s2, s.s3 );
}
Run Code Online (Sandbox Code Playgroud)

test_cpp.cpp:

extern "C"{
    #include <test_header.h>
}
#include <stdio.h>

int main()
{
    test_struct s;
    s.s1=1;
    s.s2=2;
    s.s3=3;
    printf("Hello %d %d %d\n", s.s1, s.s2, s.s3 );
}
Run Code Online (Sandbox Code Playgroud)

注意我是如何以相同的方式对结构和枚举进行类型化.当我在直接C中编译时gcc -I. test.c -o test它运行正常,但是当用C++编译时gcc -I. test_cpp.cpp -o test_cpp,我得到以下错误:

./test_header.h:1:14: error: use of enum ‘_test_enum’ without previous declaration
Run Code Online (Sandbox Code Playgroud)

所以我的问题是双重的:为什么这在C中起作用而不是C++,为什么编译器接受结构但不接受枚举?

在enum上面声明结构时,我得到了相同的行为.我正在使用GCC 4.8.2.

小智 6

枚举是一个整数类型,编译器根据枚举值的范围选择确切的类型.所以你不能做一个enum的前向声明.


Ale*_*aev 5

ISO C标准禁止对enum类型进行前向引用.我不完全确定,但我想这是确认它的摘录:

6.7声明

1 [...]

约束

2声明应至少声明声明者(函数的参数或结构或联合的成员除外),标记或枚举的成员.

3 [...]

如果这是错误的,并且您知道哪个部分确切地引用了该问题,请更正我.

因此,您不能在C中使用枚举的前向声明.虽然,GCC允许它作为扩展,但如果启用-Wpedantic切换肯定会发出警告.

顺便说一下,你可以像这样写:

typedef enum {
  red,
  green,
  blue,
} test_enum;
Run Code Online (Sandbox Code Playgroud)

并且根据标准完全没问题,因此即使编译也是如此-Wpedantic -Werror.