- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
好吧,我知道可移植性不是 C++ 的强项,但我必须让我的代码在 Mac 和 Windows 上运行。我想出了一个解决方案,但它并不完美,我很想看看是否有人可以提出更好的解决方案。
我需要在几个 DLL/bundle 中使用一个类层次结构——例如,我有一个抽象基类 BaseClass;然后我扫描给定的 DLL 目录,对于每个 DLL,我寻找工厂方法 BaseClass* CreateObject(); - 它返回一个“BaseClass”。我有一个“共享头文件”,它包含在“主可执行文件”和 DLL 中,它像这样声明 BaseClass
#ifdef _MAC
#define DECLSPEC
#else
#ifdef COMPILING_DLL
#define DECLSPEC __declspec(dllexport)
#else
#define DECLSPEC __declspec(dllimport)
#endif
#endif
class DECLSPEC BaseClass{
[.. base "interface" declaration .. ]
}
然后,在我的 DLL 中,我通常会包含 BaseClass 声明,并声明我自己的“具体”类:
class MyDllClass:public BaseClass{
[.. actual DLL class definition/implementation here goes here ...]
}
到目前为止,还不错。现在出于某种原因,我需要在我的主要可执行文件中区分两种不同类型的 BaseObjects - 假设我有一个 DescriptionClass 和一个 ActionClass,它们都是 BaseClass,但接口(interface)略有不同。我的第一个实现是简单地修改“共享 header ”并添加:
class DECLSPEC DescriptionClass{
[.. base "Description interface" declaration .. ]
}
class DECLSPEC ActionClass{
[.. base "Action interface" declaration .. ]
}
然后我的 DLL 将变成:
class MyDllClass:public ActionClass /* or DescriptionClass, depending on case*/ {
[.. actual DLL class definition/implementation here goes here ...]
}
在我的主要可执行文件中,我会像这样使用它:
BaseClass* obj = CreateObjectFromDLL("path_to_dll");
ActionClass* action_obj = dynamic_cast<ActionClass*>(obj);
if(action_obj){
// Do the stuff that is relevant for Action objects
}
DescriptionClass* description_obj = dynamic_cast<ActionClass*>(obj);
if(description_obj){
// Do the stuff that is relevant for Description objects
}
这就是问题所在:虽然它可以在带有 Visual Studio 的 Windows 上运行(可能是由于 MS declspec 扩展),但它在 Mac 上会失败(现在不确定它是否在调试时失败,但我确定它在发布时失败) 用 GCC 编译时。原因很简单,即使不是很明显:可执行文件和动态库是分开编译的。尽管它们都包含 BaseClass、ActionClass、DescriptionClass 的声明 - 这些类相同,但它们只是存在于二进制文件和 DLL 中的“相同拷贝”。所以实际上,我在 DLL 中创建的是一个 dll'BaseClass* ,它恰好与 main'Baseclass* 具有相同的内存布局,所以指针是兼容的,所以当我将指针从 DLL 传递到 EXE 时,它一切都“按预期”工作。 OTOH,当我进入一个更复杂的类层次结构时,dll'ActionClass 和 main'ActionClass 的 vtables/RTTI 不再相同(尽管在源代码中它们是相同的),所以当我尝试转换(通过 dynamic_cast)一个 main 'BaseClass* 到 main'ActionClass* 我得到一个空结果 -> 因为我的指针实际上指向一个 dll'BaseClass 对象/dll'ActionClass 对象,虽然在 DLL 中我可以毫无问题地将“BaseClass*”转换为一个“ActionClass*”——在主可执行文件中,我无法将 dll 的“BaseClass*”转换为“ActionClass*”,因为 DLL 和“主可执行文件”版本的 Action Class 之间存在细微的运行时差异。
我已经通过在 BaseClass 中声明一个虚拟方法(类似于“bool isActionClass()”)来“修复”这个问题,所以现在我可以区分...但我对这个解决方案不是很满意。
GCC 有什么东西吗?一些类似于“__declspec”的声明 - 一种保证在“主可执行文件”和“dll”中声明的相同类将 100% 兼容的方法?
最佳答案
我实际上找到了我的问题的答案,因为我需要完整地表述它,而且我做了更好的谷歌搜索:)好像是
__attribute__((dllimport)) // import
__attribute__((dllexport)) // export
会尝试的,我想我也会把问题留在这里,以防其他人偶然发现这个问题(并且作为对人们的警告,“主二进制文件”和 DLL 中包含的相同头文件通常会导致不同实际实现,具体取决于编译器选项 - 除非您将适当的 dllimport/export 属性放在类上)。
关于c++ - 如何(可移植地)使用 C++ 类层次结构和动态链接库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1564802/
我是 Mercurial 的新手,并且不知何故仍处于评估过程中,所以这四个概念对我来说有点困惑。有些被提到等同于 Git 的 Staging/Index 概念,有些甚至比 Git 的 Staging
如何将在某些网站 (www.example1.com) 上用某种语言即 (java) 制作的 session 传送到用其他语言制作的网站,即在某些其他网站上的 (php),即 (www.example
我有以下代码行我想移植到 Torch Matmul rotMat = xmat @ ymat @ zmat 我能知道这是不是正确的顺序: rotMat = torch.matmul(xmat, tor
我正在尝试移植一个内部有一个联合的 C 结构。 Winapi.Winsock2.pas 中的默认结构记录中缺少某些字段。 但这是正确的方法吗?谢谢。 typedef struct _WSACOMPLE
我想将基于 webkit 的浏览器移植到我的堆栈中。谁能介绍一下 webkit 浏览器引擎的组织结构?目前我所知道的是它具有用于呈现 html 和解析 javascript 的核心。我想了解更多,比如
我目前有一个 ActiveX 控件,它链接到许多 c/c++ dll。问题是我们现在需要此控件在 IE 以外的浏览器(最重要的是 Firefox)上运行。 在我看来,我有以下选择: 将控件编写为 fi
我正在尝试在 Objective-C 中重写 Java 库。我想将其重写为 API,以便需要实现某些方法。我已经开始尝试重写代码,但遇到了一些问题。 Objective-C 是否支持抽象类? 如果没有
我已经有一段时间没有接触 SQL 了,所以我需要重新学习一下。我的计算机上运行着一个 SQL 数据库,我的服务器是 localhost。我在 VB.Net 中制作了一个连接到该数据库的应用程序。一切都
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit th
运行命令时出现错误 [root@himanshi busybox-1.20.2]# make ARCH=arm CROSS_COMPILE=arm-unknown-linux-gnueabi- CON
我需要将为 iPhone 编写的现有游戏移植到 Flash。 iPhone 游戏主要是用纯 C 而不是 Objective C 编写的。 我想知道是否有任何好的工具可以将 C 代码直接转换为 Acti
我将要在 Smalltalk (Pharo) 中构建一个项目。还有一个 python 库,我打算将其用于相同的目的。现在,有 3 个选项: 那些 python 库的 Smalltalk 包装器 将 p
我必须在 GPU 上移植一个广泛使用随机数的结构。一切都可以毫无问题地移植,但随机生成器函数是唯一在该类的所有函数中被广泛调用的函数。我虽然可以简单地将它重新实现为类本身的内部设备函数。下面我放了一个
我对整个移植问题有点陌生,因为 Android SDK 提供的模拟器速度很慢,所以我解决了这个问题。 我下载了 android-x86-3.2-RC2-eeepc 和 android-x86-3.2-
我们的数据库 (PostgreSQL 9.x) 中有一些 PL/pgSQL 存储过程。 这些是严格顺序的,在某些情况下,可能会非常慢。 我们正在考虑将它们移植到 PL/Java、PL/Python 或
我有一个 Android 应用程序可以处理圆顶图像。出于性能原因,我想用 C++ 编写应用程序的某些部分,然后通过 NDK 调用这些方法。我是否需要一个特定的 C++ 编译器(例如用于嵌入式系统)或者
我正在从事一个将一大堆 OS-9(微软件)代码移植到 linux 的项目。 OS-9 中的信号处理功能允许您创建自己的信号,或者至少它是如何实现的(intercept() 函数)。我对 linux 信
目前我有这个 gtk2 代码: GList *input_devices = gdk_devices_list(); while(input_devices) { GdkDevice *devic
我正在尝试移植 Aether.Physics2D从 C# 到 Xojo 的库。这本质上是 Farseer 物理引擎的调整版本。大部分已经完成,但有一部分源代码我无法解决(可能是因为 C# 不是我的主要
我们正在开发采用 RISCV 架构的多核处理器。 我们已经为单核 RISCV 处理器移植了 Linux,它正在我们自己的基于 FPGA 的主板上使用 busybox rootfs。 我现在想为多核 R
我是一名优秀的程序员,十分优秀!