次の数行のコードで、より大きなプロジェクトで経験した動作を再現しました。読みやすさを向上させるために、#ifndef
ガードと#include
ディレクティブを省略しました。を呼び出すと、リンカ エラーが生成されますmake
。メイクファイルは質問の最後に含まれています。
クラス C は、A を継承する B を継承します。O はまったく異なるクラスです。
リンカーは不平を言います:
g++ -o main main.cpp -L. -lABC -lO
./libO.a(O.o): In function `O::foo(A)':
O.cpp:(.text+0x1f): undefined reference to `C::C(A const&)'
これがソースコードです。できるだけ小さく、読みやすいようにしました。何が問題なのですか?
/***** A.h *****/
class A
{
public:
A();
A(const A& a);
};
/***** A.cpp *****/
A::A() {}
A::A(const A& a) {}
/****** BC.h *******/
class B : public A
{
public:
B(const A& a);
};
class C : public B
{
public:
C(const A& a);
};
/******* BC.cpp ********/
B::B(const A& a) : A(a) {}
C::C(const A& a) : B(a) {}
/***** O.h *****/
class O
{
public:
void foo(A a);
};
/***** O.cpp *****/
void O::foo(A a)
{
C c(a);
}
主なものは次のとおりです。
/******* main.cpp *******/
int main()
{
A a;
O o;
o.foo(a);
return 0;
}
そして、ここにmakefileがあります:
%.o: %.cpp %.h
g++ -c $<
.PHONY: all
all: mklibs main
main: main.cpp
g++ -o $@ main.cpp -L. -lABC -lO
mklibs: libABC.a libO.a
libABC.a: A.o BC.o
ar -r $@ $^
libO.a: O.o
ar -r $@ $^