c++ - 与 const char * [] 声明关联的重复符号错误

标签 c++ linker

我很乐意帮助诊断我在尝试使用 g++ 4.2.1 进行编译时收到的重复符号错误的来源。

具体错误是

ld: duplicate symbol _SOCIODEM_FILENAMES in /var/folders/c+/c+eq1Qz1Feye7vxs5mQOUE+++TI/-Tmp-//ccP3yVgF.o and /var/folders/c+/c+eq1Qz1Feye7vxs5mQOUE+++TI/-Tmp-//cc1NqtRL.o 
collect2: ld returned 1 exit status

仅当我将此声明包含在名为 Parameters.h 的文件中时才会发生错误:

// Parameters.h

#ifndef PARAMETERS_H
#define PARAMETERS_H

// ...[code snipped]...
const int NUM_SOCIODEM_FILES = 5;
const char * SOCIODEM_FILENAMES[ NUM_SOCIODEM_FILES ] = { "LSPAN_PDF.txt", 
     "FLEDGE_PDF.txt", 
     "PAIR_PDF.txt", 
     "BIRTH_AGE_PDF.txt",  
     "SPLIT_PDF.txt"  };
// ...[code snipped]...
#endif

我搜索了所有文件,这是唯一声明 SOCIODEM_FILENAMES 的地方。当我注释掉声明时,“重复符号”错误消失了。

我不熟悉链接器错误(如果是这样的话),希望能帮助解决问题。我所有的头文件都有 #ifndef...#define...#endif 包装器。我的编译命令是

g++ -o a.out -I /Applications/boost_1_42_0/ Host.cpp Simulation.cpp main.cpp Rdraws.cpp

提前致谢。


解决方案总结

我现在在 Parameters.h 中有:

const char * const SOCIODEM_FILENAMES[ NUM_SOCIODEM_FILES ] = { "LSPAN_PDF.txt", 
                 "FLEDGE_PDF.txt", 
                 "PAIR_PDF.txt", 
                 "BIRTH_AGE_PDF.txt",  
                 "SPLIT_PDF.txt"  };

Parameters.h 中的所有其他定义和声明均保持不变。 Andrey 和其他评论者总结了一种使用 extern 的替代方法,这对我的目的来说太过分了。

最佳答案

出于某种原因,到目前为止,没有一个答案关心解释整数 NUM_SOCIODEM_FILES 对象和数组 SOCIODEM_FILENAMES 对象之间的区别。后者出于已经解释的原因触发链接器错误:因为您将头文件包含到多个实现文件中。然而,前者可以毫无问题地链接(因为 NUM_SOCIODEM_FILES 声明确实没有问题)。为什么?

这是因为您的 NUM_SOCIODEM_FILES 对象被声明为 const。在 C++ 中,const 对象默认具有内部链接,这意味着即使在多个实现文件中定义它们也不会导致链接问题。换句话说,在 C++ 中,您的 NUM_SOCIODEM_FILES 等同于

static const int NUM_SOCIODEM_FILES = 5; /* internal linkage */

这就是为什么它不会导致任何链接问题。

与此同时,您的 SOCIODEM_FILENAMES 未声明为常量,这就是它默认获得外部链接 并最终导致链接器错误的原因。但是,如果您也将 SOCIODEM_FILENAMES 声明为 const,问题就会消失

const char * const SOCIODEM_FILENAMES[ NUM_SOCIODEM_FILES ] = {
  ...

注意额外的 const 在声明中的位置。如果您只是添加额外的 const 并将其他所有内容保持原样(即保留头文件中的定义 if SOCIODEM_FILENAMES),链接器将不会报告错误,即使您将您的头文件包含到多个翻译单元中。

虽然这不是推荐的方法,因为那样您将提供 SOCIODEM_FILENAMES 内部链接,并最终在每个翻译单元中得到 SOCIODEM_FILENAMES 数组的独立拷贝 -一些可能工作正常但仍然没有什么意义的东西。因此,对于您的数组,通常最好使用其他答案中推荐的 extern 方法。

但是,请注意,您通常不应该为 NUM_SOCIODEM_FILES 声明这样做!!!它很好,在头文件中定义。除非你想做一些不寻常的事情,否则标量常量通常应该在头文件中用初始化程序定义 - 这样它们就可以被视为所有翻译单元中的编译时常量,这是一个相当有值(value)的拥有的东西。因此,请注意其他一些答案中出现的将 NUM_SOCIODEM_FILES 的定义也移动到 .cpp 文件中的奇怪建议 - 这实际上没有任何意义,而且是完全错误的事情做。

关于c++ - 与 const char * [] 声明关联的重复符号错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3088236/

相关文章:

C++ 返回 bool 总是假的?

c++ - 使用 vector 作为私有(private)/公共(public)成员的类设计?

c++ - 如何使函数能够获取指向不同类的指针?

linux - 链接到 libcurl 时 _end 符号消失

c++ - 加载共享库时出错

c++ - C++ 中的列表析构函数

c++ - 有没有办法让一个类的每个对象都有一个提升线程?

c++ - Qt - 使用预处理器指令检查 Windows 上链接的 .lib 的可用性

math - 为什么在 Math.h 中调用函数时会出现链接错误?

linux - 如何打印 ld(linker) 搜索路径