c++ - 在 Objective-C++ 的 .h 文件中添加指向定义的 Objective-C 类的指针

标签 c++ objective-c macos objective-c++ cross-language

是否可以在 Objective-C++ 中创建类,它的字段是指向 Objective-C 接口(interface)的指针(在 .h 文件中)?

为了澄清一点,我创建了一个 Objective-C 接口(interface):

// Our platform independent class
@interface LoggerDelegate : NSObject<SomeDelegate>
@end

我可以在 .mm 文件中创建它,它可以工作,但它是用于全局变量的,我不允许更改它的范围。所以我想在 Logger.h 中使用它:

class Logger {
public:

private:

LoggerDelegate* _ptr;   //<--- pointer to the Objective-C interface
};

当我在 Logger.mm 中定义我的接口(interface)时,它可以工作。但是是否可以在 .h 文件中合并来自 Objective-C 和 C++ 的 Objective-C++ 代码?还是只能在 .mm 文件中使用?

最佳答案

语系

Objective-C使用 .h header 和 .m 实现文件。如果是抽象接口(interface),您可能不需要后者。所有这些文件都必须符合 Objective-C 语法:

  • 您可以在.h.m 中使用一些C 语法;它将被识别,因为 Objective C 是 C 的语言扩展。
  • Objective-C 编译器无法识别纯 C++ 语言元素(例如 classtemplate)。

Objective-C++还使用 .h header ,但使用 .mm 实现文件。两者都必须符合 Objective-C++ 语法:

  • 大多数 C++ 语法都会被识别,因为 Objective C++ 是基于 C++ 的。
  • 所有 Objective-C 语法都将被识别,因为 Objective C 是 Objective-C++ 的子集。

Standard C++使用 header (通常是 .h.hpp)和实现文件(通常是 .cc.cpp)。这些必须仅使用 C++ 语言元素。不属于 C++ 的 Objective-C 和 Objective-C++ 语言元素,例如 @interface 将触发 C++ 编译错误。

混合语言——语法问题

混合使用 Objective-C 和 Objective-C++:如果您在 Objective-C 和 Objective C++ 之间共享 header (.h),则它们必须同时遵守 Objective-C 和 Objective-C++。因此, header 只能使用两种语言之间的共同点,即 Objective-C(和 C)语言元素。

混合标准 C++ 和 Objective-C/C++:您可以仅在 Objective-C++ 中将 C++ 语言与 Objective-C 或 Objective-C++ 结构组合(因此 .mm 文件, 和最终 .h 当且仅当不包含在任何标准 C++ 或 Objective-C 源代码中时)

这就是为什么您可以在 .mm 文件中使用 Objective-C 接口(interface)的原因。但是您不能在将包含在 C++ 代码中的 .h 文件中使用它。

混合语言——语义问题

标准C++和Objective-C/C++的对象模型和生命周期是不一样的。这使得互操作性变得困难。例如,C++ 类不能从 Objective-C 类继承,如果 Objective-C 类包含 C++ 对象,则 construction可能很棘手。并且没有任何 C++ 语言结构可以保证与 Objective-C++ @interface 原生兼容。

但是没有办法与指针互操作吗?

使用 PIMPL idiom 可以管理从 C++ 到 Objective-C++ 对象的指针(也称为“不透明指针”)。 This brilliant article from Phil Jordan通过您可以轻松重用的示例说明如何执行此操作。

Phil Jordan 的主要想法是使用 header 将 Objective-C++ 对象包装成一个标准的 C++ 类,使用一个指向不完整对象的指针。 Phil 以一种巧妙的方式使用条件编译,允许使用相同的 .h 来实现跨语言目的:

#ifdef __OBJC__
@class XXXX;       // if header used in Objective C++
#else
struct XXXX;        // if header used in standard C++
#endif

class WrapperClass {   // standard C++ class also recognuzed by Objective C++
    XXXX *wrapped_objective_c_object_pointer; 
public:  
    ...// public interface for C++
};

这确保包含 header 的 C++ 代码永远不会使用指针对包装对象执行任何操作。 C++ 包装器类的实现将在 .mm 文件中提供,该文件了解 Objective C++ 语义。我强烈建议您阅读这篇文章以了解该解决方案的所有细节。

关于c++ - 在 Objective-C++ 的 .h 文件中添加指向定义的 Objective-C 类的指针,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46388787/

相关文章:

c++ - C++ 中特定重构的惯用语

c++ - sizeof 和 unsigned char 数组错误? (c/c++)

objective-c - 是否可以在 Cocoa/ObjC 中使用 "unpair"蓝牙设备?

swift - 如何使用 NSURL 获取用户路径

c++ - C++ 编译器如何从 utf8 源文件生成 unicode 字符串文字

c++ - 为什么在返回临时(右值)时在 move 构造函数之前调用析构函数

ios - 如何做json类型响应?

objective-c - WebRTC:如何将 RTCVideoEncoderSettings 传递到 RTCVideoEncoder

java - OSX JNLP 下载未达到 KeyChain 上的证书

python - PIP 安装和 Python 路径