如何解决为什么从 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? ?
解决方法
您必须通过将 objc_funcs 转换为 .mm 文件或使用 extern "C" {}
在 main.mm 中包装导入来使其对 C++ 编译器有效:
extern "C" {
#import "objc_funcs.h"
}
您可以阅读更多相关信息here
,问题与C vs C++ linkage有关。
在 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);
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。