Portál AbcLinuxu, 12. července 2025 23:00


Dotaz: undefined reference to `vtable.. + abstraktní třídy

13.7.2010 00:52 pnemec | skóre: 8 | blog: jotter
undefined reference to `vtable.. + abstraktní třídy
Přečteno: 3289×
Odpovědět | Admin
Mám takový problém s kompilací svých zdrojáků. Kód implementuje grafické widgety, které jsou definovány v abstraktních třídách.

Část kódu je implementována v rodičích a část v "mém" rozšíření. Implementoval jsem všechny třídy označené jako pure virtual.

Kód jde bez problémů zkompilovat do object file. Ve chvíly kdy zkusím zkompilovat testovací spouštěcí kód (pouze zavolám new Object()) tak se na mne vyvalí mraky hlášek o chybějící referenci ve vtables.

Co se týče struktury kódu. Tak je tam jedna WidgetFactory, která má funkce createSomeWidget ...

hláška vypadá asi takto: tmp/ccejkFGc.o: In function `YJCheckBoxFrame::YJCheckBoxFrame(YWidget*, std::basic_string<char, std::char_traits<char>, std::allocator <char> > const&, bool)': YJCheckBoxFrame.cpp:(.text+0xcc): undefined reference to `vtable for YJCheckBoxFrame' /tmp/ccejkFGc.o: In function `YJCheckBoxFrame::YJCheckBoxFrame(YWidget*, std::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, bool)': YJCheckBoxFrame.cpp:(.text+0x11a): undefined reference to `vtable for YJCheckBoxFrame' /tmp/cc1DG5TS.o:(.data.rel.ro._ZTV15YJWidgetFactory[vtable for YJWidgetFactory]+0xc): undefined reference to `YJWidgetFactory::createLayoutBox(YWidget*, YUIDimension)'

kód kompiluji jednoduše z řádky:
g++ *.cpp -l yui -I/usr/include/YaST2/yui
V c++ jsem celkem nováček, ale co jsem se koukal co tahle chyba znamená, tak by to mělo být tak, že neimplementuji některé funkce, které předek, nebo jeden z předků označil jako pure virtual a čeka že budou vloženy do tabulky funkcí (vtable). Z chybových hlášek ale žádné chybějící funkce nevidím, tak mi to není moc jasné.

Tuhle chybu jsem ve fórech navíc viděl je v souvislosti s QT a chybějícím includem moc souboru. To ale není můj případ.

Budu vděčen za jakékoliv nakopnutí.


Řešení dotazu:


Nástroje: Začni sledovat (0) ?Zašle upozornění na váš email při vložení nového komentáře.

Odpovědi

13.7.2010 02:06 vase jmeno
Rozbalit Rozbalit vše Re: undefined reference to `vtable.. + abstraktní třídy
Odpovědět | | Sbalit | Link | Blokovat | Admin
Rekl bych, ze problem bude v tom, ze pure virtual metoda predka je sice potomkem deklarovana (tudiz kompilator nerve nic o instanciaci abstraktnich trid), ale chybi jejich implementace (tudiz rve az linker). Zkuste se podivat, jestli jsou vsechny deklarovane metody opravdu definovane a jestli se vsechny opravdu linkuji k vysledne binarce.
13.7.2010 10:45 vencas | skóre: 32
Rozbalit Rozbalit vše Re: undefined reference to `vtable.. + abstraktní třídy
Všechny třídy, které nejsou abstraktní, musejí mít (aby se jim vytvořila vtable) aspoň jednu virtuální metodu implementovanou ne-inline, v implementačním filu, který se linkuje. Podrobnějí GCC FAQ. Obvykle to řeším tak, že v hlavičce deklaruju virtuální destruktor a do implementace napíšu jeho implementaci, která je prázdná.
14.7.2010 14:39 pnemec | skóre: 8 | blog: jotter
Rozbalit Rozbalit vše Re: undefined reference to `vtable.. + abstraktní třídy
O vtables jsem si musel přečíst už před tím, takže tohle jsem tak nějak věděl.

Nicméně zmínka o destruktoru mě přivedla na zbytek řešení těch chybových hlášek.

Musím říci že mi ty abstraktní třídy v c++ vůbec nejsou jasné. V jave (můj "domovský" jazyk) je to takž že jsou interface ( vše je "pure virtual") abstract (nelze vytvořit instanci) a pak standardní třída. Zdá se že v c++ je možné do abstraktní třídy linkovat implementace až za běhu pomocí vtables?

Neznáte na to náhodou nějakou podrobnější dokumentaci?

14.7.2010 17:09 Sten
Rozbalit Rozbalit vše Re: undefined reference to `vtable.. + abstraktní třídy
Nemusí být explicitně v definičním souboru, ale musí být definované (nestačí deklarace) a definice, pokud je v hlavičkovém souboru, musí být použita (includována) nejméně v jenom definičním souboru
14.7.2010 14:35 pnemec | skóre: 8 | blog: jotter
Rozbalit Rozbalit vše Re: undefined reference to `vtable.. + abstraktní třídy
Původně jsem si myslel že jsou, ale nakonec dle upozornění od Stena jsem zjistil že ne.

Ten projekt (Yast UI) je trochu složitější na učení C++.
Řešení 1× (pnemec (tazatel))
13.7.2010 16:54 Sten
Rozbalit Rozbalit vše Re: undefined reference to `vtable.. + abstraktní třídy
Odpovědět | | Sbalit | Link | Blokovat | Admin
Chybějící funkce tam jsou, dokonce jsou i v chybových hláškách:
/tmp/cc1DG5TS.o:(.data.rel.ro._ZTV15YJWidgetFactory[vtable for YJWidgetFactory]+0xc): undefined reference to `YJWidgetFactory::createLayoutBox(YWidget*, YUIDimension)'
14.7.2010 14:24 pnemec | skóre: 8 | blog: jotter
Rozbalit Rozbalit vše Re: undefined reference to `vtable.. + abstraktní třídy
Chvíli jsem na to koukal a zjistil jsem že v polovině funkcí ve WidgetFactory chybý jméno class před jménem funkce. Holt jiný návyk z javy :)

Přidání classy tu chybovou hlášku pro YJWidgetFactory opravilo.
14.7.2010 14:33 pnemec | skóre: 8 | blog: jotter
Rozbalit Rozbalit vše Re: undefined reference to `vtable.. + abstraktní třídy
A zbytek těch hlášek byl chybějící konstruktor, který jsem z mladické nerozvážnosti připsal do YJCheckBoxFrame.h.

gcc má skutečně kryptické hlášky :)

Dík za radu
14.7.2010 14:41 pnemec | skóre: 8 | blog: jotter
Rozbalit Rozbalit vše Re: undefined reference to `vtable.. + abstraktní třídy
Odpovědět | | Sbalit | Link | Blokovat | Admin
Dík moc za rady, jako řešení jsem nakonec označil to co mě nejvíce nakoplo.

Založit nové vláknoNahoru

Tiskni Sdílej: Linkuj Jaggni to Vybrali.sme.sk Google Del.icio.us Facebook

ISSN 1214-1267, (c) 1999-2007 Stickfish s.r.o.