c++ - 具有公共 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
.
解决方案
这是一种可能接近您想要实现的技术。
您可以定义一个仅依赖于A的公共接口的接口,因此对B没有任何依赖。接口类包括一个工厂。此接口将成为您的库的头文件的一部分。
class Interface {
public:
virtual void foo () = 0;
virtual void bar () = 0;
static std::unique_ptr<Interface> make ();
virtual ~Interface () = default;
};
在库的源文件中,您将包含A和B的头文件,并创建接口的实现以及工厂的定义。
class Implementation : public Interface {
//...
};
std::unique_ptr<Interface>
Interface::make () {
return std::make_unique<Implementation>();
}
因此,您的库的用户可以访问该接口,并且可以在不了解私有成员或私有方法的情况下调用公共方法。
推荐阅读
- firebase - 运行 Gradle 时出错,gradlew.bat 异常退出,评估项目时出现问题
- r - R dplyr 从具有列名的列中选择值以在单独的列中选择
- sql - 改进连接表的 SQL 查询
- laravel - 将 Laravel (with Valet) 与 Sequel Pro 连接,我该如何解决问题?
- python - 网络爬取域名问题
- reactjs - GraphQL 错误:Int 类型的变量 junctionId!用于期望 bigint 的位置
- ios - Flutter 我想设计这样的界面,你们能帮帮我吗?我正在设计一个关于页面,但我遇到了问题
- kdb - KDB+ 中符号数据类型的最大大小是多少?
- javascript - 使用 vue.js 更改按钮颜色并在单击时向子组件发送道具
- typescript - 在 TypeScript 中为影子根元素自定义 TestCafe 选择器