调用程序中对库类成员的未定义引用错误

2024-04-27

下面添加了其他问题,2011 年 4 月 11 日

我正在用 C++ 开发一组跨平台的共享库 DLL/Sos 和测试程序,尽管我必须能够支持 C。这些库将仅作为目标代码发布,但测试程序将随源代码一起发布,因此我们的客户可以获得示例代码。因此,我正在设计要在运行时加载的库,即使用 dlopen()/LoadLibraryA() 进行动态链接。

我在 Umbutu 10.04 上使用 g++ 4.4.3-4,在 Vista/64 上使用 VC++ 2008(32 位模式)。

一切似乎在 Windows 上运行得很好(现在)。然而,当我在 Linux 上编译时,我遇到了一些我无法弄清楚的错误。

测试器和库有几个用多个 .cpp 和 .h 编码的类。除了主要入口点之外,库中的类和大多数内容都位于命名空间 DISCOVER_NS 中。

以下是该项目的简要概述:

首先,承认,我缩短了一些名称,以便代码更具可读性。

发现.cpp

  • 创建一个带有指向它的指针的类对象,称为 DiscoverObject 类型的 theMainObject。

  • 有一个 extern“C”函数,将 MainObject 作为 void* 返回给调用者程序。

  • DiscoverObject 有多种方法,并实例化在单独的 cpp 和 .h 中找到的其他类。一种特殊的方法名为 Hello(),它会执行您所期望的操作,它会打印一条“hello”测试消息。


测试程序.cpp

  • 获取库的句柄

  • 获取指向返回 theMainObject 的函数的函数指针。

  • 执行函数(指针)并将返回的地址从 void* 转换为 DISCOVER_NS::DiscoverObject* aDiscoverObject。

  • 运行aDiscoverObject->Hello()。


我编译:

抄送=@g++

gflags = -g3

cflags = -fPIC -Wall -pedantic

lib_linkflags := -shared -fPIC -lstdc++ -lrt -lpthread -rdynamic

tester_linkflags := -ldl -lpthread

定义 = -D_linux_ -D_DEBUG -D_IPC_ARCH_INTEL=1 -D_THREAD_SAFE


现在,当我编译时,我收到以下错误: *Tester.cpp:142: 对 `Discover_NS::DiscoverObject::hello()' 的未定义引用*

我还从 discovery.so 收到了一堆其他未定义的引用错误,例如: *discover.so:对 `Discover_NS::DeviceList::~DeviceList()* 的未定义引用


我已经尝试过在SO extern“C”中制作几乎所有内容。没有不同。

我尝试将语句放入 discovery.cpp 中,如下所示: extern void Discover_NS::OtherClass::method( args ); 但这给了我关于“类外声明不是定义”错误的错误。


我知道查看代码会有所帮助,但我需要时间敲出一些小东西来发布。

任何人都可以提供解决这个混乱的想法吗?

Thanks,

Wes

德米特里的解决方案并不完全是解决方案的全部,但却是解决方案中的必要元素。在检查我的 makefile 时,我发现了几行无意中重复的行(我将其删除),以及两个“打字错误”,其中 -o 的编码路径错误,并被编码到了编译步骤中。断断续续的步骤编译了logger.cpp和RemException.cpp:

./common/logger.o : ./common/logger.cpp
    $(CC) $(gflags)  $(cflags) -c  $(defines)  -I ./common  
        -I ./EdgeIO  -I ./Discover   
        -o ./common/Debug/logger.o   <+++++++++ path to .o was wrong
        ./common/logger.cpp   2>&1  | tee ./RemKonTester/logger.ERR

然后我发现了真正的错误。我完全忽略了这样一个事实:我没有在 Discover 目录中编译所有 .cpp!。花了一个小时才删除所有的小问题,但现在她从 makefile 中恢复过来了。

原始问题的新版本:现在我知道它可以通过 makefile 工作,我如何让 Eclipse 执行与 makefile 相同的操作?

谢谢德米特里。

Wes

嗯,我的问题仍然在这里。

我已经根据 Dmitry (@Dmitry) 的建议编译了我的代码。只是,它们似乎引起了一个单独的问题。我希望我的库在运行时动态链接到主测试程序。添加-l 发现 -l EdgeIO到链接可以编译所有内容,但它给了我静态链接。

