c++ - 如何处理 C++ 的 objective-c 委托(delegate)?

标签 c++ objective-c objective-c++

简短版本:我有一个 Qt/C++,我必须向其中添加有限数量的 Cocoa/Objective-C 代码。我已将 .cpp 文件更改为 .mm 文件,并将 objective-c 代码/对象添加到所述文件,它可以编译并运行。我现在需要我创建的其中一个对象的委托(delegate)——确切地说是一个 NSPopUpButton(或者更确切地说,它的菜单)——我被卡住了。如何为此对象添加委托(delegate)?

详细信息: 有问题的文件:

reportwindow.h, reportwindow.cpp 重命名为 reportwindow.mm - 这些文件包含我的原始 C++ 实现以及一些 objective-c 代码(打开包含 NSPopUpButton 的 NSSavePanel)。 reportwindow.h 额外包含在 .cpp 文件中,如果有区别的话。

menuHandler.h, menuHandler.mm - 这些文件包含一个(当前为空)objective-c 类,我打算将其用作委托(delegate)

我的第一个想法是我可以简单地让 C++ 类成为委托(delegate),但这显然行不通,因为直接的 C++ 不理解委托(delegate)。然后我想我会创建一个单独的 objective-c 类作为 NSMenuDelegate 并将它的一个实例作为成员对象添加到我的 C++ 类中。因为我已经能够将其他 objective-c 对象添加为成员,所以我认为这应该可行。然而,一旦我在 C++ 类头文件中包含了我的新 objective-c 类的头文件,我就从苹果头文件(NSValue.h , NSObject.h, 等)所以显然这没有用,至少不是原样。在我的类头文件中包含任何 cocoa 头时,我得到相同的结果。

然后我想我会尝试对 objective-c 类进行前向声明(这就是我让其他 objective-c 对象工作的方式)。然而,这也不起作用——如果我将它声明为“class myClassName”,我会收到关于将类重新定义为不同类型符号的错误(大概是 c++ 类与 objective-c 协议(protocol))。如果我尝试将其转发声明为@protocol myClassName,我会收到有关“'@' 标记前的预期不合格 ID”的错误消息。那么我怎样才能让它发挥作用呢?

最佳答案

确定回答您的问题:

reportwindow.h is additionally included in a .cpp file, if that makes a difference.

它确实有所作为。任何涉及 Objective-C 代码的编译单元(在本例中为 cpp 文件)都必须重命名为 .mm 或 .m。在 C++ 文件中包含 Objective-C 内容的 header 将导致 C++ 编译器看到它无法处理的 Objective-C 代码的问题。

将 cpp 文件重命名为 mm 将在编译期间选择 Objective-C 选项(当文件命名为 cpp 或 c 时则不是),因此允许使用 Objective-C 标记(主要是“@”你的情况)。

另一种方法是不将 Objective-C 委托(delegate)类包含到您的 C++ 类中,而是在 Objective-C 委托(delegate)中包含指向您的 C++ 类的指针(即以相反的方式实现)。通过这种方式,您可以安排一些事情,使 Objective-C 代码不接触 C++ 代码。

编辑:实际上,我更喜欢第二个建议。这是一个例子:

委托(delegate)类.h:

class MyCPPClassHandlingStuff;

@interface MyDelegateObject : NSObject <SomeDelegateProtocol> {
  MyCPPClassHandlingStuff *m_handlerInstance;
}

- (id) initWithCPPInstance:(MyCPPClassHandlingStuff*)cppInstance;

- (void) theDelegateProtocolMethod;

@end

委托(delegate)类.mm

#include "MyCPPClassHandlingStuff.h"

@implementation MyDelegateObject

- (id) initWithCPPInstance:(MyCPPClassHandlingStuff*)cppInstance
{
  self = [super init];
  if (self) {
    m_handlerInstance = cppInstance;
  }
  return self;
}

- (void) theDelegateProtocolMethod
{
  if (m_handlerInstance)
    m_handlerInstance->handleDelegateMethod();
}

@end

还有 MyCPPClassHandlingStuff.h:

#ifndef __MyCPPClassHandlingStuff_H__
#define __MyCPPClassHandlingStuff_H__

class MyCPPClassHandlingStuff
{
public:
  MyCPPClassHandlingStuff();
  void handleDelegateMethod();
};

#endif /* __MyCPPClassHandlingStuff_H__ */

MyCPPClassHandlingStuff 可以从 Objective-C 初始化,但您不能从那里的 C++ 代码初始化任何 Objective-C 类。如果您需要在 C++ 代码中使用 Objective-C,则必须将其编译为 Objective-C(即使用 .mm 文件)。我将 .cpp 的详细信息留给读者作为练习 ;)

关于c++ - 如何处理 C++ 的 objective-c 委托(delegate)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7892382/

相关文章:

c++ - boost 互斥顺序

c++ - 如何使用 std::fpos?

iphone - MPVolumeView,避免显示 "No Volume Available"

c++ - Objective-C 垃圾回收和 C++

objective-c - 无法将 Obj-C++ 子项目添加到 Xcode Obj-C 项目中

c++ - 模板模拟上的 EXPECT_CALL

c++ - 理解 C++ 中的 volatile 关键字

objective-c - iPad:如果用户计算机上的iOS不支持,iAds会怎样?

ios - CoreData : error: Serious application error. 核心数据更改处理期间捕获到异常

objective-c - 如何将 unsafemutablerawpointer 作为 C 指针传递给 Objective-C++?