C++:文件与自身冲突

C++: File conflicting with itself

本文关键字:冲突 文件 C++      更新时间:2023-10-16

>我创建了自定义随机数生成器,并将它们的全局函数放在一个名为SamRandom.h的文件中。该文件如下所示:

#ifndef _SAM_RANDOM_H
#define _SAM_RANDOM_H
#include <cstdlib>
#include <ctime>
void InitialiseRandomSeed()
{
    //...
}
//...
#endif

我正在一个非常复杂的面向对象程序中工作,该程序从这里和那里有许多不同的组件。每次添加与此文件相关的文件时,都会收到以下冲突消息:

LaserBase.obj:-1: error: LNK2005: "void __cdecl InitialiseRandomSeed(void)" (?InitialiseRandomSeed@@YAXXZ) already defined in main.obj
error: LNK1169: one or more multiply defined symbols found

在MSVC和MinGW上,我得到:

In function `Z20InitialiseRandomSeedv':
SamRandom.h:8: multiple definition of `InitialiseRandomSeed()'
error: first defined here
:-1: error: collect2: ld returned 1 exit status

为什么会这样?我认为预处理器指令应该可以防止此问题发生......这真的让我发疯!!

PS:问题与函数名称无关。

感谢您的任何努力

当您将文件包含在.cpp文件中时,预处理器会直接将其复制到该文件中。因此,当您编译该.cpp文件时,目标文件(即使您不认识编译器也始终执行此步骤)将包含该函数。如果将其包含在多个.cpp文件中,则每个对象文件都将包含该函数。当链接器尝试将所有对象文件链接到可执行文件时,它会找到函数 x 次并给出该错误。"找到一个或多个多重定义的符号"表示它不止一次找到您的一个函数(或变量),并且不知道如何处理它。

要避免这一点,您应该只放置您的功能的签名,例如

//whatever.h
#ifndef _SAM_RANDOM_H
#define _SAM_RANDOM_H
void InitialiseRandomSeed();
#endif

并将实现放入单独的.cpp文件中:

//whatever.cpp
#include "whatever.h"
#include <cstdlib>
#include <ctime>
void InitialiseRandomSeed()
{
    //...
}

标头保护确保不会在单个编译单元(.cpp文件)中多次定义函数。但是,它们不会阻止在多个编译单元中一次性定义函数。这是因为标头包含函数的定义,因此包含标头的每个.cpp都有自己的函数定义。

您可以将定义移动到 SamRandom.cpp并在标头中声明函数,如下所示:

void InitialiseRandomSeed();

或者,您可以指定该函数是内联的,这将使多个定义可以接受:

inline void InitialiseRandomSeed()
{
    //...
}

假设该函数不是性能关键型的,并且不经常被调用,我会使用前一种方法。这使标头更具可读性,并且(通常)减少了编译时间。

您可以将

定义分离到一个.cpp文件中,只需在标头中声明该函数。