C++ c++;:标题函数未从库正确链接到exe
我在库中有一个头文件(alibrary.lib)。该库是一个静态库(.lib),它正确链接到exe 现在,我有一个类:Vector3dC++ c++;:标题函数未从库正确链接到exe,c++,linker,inline-functions,C++,Linker,Inline Functions,我在库中有一个头文件(alibrary.lib)。该库是一个静态库(.lib),它正确链接到exe 现在,我有一个类:Vector3d class Vector3d { void amethod() { blah } }; Vector3d cross(const Vector3d &v0, const Vector3d &v1) { float x,y,z; x = v0.y*v1.z-v0.z*v1.y;
class Vector3d
{
void amethod()
{
blah
}
};
Vector3d cross(const Vector3d &v0, const Vector3d &v1)
{
float x,y,z;
x = v0.y*v1.z-v0.z*v1.y;
y = v0.z*v1.x-v0.x*v1.z;
z = v0.x*v1.y-v0.y*v1.x;
return Vector3d(x,y,z);
}
Vector3d在头文件(Vector3d.h)中声明和定义。在类声明之后,我创建了交叉函数
lib compile是一个文件,但当它链接到unit test exe时,我得到以下错误:
flywindow.obj :error LNK2005: "class Vector3d __cdecl cross(class Vector3d const &,class Vector3d const &)" (?cross@@YA?AVVector3d@@ABV1@0@Z) already defined in fly.obj
有什么想法吗
谢谢最有可能的解释是,您的include文件中有代码(特别是交叉的定义),并且您的include文件由两个源文件包含,因此是双重定义 头文件中应该有声明,而不是定义。声明(表示某物存在)是类型定义、类声明、枚举等 定义(赋予存在的事物意义)是函数、变量定义等 您的交叉函数应在头文件中声明:
Vector3d cross(const Vector3d &v0, const Vector3d &v1);
但是定义在一个单独的源文件中。最可能的解释是,您的include文件中有代码(特别是交叉的定义),并且您的include文件由两个源文件包含,因此是双重定义 头文件中应该有声明,而不是定义。声明(表示某物存在)是类型定义、类声明、枚举等 定义(赋予存在的事物意义)是函数、变量定义等 您的交叉函数应在头文件中声明:
Vector3d cross(const Vector3d &v0, const Vector3d &v1);
但在单独的源文件中定义。如果定义自由函数(不是类的成员),则必须在单独编译的.cpp文件中定义,或在头文件中定义并标记为内联。因此,在您的情况下,您可以通过以下方式使其编译:
inline Vector3d cross(const Vector3d &v0, const Vector3d &v1) {
float x,y,z;
x = v0.y*v1.z-v0.z*v1.y;
y = v0.z*v1.x-v0.x*v1.z;
z = v0.x*v1.y-v0.y*v1.x;
return Vector3d(x,y,z);
}
导致此错误的原因是您在标头中有函数的定义,但尚未将其标记为内联。如果现在将该头包含到两个单独编译的文件中,那么链接器在尝试链接已编译的对象文件时将抛出一个错误,因为它会看到交叉函数被定义两次
它不需要为类的成员函数显式地内联,因为在类定义中定义的成员函数是隐式内联的
然而,通常在标题中定义函数并不是一个好主意。如果您的函数依赖于其他类型,而不仅仅是向量(在您的情况下,这很好,但当然有争议-有些人不喜欢),那么您需要包含这些类型的标题。这将不必要地膨胀头中间接包含的代码。相反,在这些情况下,您只需将函数声明放在标题中:
Vector3d cross(const Vector3d &v0, const Vector3d &v1);
但是在单独编译的.cpp文件中定义它。当然,应该删除内联
让我添加一个定义和声明的小列表,只是为了帮助您弄清楚声明和定义对于函数和类意味着什么。请注意,每个定义也是一个声明,但并非相反:
// class _declaration_ of boo
class boo;
// class _definition_ of foo.
class foo {
// member function _declaration_ of bar
void bar();
// member function _definition_ of baz
void baz() { }
};
// function _definition_ of fuzz
inline void fuzz() { }
// function _declaration_ of fezz
void fezz();
如果定义自由函数(不是类的成员),则必须在单独编译的.cpp文件中定义,或者在头文件中定义并标记为内联。因此,在您的情况下,您可以通过以下方式使其编译:
inline Vector3d cross(const Vector3d &v0, const Vector3d &v1) {
float x,y,z;
x = v0.y*v1.z-v0.z*v1.y;
y = v0.z*v1.x-v0.x*v1.z;
z = v0.x*v1.y-v0.y*v1.x;
return Vector3d(x,y,z);
}
导致此错误的原因是您在标头中有函数的定义,但尚未将其标记为内联。如果现在将该头包含到两个单独编译的文件中,那么链接器在尝试链接已编译的对象文件时将抛出一个错误,因为它会看到交叉函数被定义两次
它不需要为类的成员函数显式地内联,因为在类定义中定义的成员函数是隐式内联的
然而,通常在标题中定义函数并不是一个好主意。如果您的函数依赖于其他类型,而不仅仅是向量(在您的情况下,这很好,但当然有争议-有些人不喜欢),那么您需要包含这些类型的标题。这将不必要地膨胀头中间接包含的代码。相反,在这些情况下,您只需将函数声明放在标题中:
Vector3d cross(const Vector3d &v0, const Vector3d &v1);
但是在单独编译的.cpp文件中定义它。当然,应该删除内联
让我添加一个定义和声明的小列表,只是为了帮助您弄清楚声明和定义对于函数和类意味着什么。请注意,每个定义也是一个声明,但并非相反:
// class _declaration_ of boo
class boo;
// class _definition_ of foo.
class foo {
// member function _declaration_ of bar
void bar();
// member function _definition_ of baz
void baz() { }
};
// function _definition_ of fuzz
inline void fuzz() { }
// function _declaration_ of fezz
void fezz();