不同类型的Typedef重定义('structTGAME'与'structTGAME')
我有标题的编译错误。
我听说这可能是因为我给结构起的名字,但我多次将其更改为随机名称,但并没有修复它。
我的 IDE (CLion),显示 PLBR.c 中的错误
这是我的文件:
PLBR.h
#ifndef UNTITLED2_PLBR_H
#define UNTITLED2_PLBR_H
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct {
int players;
char diff[5];
int numpal;
char *palabras;
}TGAME;
PLBR.c
#include "PLBR.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct{
int players;
char diff[5];
int numpal;
char *palabras;
}TGAME;
在 main.c 中,我只有 #include "PLBR.h" 和一些代码,但没有任何关系。
回答
简单的部分
这是一个有趣的问题,因为完全回答它很复杂。让我们从简单的答案开始:为避免出现问题,不要在源文件中重新定义类型。在头文件中定义它并在源文件中包含头文件就足够了。
复杂的部分
现在是复杂的部分。typedef如果将名称重新定义为相同类型,则允许重复定义。C 2018 6.7 3 说:
……标识符的声明不得超过一个……除了:
— 可以重新定义 typedef 名称以表示与当前相同的类型,前提是该类型不是可变修改的类型;...
所以,如果你有typedef int foo;两次或typedef char bar[3];两次,那很好。你也可以吃typedef struct MyStruct foo;两次。
但是,您不能有typedef struct { int i; } foo;两次。即使这两个声明具有相同的文本,它们也声明了不同的类型。这是因为 C 2018 6.7.2.3 5 中的一条规则:
… 不包含标签的结构、联合或枚举类型的每个声明都声明了一个不同的类型。
这样做的一个原因是有时我们出于不同目的使用具有相同内容的结构。例如,我们可能有一个包含两个double值的结构用于复数(实部和虚部)和一个包含两个double值的结构用于平面中的点(x和y坐标):
typedef struct { double d[2]; } ComplexNumber;
typedef struct { double d[2]; } Point;
让编译器将它们视为不同的类型意味着它可以向我们发出有关错误的警告,例如将 aPoint作为参数传递给aComplex预期。
因此,每次重复定义时,您的结构都是不同的类型。
有一种方法可以多次引用同一个结构类型,那就是给它一个标签:
typedef struct MyTag {
int players;
char diff[5];
int numpal;
char *palabras;
} TGAME;
typedef struct MyTag {
int players;
char diff[5];
int numpal;
char *palabras;
} TGAME;
然后我们可以重新定义TGAME而不会出错;typedef struct MyTag TGAME;可能出现,甚至多次,并且它重新定义TGAME为相同的类型,struct MyTag,这是允许的。
但是,虽然可以重复 ,但typedef不能重复结构定义。如果你有两次:
编译器会抱怨。这是因为 C 2018 6.7.2.3 说:
一个特定类型的内容最多定义一次。
因此,尽管您可以typedef根据需要多次重复相同类型的a,但您不能重复完整的结构定义。要再次使用结构类型,您只能单独使用标签重复它,而不能使用其内容的完整定义。
综上所述,关于结构标签和类型的规则仍然不完整。仍然存在需要考虑的范围问题。使用struct MyTag新的范围内可以参考以前的类型或创建一个新的类型,这取决于如何使用它!
这是一个例子。以下得到一个错误,因为typedef TypeA TypeB;没有重新定义TypeB为它第一次定义的相同类型:
typedef struct Tag { int x; } MyType; // Create a new type.
void foo(void) // Start a new scope.
{
typedef struct Tag TypeA; // Define TypeA as previous type.
typedef struct Tag { int x; } TypeB; // Make a new type.
typedef TypeA TypeB; // Error, TypeA is not same as TypeB.
}
但插入struct Tag;改变了含义:
typedef struct Tag { int x; } MyType; // Create a new type.
void foo(void) // Start a new scope.
{
struct Tag; // Declare struct Tag to be some new type.
typedef struct Tag TypeA; // Define TypeA as new type.
typedef struct Tag { int x; } TypeB; // Define new type and define TypeB to be that type.
typedef TypeA TypeB; // Works, TypeA is same as TypeB.
(void) (TypeB *) 0; // (Avoid compiler warning about unused name.)
}