首页 > 解决方案 > 具有公共 api 的 C++ 动态库,可隐藏依赖库

问题描述

我正在尝试在 C++ 中创建一个多平台库,以供 C++ 消费者应用程序使用。打电话给我的图书馆A。我想为每个目标平台提供一个动态库文件,以及一个消费者应用程序可以用来编译和执行的头文件(称为 export.h)。我的库依赖第三方开源库,用c编写,很难正确链接;调用这个库B

为了让我的消费者免于链接到 B 的痛苦,我想抽象出对它的每次调用,这样消费者甚至不需要来自 B 的单个头文件。消费者应用程序 ( C ) 应该能够仅使用 A 进行编译。 dll、B.dll 和 export.h;并仅使用 A.dll 和 B.dll 作为依赖项运行(根据需要将特定于平台的后缀替换为共享库)。

B 定义了很多类型,主要是结构。(B 不是写在目标 c 中,尽管它可能应该是。) A 的部分工作是生成包含和管理围绕逻辑行的结构组的类,这很明显。C 需要调用属于 A 中的类的函数,因此函数原型需要在 export.h 中,但 B 类型不能在 export.h 中,否则 C 将需要包含来自 B 的标头。

是否有一种语法可以让我定义一个类的公共成员(在 A 中)而不定义所有私有成员?

显然,A 中没有依赖 B 类型的公共成员。到目前为止,我发现的最接近的是这个 question。不透明的指针可能是解决方案的一部分,但是 C 需要访问 A 中类的函数。我真的不想为每个公共 A 类成员编写一个辅助函数,尽管这可能会起作用。有任何想法吗?

编辑:根据要求,解释代码。

工人.h:

#include <SomeThirdPartyLib.h>
class Worker {
public:
  Worker();
  ~Worker();
  void DoWork();
private:
  Truck t;
}

SomeThirdPartyLib.h:

typedef struct TruckS {
  char data[200];
  char* location;
} Truck;

工人.cpp:

#include "worker.h"
Worker::Worker() {}
Worker::~Worker() {}
Worker::DoWork() {
  t.location = "Work";
}

主.cpp:

#include <export.h>
int main(int argc, char** argv) {
  Worker w();
  w.DoWork();
}

现在我正在寻找放入 export.h 的语法,它允许使用该头文件和我的 dll 编译这个外部应用程序,但不需要访问SomeThirdPartyLib.h.

标签: c++dllshared-librariesdllexport

解决方案


这是一种可能接近您想要实现的技术。

您可以定义一个仅依赖于A的公共接口的接口,因此对B没有任何依赖。接口类包括一个工厂。此接口将成为您的库的头文件的一部分。

class Interface {
public:

    virtual void foo () = 0;
    virtual void bar () = 0;

    static std::unique_ptr<Interface> make ();

    virtual ~Interface () = default;
};

在库的源文件中,您将包含AB的头文件,并创建接口的实现以及工厂的定义。

class Implementation : public Interface {
    //...
};

std::unique_ptr<Interface>
Interface::make () {
    return std::make_unique<Implementation>();
}

因此,您的库的用户可以访问该接口,并且可以在不了解私有成员或私有方法的情况下调用公共方法。

在线尝试!


推荐阅读