仅供参考,未使用的“pi”是 SO 中包含浮点数,因此将使用浮点支持进行编译。如果调用者想要使用浮点数,则为必需。有人有更好的方法来强制 g++ 进行包含浮点的编译吗?

在修复了 Dmitry 帮助我找到的许多错误后,我现在得到以下输出:

make
./Discover/dllmain.cpp: In function ‘void InitalizeLibraryServices()’:
./Discover/dllmain.cpp:175: warning: unused variable ‘pi’


./EdgeIO/dllMain.cpp: In function ‘void InitalizeLibraryServices()’:
./EdgeIO/dllMain.cpp:158: warning: unused variable ‘pi’


linking RemKonTester
    gflags = -g3
    tstlinkflags = -ldl  -lpthread 
    defines =  -D__linux__   -D_DEBUG   -D_IPC_ARCH_INTEL=1   -D_THREAD_SAFE

./RemKonTester/Debug/RemKonTester.o: In function `main':
/home/wmiller/Projects/Eclipse/./RemKonTester/RemKonTester.cpp:81: undefined 
    reference to `RemKon_EdgeIO::EdgeIoObject::hello()'
/home/wmiller/Projects/Eclipse/./RemKonTester/RemKonTester.cpp:111: undefined 
    reference to `RemKon_Discover::DiscoverObject::hello()'
/home/wmiller/Projects/Eclipse/./RemKonTester/RemKonTester.cpp:116: undefined 
    reference to `RemKon_Discover::DiscoverObject::SetLogLevel(unsigned int)'
/home/wmiller/Projects/Eclipse/./RemKonTester/RemKonTester.cpp:117: undefined 
    reference to `RemKon_Discover::DiscoverObject::hello()'
/home/wmiller/Projects/Eclipse/./RemKonTester/RemKonTester.cpp:118: undefined 
    reference to `RemKon_Discover::DiscoverObject::LocalIpAddress(int)'
/home/wmiller/Projects/Eclipse/./RemKonTester/RemKonTester.cpp:122: undefined 
    reference to `RemKon_Discover::DiscoverObject::RegisterCallback(bool(*)
    (void*), void*)'
/home/wmiller/Projects/Eclipse/./RemKonTester/RemKonTester.cpp:123: undefined 
    reference to `RemKon_Discover::DiscoverObject::Search()'

collect2: ld returned 1 exit status

我从 Eclipse 收到相同的错误消息集。

RemKonTester.cpp 包含声明这些项目的所有 .h。我已经用 extern "C" 声明尝试过它们,但没有。

希望得到帮助,

Wes


你的问题似乎是-l<library>:

$(CC)  $(gflags)  $(tstlinkflags) $(defines)      -L ./Debug    -ldiscover   
        -ledgeio -o ./Debug/RemKonTester  ./RemKonTester/Debug/RemKonTester.o  
        ./RemKonTester/Debug/logger.o  ./RemKonTester/Debug/libraryClass.o   
        2>&1  | tee ./RemKonTester/make.ERR

它们应该位于目标文件之后,因为链接器会在命令行遇到它们时加载它们并搜索未定义的符号。

See man ld(具体来说-l选项)了解更多信息:

-l 名称规范

...

链接器只会在命令行指定的位置搜索存档一次。如果存档定义了 在命令行上的存档之前出现的某个对象中未定义的符号,链接器将包括 档案中的适当文件。但是,稍后出现在命令行上的对象中的未定义符号不会导致 链接器再次搜索存档。

这应该适合你:

$(CC) $(gflags) $(tstlinkflags) $(定义) -L ./Debug -o ./Debug/RemKonTester ./RemKonTester/Debug/RemKonTester.o ./RemKonTester/Debug/logger.o ./RemKonTester/调试/libraryClass.o-ldiscover -ledgeio2>&1 |三通 ./RemKonTester/make.ERR

附:请注意,StackOverflow 中有一个用于编辑问题的选项,发布附加信息作为答案并不是一个好的做法。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

调用程序中对库类成员的未定义引用错误 的相关文章

随机推荐