(Owed by: 春夜喜雨 http://blog.csdn.net/chunyexiyu)
例如一个头文件headfile.h这样写
#pragma once
bool Func (){return true;}
在这个头文件被多个地方包含的时候就会出问题,链接时报错: (FuncB报重定义)
“fatal error LNK1169: 找到一个或多个多重定义的符号”
原因是,在headfile.h中定义了函数及其实现,如果被包含时,则会把函数实现放入包含的位置,被包含多次时,则会被放入多次,从而导致Func重定义。
那怕是在头文件中使用了#pragma once或是#ifdef __xxx /#define __xxx/#endif也不能解决这种问题。原因是每个.cpp的编译都是独立的,对于每个cpp来说,都是包含了Func的声明和实现,所以在链接时就不清楚到底是链接哪一个同名函数了。
解决方法也很简单:
方法一: 添加inline标识,添加完inline标识后,函数的内容代码被直接解释到调用处了,链接时相当于不存在这个函数,也就不存函数重定义的情况。
Inline bool Func () {return true;}
不过inline是建议权,建议展开函数到调用处。(补充@2021-12)
编译器是否展开取决于编译器,通常编译器对短的函数接受展开建议,长的的函数不接受建议。
那么安全起见,可以配合static一块用,这样比较稳妥,也是许多开源软件中常见的方式。
static inline bool Func () {return true;}
方法二: 添加static标识
static bool Func () {return true;}
方法三: 放入到类中
class ClassA
{
public:
bool Func () {return true;}
}
方法四:放入到namespace中
namespace test
{
bool Func () {return true;}
};
对于方法二/方法三/方法四,个人觉得:
对于静态函数、类、namespace,无论有多少文件包含它,也只会维护一份,链接时也就只能找到这一份,所以也是没有问题。
(Owed by: 春夜喜雨 http://blog.csdn.net/chunyexiyu)