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

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...