参见文章 C 语言里面的 extern "C" ,并没有那么简单!
开始正题
my_handle.c 源码
gcc编译生成.o中间文件,并查看符号表 nm xxx.o ,可以看到符号名字与源文件的名字保持一致
查看段 objdump xxx.o
My_handle_client.cpp 源码
使用g++编译生成中间文件并查看符号表,可以看到符号名字被g++编译器粉碎重构了
此时,两个目标中间文件:my_handle.o是用gcc生成的,而my_handle_client是用g++生成的
此时使用g++或gcc编译 ,链接器都将将报告相关的“符号未定义”错误,这是由于两个目标文件.o对于同一对象的命名不一
解决办法就是在源文件(头文件)中给出声明。exter "C" {} ,可以声明链接规范
修改my_handle.h头文件
随后使用g++重新编译my_handle_client.c,可以看到这三个函数的符号就是按照gcc的规则来了
再次链接就没错了,程序也可以正常执行
从中我们可以看出,此时,用extern "C" 修饰了的声明,其生成的符号和C语言编译器生成的符号保持了一致。这样,当你再次把my_handle.o和my_handle_client.o放在一起链接的时候,就不会再有之前的“符号未定义”错误了。
由于这个my_handle是之前使用gccc编译的(还没加extern “C”)时,这时我们尝试使用gcc再编一次my_handle.c
可以看到C语言编译器将会报告“语法错误”,因为extern"C"是C++的语法,C语言编译器不认识它。此时,可以按照我们之前已经讨论的,使用宏__cplusplus来识别C和C++编译器(__cplusplus是一个C++规范规定的预定义宏,所有的现代C++编译器都预先定义了它;而所有C语言编译器则不会
)。
改写my_handle.h,如下
随后再编译,就没问题了