为什么从objc++调用objc函数会导致构建失败?



给定MVP代码:

objc_funcs.h

#import <Foundation/Foundation.h>
NSString* doFoo(void);

objc_funcs.m

#import <Foundation/Foundation.h>
NSString* doFoo()
{
return @"fffuuu";
}

main.mm

#import "objc_funcs.h"
int main(int argc, char * argv[]) {
doFoo();
return 0;
}

如果我这样做,构建结果将是

Undefined symbols for architecture x86_64:
"doFoo()", referenced from:
_main in main.o
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)

虽然如果我重命名main.mm->main.m,构建将顺利进行。

WAIDW吗?😖

问题涉及C与c++的联动。

在C头中处理的正常方式是测试__cplusplus预处理器宏,并在需要时插入extern "C"CoreFoundation提供了CF_EXTERN_C_BEGINCF_EXTERN_C_END宏来处理这个问题:

#if !defined(CF_EXTERN_C_BEGIN)
#if defined(__cplusplus)
#define CF_EXTERN_C_BEGIN extern "C" {
#define CF_EXTERN_C_END   }
#else
#define CF_EXTERN_C_BEGIN
#define CF_EXTERN_C_END
#endif
#endif

使用这些你的objc_funcs.h变成:

#import <Foundation/Foundation.h>
CF_EXTERN_C_BEGIN
NSString* doFoo(void);
CF_EXTERN_C_END

如果你不想用它们,你可以用

#import <Foundation/Foundation.h>
#ifdef __cplusplus
extern "C"
#endif
NSString* doFoo(void);

您必须通过将objc_funcs转换为.mm文件或将导入包装在main中使其对c++编译器有效。mm withextern "C" {}:

extern "C" {
#import "objc_funcs.h"
}

你可以在这里阅读更多信息

最新更新