我经历了无法理解的定义的崩溃。
示意性问题如下:
主项目文件有两个包括:
include <lib1.h>
include <lib2.h>
第一个标题包含来自库的其他几个标题,其中一个标题是直接(未包含名称空间)定义:
template<typename T> class SparseMatrix;
lib2.h里面有以下内容
namespace lib2
{
using namespace lib3;
class ...
{
...
SparseMatrix<double> ...
...
}
}
在lib3中,用命名空间覆盖,还有SparseMatrix类的定义。
每个库分别编译没有问题。当我尝试编译使用它的可执行文件时,编译器会产生错误:
lib2.h:70:7: error: reference to 'SparseMatrix' is ambiguous
这对我来说很奇怪,因为我写的主程序中没有任何地方
using namespace lib3;
因此,我认为这些定义不应该崩溃的原因。 我非常感谢对此问题的任何可能的解释。
当然,我可以将lib1中的定义附加到它们自己的命名空间中,但是我需要在那里修改很多文件,我宁愿不这样做。
COMMENT: 下面的答案是正确的,但我也能够通过改变包含文件的顺序来解决问题,即首先包括lib2,然后是lib1。
答案 0 :(得分:6)
在主程序中我没有写
using namespace lib3;
但如果你看一下lib2.h
,那就完全是这样了。 lib3
命名空间的内容已被引入lib2
,现在在定义SparseMatrix<double>
对象时可见。
在解决所有包含后,您可以将lib2.h
视为此类内容:
template <typename T> class SparseMatrix; // (1)
namespace lib3
{
template <typename T> class SparseMatrix; // (2)
}
namespace lib2
{
using namespace lib3; // (3)
class ...
{
...
SparseMatrix<double> ... // (4)
// ::SparseMatrix<double> would only see (1)
// lib2::SparseMatrix<double> would only see (2)
}
}
标记为(1)的行声明SparseMatrix
,它在第(4)行立即可见。第(2)行的声明不会,但由于第(3)行将它带入命名空间lib2
,现在它也可以在第(4)行看到。
你可以通过完全限定类型来解决这个问题:
::SparseMatrix<double> ...
没有前面名称空间的 ::
表示全局名称空间。
另一种方法是在using namespace lib3;
中没有lib2.h
并正确限定lib3
命名空间的内容。