1

我用 gcc 编译了一个共享库并将其链接到我的主库。主类应该初始化一个记录器类,它应该在共享库中可见,但看起来好像共享库有它自己的实例。

包含文件如下所示:

extern Log gLog;

主要是声明的。

Log gLog(new StreamWriter());

当我尝试链接它时,我undefined symbol _gLog在共享库中收到链接器错误。我以为可能是因为它是一个类实例,所以我把它改成了一个指针,但我得到了同样的结果。更糟的是,我想我可以创建一个小的虚拟模块,在共享库中创建相同的全局变量,然后调用一个函数来初始化它。但是对于这个函数,我也会得到一个链接器错误,因为它在 main.js 中不可见。

在共享库中:

Log *gLogger;

int initLibrary(Log *pLogger)
{
    gLogger = pLogger;
}

主要是:

Log gLog(new StreamWriter());
int initLibrary(Log *pLogger);
main()
{
    initLibrary(&gLog);
}

我再次在链接器中得到一个未定义的符号,这次是我的 initLibrary 函数。

现在我通过创建一个有效的虚拟类来解决这个问题。但是,我想知道如何跨共享库边界正确定义符号,因为我的理解似乎是错误的。

使用谷歌时,我在这里找到了一些线程使用共享库中的全局变量和全局变量、共享库和 -fPIC 效果作为示例(还有其他几个与此问题有关的问题)。但是,我尝试使用-fpic主模块重新编译所有内容,但它仍然无法正常工作。该-rdynamic选项是未知的,所以我不知道这是从哪里来的。

我可以使用共享库中的类,反之亦然,所以这只影响全局符号。那么我做错了什么,主代码和共享库不能看到彼此的符号?

4

3 回答 3

1

正确的方法是在共享库中创建 Logger 的实例,使用全局变量(如果封装在命名空间中更好)或 Singleton 类。然后让你的主程序使用它。

于 2013-05-24T07:29:10.860 回答
0

我终于找到了为什么这不适用于全局符号的问题(类没有问题)。

我在cygwin下编译,显然共享对象被编译为DLL。我试图查看该库并注意到它是 EXE 格式而不是 ELF。因此,我尝试使用 Microsoft DLL 语法来导出符号,然后它突然起作用了。__declspec(dllexport)在符号上添加 a就可以了。

我希望我必须__declspec(dllimport)在主项目中使用来导入符号,但这不起作用。不确定我是否误解了这个参数,或者 cygwin 版本的 gcc 是否做了一些神奇的工作。

所以当你在 cygwin 下编译共享库时,如果你想导出符号,它必须看起来像这样。

共享库 foo.h:

__declspec(dllexport) int foo(int a);

共享库 foo.cpp:

int foo(int a)
{
 ....
}

在可执行文件 foo.h 中:

int foo(int a);

在可执行的 main.cpp 中:

main()
{
     foo(1);
}

共享库必须用-fpicswitch 编译并用-shared.

于 2013-05-24T22:08:42.153 回答
0

您需要声明具有默认可见性的变量,以使其对其他共享库或主程序可见。您似乎正在使用 进行编译-fvisibility=hidden,因此库中的符号不​​会解析为主程序或其他库中的定义,反之亦然。使用 GCC 的可见性属性,您可以反转这种效果。

简而言之

  • 跨单个对象文件的实体(全局、本地、..)的声明的范围可见性
  • 跨多个对象文件(外部、内部)实体的声明的链接可见性
  • 跨不同共享库的实体的声明的可见性可见性(默认,隐藏)。

另一种可能性是您正在混合 C 和 C++ 代码并弄乱了语言链接。

于 2013-05-24T08:48:16.917 回答