为什么从 objc++ 调用 objc 函数会导致构建失败?
Why calling objc function from objc++ could result in build failure?
鉴于此 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);
鉴于此 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" {}
:
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);