【问题标题】:Having issue linking shared object in linux在linux中链接共享对象时遇到问题
【发布时间】:2018-11-12 12:32:21
【问题描述】:

我已获得 .so 文件。

ldd libTodoAPI.so
    linux-vdso.so.1 =>  (0x00007ffc766b8000)
    libcrypto.so.1.0.0 => /lib/x86_64-linux-gnu/libcrypto.so.1.0.0 (0x00007fead8e3f000)
    libsqlite3.so => /usr/lib/libsqlite3.so (0x00007fead8b6a000)
    libQt5Network.so.5 => /usr/lib/libQt5Network.so.5 (0x00007fead8804000)
    libQt5Script.so.5 => /usr/lib/libQt5Script.so.5 (0x00007fead838b000)
    libQt5Core.so.5 => /usr/lib/libQt5Core.so.5 (0x00007fead7c4f000)
    libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007fead78cd000)
    libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007fead76b7000)
    libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fead72ed000)
    libdl.so.2 => /lib/x86_64-linux-gnu/libdl.so.2 (0x00007fead70e9000)
    libpthread.so.0 => /lib/x86_64-linux-gnu/libpthread.so.0 (0x00007fead6ecc000)
    libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007fead6bc3000)
    libicui18n.so.53 => /usr/lib/libicui18n.so.53 (0x00007fead6777000)
    libicuuc.so.53 => /usr/lib/libicuuc.so.53 (0x00007fead63ec000)
    libgthread-2.0.so.0 => /usr/lib/x86_64-linux-gnu/libgthread-2.0.so.0 (0x00007fead61ea000)
    librt.so.1 => /lib/x86_64-linux-gnu/librt.so.1 (0x00007fead5fe2000)
    libglib-2.0.so.0 => /lib/x86_64-linux-gnu/libglib-2.0.so.0 (0x00007fead5cd1000)
    /lib64/ld-linux-x86-64.so.2 (0x00007fead949a000)
    libicudata.so.53 => /usr/lib/libicudata.so.53 (0x00007fead4649000)
    libpcre.so.3 => /lib/x86_64-linux-gnu/libpcre.so.3 (0x00007fead43d9000)

并带有标题

#ifndef TODO_API_H
#define TODO_API_H

#include <string>

using namespace std;

namespace todoarea {
    class TodoAPI{
    public:
        static string checkDoer();
        static string getInformation();
        static string callTask(string funcName, string param);
        static string Done(string param);
        static string returnStatus(string param);
        static string Close();
    };
}

#endif //TODO_API_H

我正在尝试创建一个程序来从这个共享对象调用并且遇到问题。

//load.cpp
#include <iostream>
#include "TodoAPI.h"

using namespace todoarea;

int main(int argc, char **argv)
{
 //  TodoAPI* myTodo = new TodoAPI();
 //  myTodo>checkDoer();
 //   TodoAPI m;
   string data = TodoAPI::checkDoer();
   string a = TodoAPI::getInformation();
}

当我尝试编译它时

g++ load.cpp -lTodoAPI
/tmp/ccs1hwWP.o: In function `main':
load.cpp:(.text+0x27): undefined reference to `todoarea::TodoAPI::checkDoer[abi:cxx11]()'
load.cpp:(.text+0x33): undefined reference to `todoarea::TodoAPI::getInformation[abi:cxx11]()'
collect2: error: ld returned 1 exit status

如果我尝试查看 so 文件

nm -s libTodoAPI.so
nm: libTodoAPI.so: no symbols

还有

nm -DC libTodoAPI.so  | grep todoarea
000000000000a63a T todoarea::TodoAPI::Done(std::string)
000000000000a098 T todoarea::TodoAPI::callTask(std::string, std::string)
0000000000009e9c T todoarea::TodoAPI::getInformation()
000000000000a398 T todoarea::TodoAPI::returnStatus(std::string)
0000000000009ca0 T todoarea::TodoAPI::checkDoer()
000000000000a8dc T todoarea::TodoAPI::Close()

当我使用 -D 选项尝试 nm 时,它列出了许多功能(大部分看起来像 Qt)。

问题:如何使用这个库在 linux 中创建应用程序?我究竟做错了什么?我试图动态调用函数但失败了。任何帮助将不胜感激。

【问题讨论】:

  • 我认为您可能需要指定共享库-L/path/to/your/so -lTodoAPI 的路径。试试这个:gist.github.com/gubatron/32f82053596c24b6bec6
  • 我确实尝试了 -L 选项,我什至将这个 .so 文件复制到 /usr/lib 和 /usr/local/lib 都一样。
  • 我怀疑这个 .so 文件是用 Qt 库编译的,可能有不同的格式或结构,但我不确定。我是 linux 新手。

标签: linux g++ shared-libraries


【解决方案1】:

您在 GCC 版本与 libTodoAPI.so 之间遇到了 ABI 中断 已构建,并且您正在使用它来编译代码。见Dual ABI Troubleshooting

疑难解答

如果您收到有关未定义符号引用的链接器错误,这些符号涉及 std::__cxx11 命名空间或标签 [abi:cxx11] 那么它可能表明你是 试图将使用不同值编译的目标文件链接在一起 _GLIBCXX_USE_CXX11_ABI 宏。这通常发生在链接到第三方库时 是用旧版本的 GCC 编译的。如果无法重建第三方库 使用新的 ABI,那么您将需要使用旧的 ABI 重新编译您的代码。

【讨论】:

  • 谢谢,我添加了 -D_GLIBCXX_USE_CXX11_ABI=1 选项,它通过了。
  • 当我尝试执行程序时出现“符号查找错误:/usr/lib/libssl.so.1.0.0: undefined symbol: EVP_idea_cbc”是不同的问题吗?
  • @THEn 那似乎是this bug的一种表现形式
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多