在c ++代码中使用objc对象?

时间:2013-11-07 07:27:42

标签: c++ objective-c macos xcode5

我有一个名为Log的ObjC类。在这个类中,我有一个公共静态方法调用错误。

我想从我的C ++代码中调用此方法(如果可能,请执行以下操作:Log :: error(string);)。

我在互联网和堆栈溢出,但我发现只有这个:

http://philjordan.eu/article/mixing-objective-c-c++-and-objective-c++

和此:

Calling Objective-C method from C++ method? (我尝试使用第一种方法,但我不知道如何创建void * objectiveCObject变量。)

1 个答案:

答案 0 :(得分:2)

Log类的头文件中,您需要提供一个可以由C ++使用的C-linkage的回调机制,您需要防止C ++看到Objective-C的东西。然后,此标头可由C ++和Objective-C共享。另一种方法是提供一个单独的头文件和实现文件,仅供C ++使用,但这会给实现增加更多的麻烦。

根据你的评论,我已经添加了C ++可访问的方法来创建和销毁Objective-C Log对象,但是我确信这个工作你必须使用MRR而不是ARC这样你可以管理一生。因此,您需要使用Log.m编译-fobjc-no-arc

Log.h:

#ifdef __OBJC__
#import <Foundation/Foundation.h>
#else // !__OBJC__
#include <stdarg.h>
// And maybe other C++ headers...
#endif // __OBJC__

typedef void *ObjcLog;

#ifdef __cplusplus
extern "C" {
#endif

extern ObjcLog *logCreate(const char *filename);
extern void logDestroy(ObjcLog *logObj);
extern void logInfo(ObjcLog *logObj, const char *msg, ...);
extern void logError(ObjcLog *logObj, const char *msg, ...);

#ifdef __cplusplus
}  // extern "C"
#endif

#ifdef __OBJC__
@interface Log : NSObject {
    // stuff
}

// Other stuff

@end
#endif // __OBJC__

Log.m:

#import "Log.h"

ObjcLog *logCreate(const char *filename) {
    // Assumes [Log initWithFilename:]
    Log *log = [[Log alloc] initWithFilename:[NSString stringWithUTF8String:filename]];
    return (ObjcLog *)log;
}

void logDestroy(ObjcLog *logObj) {
    Log *log = (Log *)logObj;
    [log release];
}

void logInfo(ObjcLog *logObj, const char *msg, ...) {
    char buffer[8192];
    va_list va;
    va_start(va, msg);
    vsprintf(buffer, msg, va);
    va_end(va);

    Log *log = (Log *)logObj;
    // Assumes [Log info:]
    [log info:[NSString stringWithUTF8String:buffer]];
}

void logError(ObjcLog *logObj, const char *msg, ...) {
    char buffer[8192];
    va_list va;
    va_start(va, msg);
    vsprintf(buffer, msg, va);
    va_end(va);

    Log *log = (Log *)context;
    // Assumes [Log error:]
    [log error:[NSString stringWithUTF8String:buffer]];
}

@implementation Log

...

@end

您的C ++代码应该能够像这样使用Objective-C Log对象:

ObjcLog *logObj = logCreate("/path/to/file.log");
...
logInfo(logObj, "The answer is %d.  What is the question?", 42);
...
logDestroy(logObj);

我可能赞成为Objective-C类创建一个C ++包装类,这样可以更轻松地管理对象的生命周期并简化对它的访问。但是我会避免在这里添加它,因为它可能会使已经过于复杂的东西变得复杂。