给定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_BEGIN
和CF_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"
}
你可以在这里阅读更多信息