和同事合作完成一个小项目,可以正常编译,运行阶段segmentation fault,经排查发现是我们在不同的.h文件里定义了相同名字的类,所以猜测是链接的时候链接到了对方的类中。
于是写了一个小例子验证:
// A1.h
#include <iostream>
class A {
public:
void PrintA() {
std::cout << " A1" << std::endl;
}
void PrintAA() {
std::cout << " AA1" << std::endl;
}
};
// A2.h
#include <iostream>
class A {
public:
void PrintA() {
std::cout << " A2" << std::endl;
}
};
// B1.cpp
#include <iostream>
#include "A1.h"
bool executeB1() {
A a;
a.PrintA();
a.PrintAA();
}
// B2.cpp
#include <iostream>
#include "A2.h"
bool executeB2() {
A a;
a.PrintA();
}
// C.cpp
bool executeB1();
bool executeB2();
int main() {
executeB1();
executeB2();
}
使用g++ -o main B2.cpp B1.cpp C.cpp ,运行输出:
A2
AA1
A2
使用g++ -o main B1.cpp B2.cpp C.cpp ,运行输出:
A1
AA1
A1
结论:链接的不是类,而是方法,类的作用类似命名空间。在链接阶段不会优先链接cpp自己的方法,而是和编译顺序有关,优先链接先编译的.o文件。
最重要的是,一个类或方法在程序中只能有一个定义,多个定义是未定义的,刚才总结的结论也许只在我的电脑上有效。