仅针对直接依赖关系链接GCC/LD



我有一个场景,其中我有一种依赖于库a的二进制文件,而库a又依赖于库B。

我已经针对库B构建了库A,但库B:s的符号都没有从库A中泄漏出来,所有内容都包含在cpp文件中。

现在我只想将二进制与库A链接,因为在二进制中找到的所有符号都可以由库A满足。这可能吗?

在实际应用程序中,库B是一个网络协议的实现,我有很多二进制文件链接到中间库。我不希望二进制文件知道所使用的不同网络协议。

平台:Linux/GCC

代码:

liba/liba.h:

#ifndef LIBA_H
#define LIBA_H
int getANumber();
#endif

liba/liba.cp:

#include "liba.h"
#include "../libb/libb.h"
int getANumber(){ return getBNumber(); }

libb/libb.h:

#ifndef LIBB_H
#define LIBB_H
int getBNumber();
#endif

libb/libb.cpp:

#include "libb.h"
int getBNumber(){ return 42; }

main.cpp:

#include "liba/liba.h"
#include <iostream>
int main(int argc, char** argv) {
  std::cout << getANumber() << std::endl;
  return 0;
}

命令:

~/libb/ $ g++ -shared -o libb.so libb.cpp
~/liba/ $ g++ -shared -o liba.so liba.cpp -L../libb -lb
~/ $ g++ -o main main.cpp -Lliba -la # fails
~/ # These two work, but I don't want to specify libb here.
~/ $ g++ -o main main.cpp -Lliba -la -Wl,-rpath-link,libb 
~/ $ LD_LIBRARY_PATH=libb g++ -o main main.cpp -Lliba -la

解决这个问题的最佳方法是什么?我必须将其创建为插件吗?

谨致问候,Daniel

如果你想在运行时更改你使用的库,那么你不能直接链接它,而是使用"手动"加载。换句话说,调用dlopendlsym

这可能也意味着你需要在"liba"中有一个稍微不同的体系结构,因为"libb"中的每个函数都变成了一个函数指针,所以沿着以下几条线:

int (*getBNumber)() = NULL;
void initialize()
{
   void *handle = dlopen("libb", RTLD_NOW); 
   getBNumber = (int (*)())dlsym(handle, "getBNumber"); 
}
int getANumber(){ return getBNumber(); }

您需要设置一些东西,以便在某个时刻调用initialize,或者在每个函数中都有一个if (!initialized) initialize();

最新更新