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

Pao*_*aJ. 1 c++ macos cross-language objective-c objective-c++

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

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

// Our platform independent class
@interface LoggerDelegate : NSObject<SomeDelegate>
@end
Run Code Online (Sandbox Code Playgroud)

我可以在文件内创建它.mm并且它可以工作,但它适用于全局变量,并且不允许我更改其范围。所以我想在里面使用它Logger.h

class Logger {
public:

private:

LoggerDelegate* _ptr;   //<--- pointer to the Objective-C interface
};
Run Code Online (Sandbox Code Playgroud)

当我在里面定义我的界面时,Logger.mm它就可以工作了。但是是否可以将 Objective-C 和 C++.h文件内的 Objective-C++ 代码合并到一起?或者只能在.mm文件中进行?

Chr*_*phe 5

语言家族

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

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

Objective-C++也使用.h标头但不.mm使用实现文件。两者都必须遵守 Objective-C++ 语法:

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

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

混合语言 - 语法问题

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

混合标准 C++ 和目标 C/C++:您只能在 Objective-C++ 中将 C++ 语言与 Objective-C 或 Objective-C++ 结构结合起来(so.mm文件,最终.h当且仅当不包含在任何标准 C++ 或 Objective-C 源代码中时) )

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

混合语言 - 语义问题

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

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

可以使用PIMPL 习惯用法(也称为“不透明指针”)来管理从 C++ 到 Objective-C++ 对象的指针。Phil Jordan 的这篇精彩文章通过一个您可以轻松重复使用的示例解释了如何做到这一点。

Phil Jordan 的主要思想是使用一个标头,使用指向不完整对象的指针将 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++
};
Run Code Online (Sandbox Code Playgroud)

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