Вызов Objective-C родителя из C++ объекта
Создание расширений аудиоустройства на iOS требует смешивания классов C++ и Objective-C. В результате теперь у меня есть объект Objective C с объектом C++ в качестве одной из его переменных.
Мне бы хотелось, чтобы дочерний объект C++ мог уведомлять своего родителя / владельца Objective-C об определенных изменениях состояния.
В псевдокоде:
void cppChildObject::callMom() {
objectiveCParent::notificationMethod();
}
Это возможно элегантным способом?
1 ответ
Это зависит от того, как вы определяете элегантный... Это возможно довольно элегантным способом, если класс C++ находится в файле Objective-C++ (класс с расширением.mm) и не используется непосредственно кодом C++, который не находится в Objective-C++ исходный файл. Проблема в том, что код C++ может использовать типы Objective-C, только если он находится в исходном файле Objective-C++. Вот быстрый пример, надеюсь, вы найдете его полезным.
Файл Objective-C++, mylib.mm
(обратите внимание на расширение.mm):
#import "objcpp.h"
#import <stdio.h>
#import "cpp.h"
// A C++ class in an Objective-C++ file.
// This C++ class would not need to sub-class ParentNotifier, and ParentNotifier
// would not be needed at all if it were not for OutsiderCPP, which talks to its
// Objective-C parent through InsiderCPP.
class InsiderCPP : public ParentNotifie {
public:
InsiderCPP(MyClassOCPP * parent) : myParent(parent){}
void doSomething() {
callMom("To Mom from insider.");
}
void callMom(const char * msg) {
[myParent notificationMethod:msg];
}
private:
MyClassOCPP * __weak myParent;
};
@interface MyClassOCPP ()
@property InsiderCPP * insiderChild;
@property OutsiderCPP * outsiderChild;
@end
@implementation MyClassOCPP
-(id)init {
self.insiderChild = new InsiderCPP(self);
self.outsiderChild = new OutsiderCPP(self.insiderChild);
return self;
}
-(void)doWork {
self.insiderChild->doSomething();
self.outsiderChild->doSomething();
}
-(void)notificationMethod:(const char *)msg {
printf("Parent has been notified with: %s\n", msg);
}
-(void)dealloc {
delete self.insiderChild;
delete self.outsiderChild;
}
@end
Вот соответствующий заголовок, objcpp.h
:
#ifndef objcpp_h
#define objcpp_h
#import <Foundation/Foundation.h>
@interface MyClassOCPP : NSObject
-(id)init;
-(void)dealloc;
-(void)doWork;
-(void)notificationMethod:(const char*)msg;
@end
#endif
Вот "чистый" исходный файл C++ (mylib.cpp
) не имеет ничего общего с Objective-C:
#include <stdio.h>
#include "cpp.h"
void OutsiderCPP::callMom(const char * m) {
myParent->callMom(m);
}
void OutsiderCPP::doSomething() {
callMom("To Mom from outsider.");
}
и вот соответствующий заголовок (cpp.h
):
#ifndef cpp_h
#define cpp_h
class ParentNotifier
{
public:
virtual void callMom(const char *) = 0;
};
class OutsiderCPP
{
public:
OutsiderCPP(ParentNotifier * p) : myParent(p) {}
void doSomething();
void callMom(const char *);
private:
ParentNotifier * myParent;
};
#endif
Пожалуйста, обратите внимание, что этот пример только для иллюстрации и не является производственным качеством.