Если два файла С++ имеют разные определения классов с одинаковым именем, тогда, когда они компилируются и связаны, что-то выкидывается даже без предупреждения. Например,
// a.cc
class Student {
public:
std::string foo() { return "A"; }
};
void foo_a()
{
Student stu;
std::cout << stu.foo() << std::endl;
}
// b.cc
class Student {
public:
std::string foo() { return "B"; }
};
void foo_b()
{
Student stu;
std::cout << stu.foo() << std::endl;
}
Если скомпилировать и связать вместе с помощью g++, оба будут выдавать "A" (если a.cc предшествует b.cc в порядке командной строки).
Аналогичная тема здесь. Я вижу, что пространство имен решит эту проблему, но я не знаю, почему компоновщик даже не снимает предупреждение. И если одно определение класса имеет дополнительную функцию, которая не определена в другом, скажем, если b.cc обновляется как:
// b.cc
class Student {
public:
std::string foo() { return "B"; }
std::string bar() { return "K"; }
};
void foo_b()
{
Student stu;
std::cout << stu.foo() << stu.bar() << std::endl;
}
Тогда stu.bar() работает хорошо. Спасибо всем, кто может сказать мне, как компилятор и компоновщик работают в такой ситуации.
В качестве дополнительного вопроса, если классы определены в файлах заголовков, должны ли они всегда быть обернуты безымянным пространством имен, чтобы избежать такой ситуации? Есть ли побочные эффекты?