如何实现在C++中调用C函数

 更新时间:2023年08月02日 09:02:10   作者:花好约猿上王者  
这篇文章主要介绍了如何实现在C++中调用C函数问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

如何在C++中调用C函数

假如在一个项目中同时包含了C和C++,当C++调用C函数时,以传统C编程#include "xxx.h"后,声明函数。

由于main.cpp 是个C++代码,以C方式的调用,g++编译器无法通过编译。

解决方案一

重写一个专门被c++用的头文件(可能存在是别人已经写好的头文件,我们无法修改等问题)

e.g. 添加一个头文件 tansfer.h

extern "C" {
#include "file1.h"
#include "file2.h"
}

最后在需要调用C函数的cpp文件(即C++文件)中,引用头文件 tansfer.h即可。

解决方案二

在xxx.h中的每个函数最前面添加:extern “C”

extern "C" { 
    void fun1(int arg1); 
    void fun2(int arg1, int arg2); 
    void fun3(int arg1, int arg2, int arg3); 
}

解决方案三

在代码兼容而且少的前提下,更改C文件为cpp文件。

C++程序调用c函数(extern “C“的使用)

概念

c++程序调用c函数,即为在xx.cpp文件中调用在func.c文件中实现的函数

extern "C"的主要作用是为了实现c++代码能够调用c语言代码。加上extern "C"后,这部分代码编译器按c语言的方式进行编译和链接,而不是c++的方式。

原理

由于c++中需要支持函数重载,所以c和c++中对同一个函数经过编译后生成的函数名是不同的,这就会导致一个问题,如果在c++中调用一个使用c语言编写模块中的某个函数,c++是按照c++的名称修饰方式来查找并链接这个函数,就会发生链接错误。

  • c函数:void func(),被编译成函数:func
  • c++函数: void func(){},被编译成函数:_Z4funcv

示例代码

1. 处理被调用的C头文件

// my_module.h
#pragma once
#include<stdio.h>
#ifdef __cplusplus
extern "C"{
#endif
	void func1();
	int func2(int a,int b);
#ifdef __cplusplus
}
#endif
// my_module.c
#include "my_module.h"
void func1() {
  printf("hello world.");
}
int func2(int a, int b) {
  return a+b;
}
// test.cpp
#include "my_module.h"
#include<iostream>
using namespace std;
int main() {
  func1();
  cout << func2(10, 20) << endl;
}

编译:

gcc -c my_module.c -o my_module.o
gcc -c test.cpp -o test.o
g++ *.o -o test   // c++代码要用g++进行链接

.c文件和.cpp不能混在一起编译,要分别编译成目标文件.o,再进行链接生成可执行程序

2. 处理调用的c++文件

// my_module.h
#pragma once
#include <stdio.h>
void func1();
int func2(int a, int b);
// my_module.c
#include "my_module.h"
void func1() {
  printf("hello world.");
}
int func2(int a, int b) {
  return a+b;
}
// test.cpp
#include <iostream>
using namespace std;
extern "C" {
#include "my_module.h"
}
int main() {
  func1();
  cout << func2(1, 2) << endl;
}

编译方式同上

基础补充

gcc、g++编译常用选项:

选项含义
-o file指定生成的输出文件名为file
-E只进行预处理
-S(大写)只进行预处理和编译
-c(小写)只进行预处理、编译和汇编

C语言分步编译:

1)预处理:宏定义展开、头文件展开、条件编译等,删除注释,这里不会检查语法

gcc -E test.c -o test.i

2)编译:检查语法,将预处理后文件编译成汇编文件

gcc -S test.i -o test.s

3)汇编:将汇编文件生成目标文件(二进制文件)

gcc -c test.s -o test.o

4)链接:C语言写的程序是需要依赖各种库的,编译后还需要把库链接到最终的可执行程序中去

gcc test.o -o test

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • FFmpeg实现多线程编码并保存mp4文件

    FFmpeg实现多线程编码并保存mp4文件

    这篇文章主要为大家介绍了FFmpeg如何持续的从指定内存中读取原始数据,再将解码数据存入队列中,并通过单独的线程进行编码,最后保存为mp4文件,感兴趣的可以了解下
    2023-08-08
  • 基于C++实现读取指定路径文件

    基于C++实现读取指定路径文件

    这篇文章主要介绍了基于C++实现读取指定路径文件,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-07-07
  • C++编程之CString、string与、char数组的转换

    C++编程之CString、string与、char数组的转换

    这篇文章主要介绍了C++编程之CString、string与、char数组的转换的相关资料,希望通过本文能帮助到大家,让大家学习理解这部分内容,需要的朋友可以参考下
    2017-10-10
  • 详解C++文件读写操作

    详解C++文件读写操作

    这篇文章主要为大家详细介绍了C++文件读写操作,感兴趣的小伙伴们可以参考一下
    2016-03-03
  • VS2019安装配置MFC(安装vs2019时没有安装mfc)

    VS2019安装配置MFC(安装vs2019时没有安装mfc)

    这篇文章主要介绍了VS2019安装配置MFC(安装vs2019时没有安装mfc),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-03-03
  • C++计数排序详解

    C++计数排序详解

    计数排序的思想我们之前接触过的例如:插入排序,归并排序,快速排序,堆排序等都是基于集合元素之间的比较这一基本的思想,它们执行的时间复杂度最优是趋于O(nlgn),而计数排序的运行机制不是基于集合元素之间的大小比较
    2016-04-04
  • C++核心编程之占位参数和默认参数

    C++核心编程之占位参数和默认参数

    这篇文章主要介绍了C++核心编程之占位参数和默认参数,c++中函数的形参列表中的形参是可以有默认值的,函数的形参列表里可以有占位参数,用来占位,调用函数时必须填补位置。下面更多相关内容的详细介绍,需要的小伙伴可以参考一下
    2022-03-03
  • Qt之ui在程序中的使用-多继承法介绍

    Qt之ui在程序中的使用-多继承法介绍

    本文将介绍Qt之ui在程序中的使用-多继承法,需要的朋友可以参考
    2012-11-11
  • C语言中四种取整方式,取余/取模运算以及负数取模问题详解

    C语言中四种取整方式,取余/取模运算以及负数取模问题详解

    这篇文章主要介绍了C语言中四种取整方式及负数取模问题,包括了算法的分析与改进,是很多程序设计竞赛中常见的算法,需要的朋友可以参考下
    2021-09-09
  • C语言 typedef:给类型起一个别名

    C语言 typedef:给类型起一个别名

    本文主要介绍C语言 typedef,这里整理了相关资料及简单示例代码帮助大家学习理解,有兴趣的小伙伴可以参考下
    2016-08-08

最新评论