我有一个这样的课......
class Container {
public:
class Iterator {
...
};
...
};
在其他地方,我想通过引用传递一个Container :: Iterator,但我不想包含头文件。如果我尝试转发声明类,我会遇到编译错误。
class Container::Iterator;
class Foo {
void Read(Container::Iterator& it);
};
编译上面的代码给出了......
test.h:3: error: ‘Iterator’ in class ‘Container’ does not name a type
test.h:5: error: variable or field ‘Foo’ declared void
test.h:5: error: incomplete type ‘Container’ used in nested name specifier
test.h:5: error: ‘it’ was not declared in this scope
如何转发声明此类,以便我不必包含声明Iterator类的头文件?
答案 0 :(得分:121)
这根本不可能。您无法转发在容器外部声明嵌套结构。您只能在容器中转发声明它。
您需要执行以下操作之一
答案 1 :(得分:21)
我不相信在一个不完整的类上声明内部类的工作(因为没有类定义,没有办法知道实际上是否是一个内部类)。因此,您必须包含Container的定义,并使用前向声明的内部类:
class Container {
public:
class Iterator;
};
然后在单独的头文件中,实现Container :: Iterator:
class Container::Iterator {
};
然后#include只包含容器标题(或者不担心前向声明并且只包括两者)
答案 2 :(得分:1)
我知道无法完全按照您的意愿行事,但如果您愿意使用模板,这是一种解决方法:
// Foo.h
struct Foo
{
export template<class T> void Read(T it);
};
// Foo.cpp
#include "Foo.h"
#include "Container.h"
/*
struct Container
{
struct Inner { };
};
*/
export template<>
void Foo::Read<Container::Inner>(Container::Inner& it)
{
}
#include "Foo.h"
int main()
{
Foo f;
Container::Inner i;
f.Read(i); // ok
f.Read(3); // error
}
希望这个习惯用法对你有用(希望你的编译器基于EDG并实现导出;)。