6. c++和c的相互调用

本文介绍了如何在C++中调用C语言的代码,包括extern关键字的作用和__cplusplus预定义宏的含义。同时强调了C++编译器对重载函数的处理与C语言的不同。给出了创建可跨C/C++使用的模块的示例,包括头文件的条件编译声明和在C++中导入C库时使用extern "C"的方法。
  • c++ 编译器能够兼容C语言的编译方式
  • c++ 编译器会优先使用c++编译的方式
  • extern关键字会强制让c++编译器进行c方式的编译
extern "C"
{
#include "c.h"  //c方式编译出来的c.c     调用c.c中的f();
}
int main(void)
{
    f();
    return 0;
}
  • 为了保证一段c代码只会以c的方式编译(上面代码c编译器无法通过)
    • __cplusplus是c++编译器内置的标准宏定义
    • __cplusplus的意义
      • 确保c代码以统一的c方式被编译成目标文件
#include <stdio.h>


#ifdef __cplusplus
extern "C" {
#endif

#include "add.h"

#ifdef __cplusplus
}
#endif


int main()
{
    int c = add(1, 2);

    printf("c = %d\n", c);

    return 0;
}
注意
  • c++编译器不能以c的方式编译重载函数
  • 编译方式决定函数名被编译后的目标名
    • c++编译方式将函数名和参数列表编译成目标名
    • c编译方式只将函数名作为目标名进行编译

应用示例

  1. 现在要写一个c语言的模块,供以后使用(以后的项目可能是c的也可能是c++的),源文件事先编译好,编译成.so或.o都无所谓。头文件中声明函数时要用条件编译包含起来,如下:
#ifdef __cplusplus
extern "C" {
#endif

//some code

#ifdef __cplusplus
}
#endif

也就是把所有函数声明放在some code的位置。
2. 如果这个模块已经存在了,可能是公司里的前辈写的,反正就是已经存在了,模块的.h文件中没有extern “C”关键字,这个模块又不希望被改动的情况下,可以这样,在你的c++文件中,包含该模块的头文件时加上extern “C”, 如下:

extern "C" {
#include "test_extern_c.h"
}
### C语言与C++相互调用的方法及实现细节 在CC++之间进行相互调用时,需要解决符号命名差异的问题。以下是详细的实现方法: #### 1. C语言调用C++代码 为了使C语言能够正确调用C++代码,必须确保C++编译器生成的符号名称与C语言兼容。这通常通过`extern "C"`关键字来实现。 - 在C++头文件中使用`extern "C"`包裹函数声明[^1]: ```c #ifdef __cplusplus extern "C" { #endif void my_function(int a, int b); #ifdef __cplusplus } #endif ``` 上述代码块的作用是:当C++编译器遇到`extern "C"`时,它会将函数名以C风格导出,而不是使用C++的名称修饰机制[^2]。 - 确保C++函数被正确声明为`extern "C"`后,可以在C语言代码中直接包含该头文件并调用相关函数[^3]: ```c #include "my_header.h" int main() { my_function(10, 20); return 0; } ``` #### 2. C++调用C语言代码 C++调用C语言代码相对简单,因为C++编译器默认支持C语言的函数调用。同样需要使用`extern "C"`来告知C++编译器这些函数是以C方式链接的。 - 在C语言头文件中定义函数,并在C++代码中包含该头文件时使用`extern "C"`[^1]: ```c // add.c int add(int a, int b) { return a + b; } ``` ```cpp extern "C" { int add(int a, int b); } int main() { int result = add(5, 7); return 0; } ``` 通过这种方式,C++代码可以正确识别C语言函数并调用它们。 #### 3. 使用C++方式编译C代码 另一种方法是使用C++编译器(如`g++`)直接编译C代码。这种方法下,C代码会被视为C++代码的一部分,但需要注意C++编译器对语法类型的检查更为严格[^1]。 - 编译命令示例: ```bash g++ -o program main.cpp add.c ``` #### 4. 注意事项 - **宏检测**:可以通过宏`__cplusplus`来检测当前代码是否由C++编译器编译,从而有条件地应用`extern "C"`[^1]。 - **类型兼容性**:确保CC++之间的数据类型兼容,避免因类型不匹配导致的错误。 - **异常处理**:C++支持异常处理,而C语言不支持。如果C++代码中可能抛出异常,则需要在调用点进行适当的异常捕获[^2]。 #### 示例代码 以下是一个完整的示例,展示如何在CC++之间进行相互调用: - C语言代码(`add.c`): ```c int add(int a, int b) { return a + b; } ``` - C++代码(`main.cpp`): ```cpp extern "C" { int add(int a, int b); } void cpp_function() { printf("This is a C++ function.\n"); } int main() { int result = add(3, 4); printf("Result from C function: %d\n", result); cpp_function(); return 0; } ``` - 头文件(`common.h`): ```c #ifdef __cplusplus extern "C" { #endif int add(int a, int b); #ifdef __cplusplus } #endif ``` #### 编译命令 ```bash gcc -c add.c -o add.o g++ main.cpp add.o -o program ``` --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值