Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/jsf-2/2.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
C++ c++;:标题函数未从库正确链接到exe_C++_Linker_Inline Functions - Fatal编程技术网

C++ c++;:标题函数未从库正确链接到exe

C++ 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;

我在库中有一个头文件(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;
      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();