c与c++之间的相互调用及函数区别示例详解

 更新时间:2023年06月13日 09:27:18   作者:p了个c  
这篇文章主要为大家介绍了c与c++相互调用的使用示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

引言

最近项目需要使用google test(以下简称为gtest)作为单元测试框架,但是项目本身过于庞大,main函数无从找起,需要将gtest框架编译成静态库使用。因为项目本身是通过纯c语言编写,而gtest则是一个c++编写的测试框架,其中必然涉及c与c++之间的相互调用。

注意,本文的前提是,c代码采用gcc等c语言编译器编译c代码,采用g++等c++编译器编译c++代码,如果c和c++代码统一使用g++编译,大部分情况是可以实现两者代码相互调用的。以下为踩坑过程的总结o_O||。

c与c++的函数区别

要了解两者之间如何实现相互调用,必须先了解c与c++之间的函数有什么不同。

c++作为c语言的升级版,两者必然有很多不同之处。其中有一个重大不同点就是,c++支持函数重载,而c语言不支持。为了使函数支持重载,c++在c语言的基础上,将函数名添加上返回值和参数的类型信息。

例如,int add(int, int)这个函数,通过c++编译器编译后,可能呈现的函数名为int int_add_int_int(int, int)(注:此处为大概地说明c++是如何将返回值和参数信息添加到函数名中的,实际中编译器不一定是这样实现的)。

从以上说明可以得出,由于c++对函数重载的支持,使得编译后的函数符号与c语言的不一致,即使是在两者函数名相同的前提下。

extern "C"的作用

那么,c与c++是不能相互调用了吗?答案是否定的,因为存在着extern "C"这个关键字可以使语句可以按照类C的编译和连接规约来编译和连接,而不是C++的编译的连接规约。这样在类C的代码中就可以调用C++的函数or变量等。

注意:extern "C"指令中的"C",表示的一种编译和连接规约,而不是一种语言。"C"表示符合C语言的编译和连接规约的任何语言,如Fortran、assembler等。

还有要说明的是,extern "C"指令仅指定编译和连接规约,但不影响语义。例如在函数声明中,指定了extern "C",仍然要遵守C++的类型检测、参数转换规则。

c++中调用c代码

对于c++,由于c++的编译器对c语言兼容,因此在c++中调用c语言编写的函数,只需要在函数声明前面加上关键字extern "C",表示采用类c语言的方式解析函数符号。例子如下:

// add.h
#ifdef __ADD_H__
#define __ADD_H__
extern "C" int add(int a, int b);
#endif
// add.c
int add(int a, int b)
{
    return a + b;
}
// main.cc
#include <iostream>
#include "add.h"
using namespace std;
int main()
{
    cout << "1 + 1 = " << add(1, 1) << endl;
}

在例子中,main.cc为c++代码,add.c为c语言代码,当c++编译器识别到extern "C"`关键字时,会去寻找add函数的实现而不是寻找类似int_add_int_int这样带参数信息的函数实现。

c语言调用c++代码

c语言调用c++代码却并不容易,原因是c语言并不兼容c++。就算c语言可以调用c++,也会因为无法识别c++新定义的符号而编译报错。因此,为了实现c语言调用c++函数,必须实现以下两个步骤:1. 将c++相关函数封装为静态库或动态库(因为调用库函数时编译器并不知道里面执行的是什么语言);2. 对外提供遵循类c语言规约的接口函数。例子如下所示:

// printNum.h
#ifdef __PRINTNUM_H__
#define __PRINTNUM_H__
extern "C" void printNum(int a);
#endif
// printNum.cc
#include <iostream>
#include "printNum.h"
using namespace std;
void printNum(int a)
{
    cout << << "num is " << a << endl;
}
// main.c
extern void printNum(int a);
printNum(5);

通过将cout函数封装为类c语言规约的接口函数,使得main.c中可以成功调用c++函数printNum。值得注意的是,main.c不可以直接引入printNum.h,因为c语言不能识别extern "C"关键字。可以利用c++预定义宏实现头文件的改写:

#ifdef __PRINTNUM_H__
#define __PRINTNUM_H__
#ifdef __cplusplus
extern "C" { 
#endif
void printNum(int a);
#ifdef __cplusplus
}
#endif
#endif

小结

小结如下:

  • c语言与c++的相互调用可以通过extern "C"关键字实现
  • c++中调用c代码,只须在c++中为c代码函数声明之前加上extern "C"
  • c语言调用c++代码,则需要将c++代码编译成静态库或动态库,然后对外提供用extern "C"声明的类c封装函数

以上就是c与c++之间的相互调用及函数区别示例详解的详细内容,更多关于c c++相互调用的资料请关注脚本之家其它相关文章!

相关文章

  • 最大子矩阵问题实例解析

    最大子矩阵问题实例解析

    这篇文章主要介绍了最大子矩阵问题实例解析,分别列举了Java和C语言的相关实现,需要的朋友可以参考下
    2015-08-08
  • VS2022+libtorch+Cuda11.3安装测试教程详解(调用cuda)

    VS2022+libtorch+Cuda11.3安装测试教程详解(调用cuda)

    这篇文章主要介绍了VS2022+libtorch+Cuda11.3安装测试(调用cuda),本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-05-05
  • C++ 将字符串值赋给CHAR数组的实现

    C++ 将字符串值赋给CHAR数组的实现

    这篇文章主要介绍了C++ 将字符串值赋给CHAR数组的实现,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-01-01
  • C++中const的常见用法详解

    C++中const的常见用法详解

    const名叫常量限定符,用来限定特定变量,以通知编译器该变量是不可修改的,本文为大家整理了const的几种使用,感兴趣的小伙伴可以跟随小编一起了解一下
    2023-06-06
  • C++逐步介绍日期类的使用

    C++逐步介绍日期类的使用

    下面小编就为大家带来一篇C++实现日期类(Date类)的方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2022-07-07
  • C++AVL树4种旋转详讲(左单旋、右单旋、左右双旋、右左双旋)

    C++AVL树4种旋转详讲(左单旋、右单旋、左右双旋、右左双旋)

    AVL树即平衡二叉搜索树,平衡因子bf=右子树的高度-左子树的高度,bf为0,-1,1时,此树即平衡,下面这篇文章主要给大家介绍了关于C++AVL树4种旋转(左单旋、右单旋、左右双旋、右左双旋)的相关资料,需要的朋友可以参考下
    2022-11-11
  • C++11/14 线程的创建与分离的实现

    C++11/14 线程的创建与分离的实现

    这篇文章主要介绍了C++11/14 线程的创建与分离的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-01-01
  • 使用C语言实现贪吃蛇小游戏

    使用C语言实现贪吃蛇小游戏

    这篇文章主要为大家详细介绍了使用C语言实现贪吃蛇小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-07-07
  • 用C语言实现一个扫雷小游戏

    用C语言实现一个扫雷小游戏

    这篇文章主要为大家详细介绍了用C语言实现一个扫雷小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-09-09
  • C语言中队列的结构和函数接口的使用示例

    C语言中队列的结构和函数接口的使用示例

    队列只允许一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有先进先出FIFO的性质;队列可用数组和链表 的方法实现,使用链表的结构实现更优一些,因为如果使用数组节,出队列时删去首元素需要将整个数组前移,效率比较低
    2023-02-02

最新评论