【问题标题】:How to handle objective-c delegation from C++?如何处理来自 C++ 的 Objective-C 委托?
【发布时间】:2015-11-06 05:32:19
【问题描述】:

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

详情: 有问题的文件:

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

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

我的第一个想法是我可以简单地将 C++ 类设为委托,但这显然行不通,因为直接的 C++ 不理解委托。然后我想我会创建一个单独的objective-c类作为NSMenuDelegate,并将它的一个实例作为成员对象添加到我的C++类中。由于我已经能够添加其他objective-c 对象作为成员,我认为这应该可以工作。但是,一旦我在 C++ 类头文件中包含了我的新 Objective-c 类的头文件,我就收到了数百个关于“在'@'令牌之前的预期不合格 ID”的错误 - 来自苹果头文件(NSValue.h , NSObject.h 等)所以显然这不起作用,至少不是原样。在我的类头文件中包含任何可可头文件时,我得到相同的结果。

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

【问题讨论】:

  • 您是否将那个特定的 cpp 文件重命名为 mm?
  • 与有问题的标题相关联的那个,是的。
  • 第二种方法是要走的路,而且可能是唯一的方法。你不应该在第一次失败的尝试后投降。提供更多详细信息,包括什么,可能还有更多的 .cpp 文件,其中包括 reportwindow.h
  • 好吧,在第一次尝试失败后,我并没有完全投降——我只是没有包括我斗争的所有血腥细节 :-)

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


【解决方案1】:

好的,回答你的问题:

reportwindow.h 还包含在 .cpp 文件中,如果这样的话 不同。

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

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

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

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

DelegateClass.h:

class MyCPPClassHandlingStuff;

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

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

- (void) theDelegateProtocolMethod;

@end

DelegateClass.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 详细信息留给读者作为练习;)

【讨论】:

  • 能否请您也提供 .cpp 实现的想法:P 您将如何从 cpp 初始化 MyDelegateObject。
【解决方案2】:

创建一个满足委托协议的 Objective-C 类,并将其委托给您的 C++ 类。问题是 AppKit 期望与 Objective-C 对象对话,因此您需要一个 shim 来委托给您的 C++ 对象。 (是的,您可以做一些运行时滥用,在您的 C++ 类中定义一个 isa 指针以使其与 ObjC 对象在某种程度上兼容(到您可以向 C++ 类发送消息的程度),但不要这样做。这很讨厌。)

因此,制作一个使用 C++ 类初始化的 shim,将其作为 ivar。它应该实现您感兴趣的委托方法,并以它可以理解的方式将它们传递给您的 C++ 类。多田,完成。

【讨论】:

    【解决方案3】:

    然后我想我会尝试对objective-c 类进行前向声明(这就是我让其他objective-c 对象工作的方式)。但是,这也不起作用-如果我将其声明为“class myClassName”,则会收到关于将类重新定义为不同类型的符号的错误(可能是 c++ 类与 Objective-c 协议)。

    为什么不将其前向声明为@class myClassName

    【讨论】:

    • 我试过了,它给出了关于“'@'符号之前的预期不合格ID”的错误。原来 Qt 正在生成一个 moc_.cpp 文件,这让事情变得很糟糕。将其重命名为 .mm 解决了这个问题。
    【解决方案4】:

    Can I separate C++ main function and classes from Objective-C and/or C routines at compile and link? 中有一些关于如何设置标题的有用信息,以便它们可以在 ObjC 和 C++ 中使用

    【讨论】:

      猜你喜欢
      • 2017-09-08
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2011-04-12
      • 1970-01-01
      相关资源
      最近更新 更多