admin管理员组

文章数量:1533099

2024年3月20日发(作者:)

already defined in *.obj

VC6 如果想在 stdafx.h

符“号已定义 ”问题原理及解决方案

中定义全局变量,由于该头文件会被 include 多次,

。 所以,经常会出现以下经典的错误: already defined in

解决方法:把该变量的定义 int g_flag

extern 一下。假如你在

放到 中,然后在使用的地方

中使用了该变量 g_flag ,那么就在

extern int g_flag;

的首部,构造函数的定义之外,添加上

许多 Visual C++ 的使用者都碰到过 LNK2005:symbol already defined

LNK1169:one or more multiply defined symbols found 这样的链接错误,

而且通常是在使用第三方库时遇到的。对于这个问题,有的朋友可能不知其然,

而有的朋友可能知其然却不知其所以然, 那么本文就试图为大家彻底解开关于它

的种种疑惑。

大家都知道,从 C/C++ 源程序到可执行文件要经历两个阶段 :(1)编译器将源

文件编译成汇编代码, 然后由汇编器 (assembler) 翻译成机器指令 (再加上其它相

关信息 )后输出到一个个目标文件 (object file,VC 的编译器编译出的目标文件默

认的后缀名是 .obj) 中;(2) 链接器 (linker) 将一个个的目标文件 (或许还会有若干程

序库 )链接在一起生成一个完整的可执行文件。

编译器编译源文件时会把源文件的全局符号 (global symbol) 分成强 (strong)

和弱 (weak) 两类传给汇编器,而随后汇编器则将强弱信息编码并保存在目标文

件的符号表中。那么何谓强弱呢?编译器认为函数与初始化了的全局变量都是强

符号,而未初始化的全局变量则成了弱符号。比如有这么个源文件

extern int errorno;

:

int buf[2] = {1,2};

int *p;

int main()

{

return 0;

}

其中 main 、 buf 是强符号, p 是弱符号,而 errorno

个外部变量的使用声明。

有了强弱符号的概念,链接器 (Unix 平台 )就会按如下规则 (参考 [1] ,

p549~p550) 处理与选择被多次定义的全局符号 :

(即不同的目标文件中不能有同名的强符号 );

则非强非弱,因为它只是

规则 1: 不允许强符号被多次定义

规则 2: 如果一个符号在某个目标文件中是强符号,在其它文件中都是弱符号,

那么选择强符号;

规则 3: 如果一个符号在所有目标文件中都是弱符号,那么选择其中任意一个;

虽然上述 3 条针对的是 Unix 平台的链接器,但据作者试验,至少 VC6.0 的

linker 也遵守这些规则。 由此可知多个目标文件不能重复定义同名的函数与初始

化了的全局变量,否则必然导致 LNK2005 和 LNK1169 两种链接错误。可是,

有的时候我们并没有在自己的程序中发现这样的重定义现象, 却也遇到了此种链

接错误,这又是何解?嗯,问题稍微有点儿复杂,容我慢慢道来。

众所周知, ANSI C/C++ 定义了相当多的标准函数,而它们又分布在许多不

本文标签: 定义符号文件目标问题