c/c++ extern “C”
常见的样式
extern “C”{ ... }
extern "C" return-type func-name(type , type ){}
extern "C" return-type func-name(type , type );
含义:在C++中,以C语言的方式编译源文件。
例子//t_extern_yes.cpp
- extern "C"
- {
- int f(void)
- {
- return 0;
- }
- }
//t_extern_non.cpp
- int f(void)
- {
- return 0;
- }
汇编
- gcc t_extern_non.cpp t_extern_yes.cpp
- 或者
- gcc t_extern_non.cpp t_extern_yes.cpp
得到
- .file "t_extern_non.cpp"
- .text
- .align 2
- .globl _Z1fv
- .type _Z1fv, @function
- _Z1fv:
- .LFB2:
- pushl %ebp
- .LCFI0:
- movl %esp, %ebp
- .LCFI1:
- movl $0, %eax
- popl %ebp
- ret
- .LFE2:
- .size _Z1fv, .-_Z1fv
- .globl __gxx_personality_v0
- .ident "GCC: (GNU) 4.2.1 (SUSE Linux)"
- .section .note.GNU-stack,"",@progbits
- .file "t_extern_yes.cpp"
- .text
- .align 2
- .globl f
- .type f, @function
- f:
- .LFB2:
- pushl %ebp
- .LCFI0:
- movl %esp, %ebp
- .LCFI1:
- movl $0, %eax
- popl %ebp
- ret
- .LFE2:
- .size f, .-f
- .globl __gxx_personality_v0
- .ident "GCC: (GNU) 4.2.1 (SUSE Linux)"
- .section .note.GNU-stack,"",@progbits
用imdiff看时只有上面红色标志的是不同的,这是函数名字编译后的ID,C++ 丰富了内容。如果用extern “C”,告诉C++编译器,使用C的编译方式编译,这样就可以连接原来的用C编译出来的C文件了。现在模拟一个C++调用C库文件的现象。
C++ 调用C库
//t_add.c
- int add(int a, int b){
- return (a+b);
- }
得到c的静态库文件t_add.o
- gcc -c t_add.c
原始的C头文件
- #ifndef T_ADD_C__
- #define T_ADD_C__
- int add(int a, int b);
- #endif
修改为可以被C++ 使用的头文件//t_add.h
- #ifndef T_ADD_C__
- #define T_ADD_C__
- extern "C" int add(int a, int b);
- #endif
可以被C++文件引用了,如//main.cpp
- #include <stdio.h>
- #include "t_add.h"
- int main(void)
- {
- printf("1 + 2 = %d\n", add(1,2));
- return 0;
- }
可以编译了
- g++ t_add.o main.c
输出
- 1 + 2 = 3
使用方式
当然了,extern “C”的方式多种,
- 可以在函数前,
- 文件开始结尾 {}hold住,
- 也可以方便的在 #inldue语句加,如
- extern "C" {
- #include "t_add.h"
- }
另外,C++变压器会有 __cplusplus 的宏定义,一般都是结合使用