当我尝试在类接口上方的头文件中定义C函数时,我总是遇到构建错误。
但是当我在实现文件中执行相同操作并在标头中给出声明时。事情很顺利。
我想知道,为什么会这样,因为我在头文件中定义了枚举,结构,常量NSStrings,为什么不用C函数呢?
答案 0 :(得分:17)
这与C链接器(或链接编辑器)的工作方式有关。当C编译器遇到函数定义时,它会准备实现该函数的汇编程序代码,并使用符号对其进行标记,该符号表示链接器“这是具有此名称的函数的开始位置”。该符号通常以下划线命名,后跟函数名称,例如_printf
。
如果在头文件中定义函数,则导入此头的每个.c
或.m
文件都将编译该函数,并使编译器发出相同的符号。链接器期望只找到每个符号的一个实例,因此这是一个错误。
这与#include
警卫的存在或使用#import
而不是#include
无关。 C编译器适用于单个转换单元 - 它意味着单个源文件。预处理器策略会阻止您将相同的头文件两次包含在单个源文件中,但不会在多个文件之间协调活动。这意味着在不同的源文件中包含相同的头文件是有效的:它还意味着当您编译不同的文件时,它们可以(合法地)包含相同的符号。
链接编辑器的工作是将这些文件放在一起,解决对编译时未知的符号的任何引用。如果您尝试将具有相同符号的对象(已编译和已组装的翻译单元的名称)链接到同一个存档,共享库或可执行文件中,那么您将收到您在此处看到的错误。
解决方案:
inline
在标题中定义函数。内联函数只是由编译器复制到它们被调用的函数中,因此永远不会为它们发出链接器符号。这有你自己的权衡may wish to read more about。