下面添加了其他问题,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