问题描述
三个文件:
/* c语言头文件:cExample.h */#ifndef C_EXAMPLE_H#define C_EXAMPLE_Hextern int add(int x,int y); //注:写成extern 'C' int add(int , int ); 也可以#endif
/* c语言实现文件:cExample.c */#include 'cExample.h'int add( int x, int y ){ return x + y;}
// c++实现文件cppFile.cpp,调用addextern 'C'{#include 'cExample.h'}//注:此处不妥,如果这样编译通不过,换成 extern 'C' int add(int , int ); 可以通过extern 'C' int add(int, int);int main(int argc, char* argv[]){ add(2, 3); return 0;}
这都是别人博客的代码,他说不会错,然后我在VS2015运行了,报错,错误 LNK2019 无法解析的外部符号 _add,该符号在函数 _main 中被引用。我换了好几个人博客把代码复制进去都是报同样的错误,我不知道是什么问题,上面注释里的我也试了还是报错。
问题解答
回答1:找到错误了,我没有把.c和.h拖至工程里,只是放在该目录下了。已解决。
回答2:刚看到,楼主已经自己解决了问题。不过,这个例子在 gcc 这里还是错的:
$ g++ -std=c++11 -pedantic -Werror cExample.c cppFile.cpp -o demo/tmp/cceHYgc4.o: In function `main’:cppFile.cpp:(.text+0xf): undefined reference to `add’collect2: error: ld returned 1 exit status
这是因为 c++ 编译器在编译 cExample.c 的时候,虽然你的意图是调用 C 编译器去编译 add 函数的代码,但是 c++ 编译器并不知道这些,它认为 add 函数是一个 c++ 函数,于是,编译后 add 函数的名字会被改为 _Z3addii。这样,即使你在 cppFile.cpp 文件中指定要使用 C 版本的 add 函数,但是这个函数事实上不存在,所以就出现了无法解析的外部符号。
要解决这个问题,只需要在 cExample.h 文件中这样写:
#ifndef C_EXAMPLE_H#define C_EXAMPLE_Hextern 'C' {int add(int x, int y);}#endif
如果 cExample.h 中声明的函数要被其他 C 模块调用,那么严谨的写法是:
#ifndef C_EXAMPLE_H#define C_EXAMPLE_H#ifdef __cplusplusextern 'C' {#endifint add(int x, int y);#ifdef __cplusplus}#endif#endif
至于为什么在 VS 中将 cExample.{h,c} 拖到工程中就可以解决问题,这就不得而知了。猜测是 VS 遇到文件后缀名为 .c 的文件,就自动用 C 编译器去编译。