为什么从 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_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);