声明没有声明任何内容:警告?

时间:2014-02-06 18:06:57

标签: c struct warnings

#include <stdio.h>
#include <stdlib.h>
#include <conio.h>

int main()
{
    struct emp
    {
        struct address
        {
              int a;
        };
        struct address a1;
    };
}

此代码显示警告: -

警告:声明未声明任何内容(默认情况下已启用)

以下代码显示无警告

#include <stdio.h>
#include <stdlib.h>
#include <conio.h>

int main()
{
    struct emp
    {
        struct address
        {
             int a;
        }a1;
    };
}   

为什么'警告'仅显示在第一个代码中?

4 个答案:

答案 0 :(得分:9)

编译器显示警告的原因是因为它没有看到您为address结构定义的emp类型变量的名称,即使您在下一行使用address声明声明一些内容,但我想编译器不够聪明,无法弄明白。

如您所示,这会产生警告:

struct emp {
  struct address {}; // This statement doesn't declare any variable for the emp struct.
  struct address a1;
};

但不是这样:

struct emp {
  struct address {} a1; // This statement defines the address struct and the a1 variable.
};

或者这个:

struct address {};

struct emp {
  struct address a1; //the only statement declare a variable of type struct address
};

struct emp {}没有显示任何警告,因为此语句不在结构定义块内。如果你把它放在其中一个中,那么编译器也会显示一个警告。以下将显示两个警告:

struct emp {
  struct phone {};
  struct name {};
};

答案 1 :(得分:4)

显示警告的原因是第一个摘录不合适C - 它违反了约束条件,符合标准的C编译器必须生成一个diagnostisc消息。它违反了C11 6.7.2.1p2

  

<强>约束

     
      
  1. 未声明匿名结构或匿名联合的 struct-declaration 应包含 struct-declarator-list
  2.   

这意味着可以写

struct foo {
    struct {
          int a;
    };
};

因为内部struct声明了一个匿名结构,即它没有被命名。

但是在您的示例中,struct address有一个名称 - address - 因此必须在结束括号之后有一个声明者列表 - 例如{{ 1}}在您的示例中,或更复杂的a1

答案 2 :(得分:3)

结构定义的语法是:

struct identifier {
    type member_name;

    // ...

};

如果在结束大括号之后添加标识符,则表示使用该定义的结构声明变量。

在第一个示例中,编译器将address结构视为成员类型。就像你写的那样:

struct identifier {

    type ; // No member name is specified
    type a1;

    // ...

}

但在第二个例子中,您指定了成员名称:

struct identifier {

    type a1; // Member name specified

    // ...

}

以下是警告示例:http://ideone.com/KrnYiE

答案 3 :(得分:0)

如果您没有正确转发在其他命名空间中找到的声明类型,那么也会抛出该错误,从而也会出现此错误。

https://stackoverflow.com/a/19001722/1330381

示例

namespace A {

// forward declarations
namespace X {
namespace Y {
class MyType;
}
}

namespace B {

class Foo() {
    void methodBar(const X::Y::MyType& mt);
}
}