c语言和c++的相互调用_c调用c++-程序员宅基地

技术标签: c++  c  相互调用  C/C++编程  extren-C  cplusplu  c/c++语言  

本文收录于微信公众号「 LinuxOK 」,ID为:Linux_ok,关注公众号第一时间获取更多技术学习文章。

在实际项目开发中,c和c++代码的相互调用是常见的,c++能够兼容c语言的编译方式,但是c++编译器g++默认会以c++的方式编译程序,而c程序编译器gcc会默认以c的方式编译它,所以c和c++的相互调用存在一定的技巧。

####1.c方式编译和c++方式编译
一般.cpp文件是采用g++去编译,.c文件是采用gcc编译,然而这不是绝对的。
(1)gcc和g++都可以编译.c文件,也都可以编译.cpp文件。g++和gcc是通过后缀名来辨别是c程序还是c++程序的(这一点与Linux辨别文件的方式不同,Linux是通过文件信息头辨别文件的)。
(2)在gcc看来,.c文件会以c方式去编译,.cpp文件则是以c++的方式去编译,注意,gcc不会主动去链接c++用到库stdc++,所以用gcc编译cpp文件时需要手动指定链接选项-lstdc++。而对于g++,不管是.c还是.cpp文件,都是以c++方式去编译。
(3)还需要注意,并不是说__cpluscplus 是g++编译器才会定义的宏,确切的说,是只有以c++编译的方式去编译文件的宏才会定义的宏,这样说来,gcc编译.cpp文件、g++编译.c、.cpp文件,这个 __cplusplus都会被编译器定义。

####2.c++调用c程序
假设c程序是之前写好的具有价值的静态库,该库是由add.o和sub.o编译而成,而add.c和sub.c是由c语言写的:
add.h和add.c

//add.h
#ifndef __ADD_H__
#define __ADD_H__
int add(int, int);
#endif /* __ADD_H__ */

//add.c
#include "add.h"

int add(int a, int b)
{
	return a + b;
}

sub.h和sub.c

//sub.h
#ifndef __SUB_H__
#define __SUB_H__
int sub(int, int);
#endif /* __SUB_H__ */

//sub.c
#include <stdio.h>
#include "sub.h"
int sub(int a, int b)
{
    printf("%d - %d = %d\n", a, b, a - b);
    return 0;
}

将各自编译成.o文件,并打包成静态库

$ gcc -c add.c
$ gcc -c sub.c
$ ar cqs libadd.a *.o

这样就生成了libadd.a静态库。
若要编译成动态库,则

$ gcc -shared -fPIC *.o -o libadd.so

main.cpp是c++写的,用g++编译。

//main.cpp
#include "add.h"
#include "sub.h"
#include <stdio.h>

int main(void)
{
    int c = add(1, 6);
    sub(8, 2);
    printf("c = %d\n", c);
    return 0;
}

编译:
这里写图片描述
报错未定义的add和sub函数。
通过nm确实是可以看到add和sub标号的存在的:
这里写图片描述
原因在于,main.cpp是c++文件,用g++编译器编译时(gcc也是一样的结果),会优先选择cpp的编译方式,也就是会用cpp的编译方式去编译add()、sub()函数。然而,它们是.c文件,用的是c语言的方式去编译的,所以出现如上问题。注意,cpp编译器是兼容c语言的编译方式的,所以在编译cpp文件的时候,调用到.c文件的函数的地方时,需要用extern "C"指定用c语言的方式去编译它:

//使得add.h、sub.h里面的代码用c语言的方式去编译
extern "C"{
#include "add.h"
#include "sub.h"
}
#include <stdio.h>
int main(void)
{
    int c = add(1, 6);
    printf("c = %d\n", c);
    return 0;
}

编译运行
这里写图片描述
特别注意extern "C"是c++方式编译才认识的关键字。

####3. c语言调用c++程序
c语言用c方式编译,c++程序用c++方式,要使得c语言能调用c++程序无非有2种方法(c++调用c也是一样)
(1) c语言程序用c++方式编译
既然是c语言调用c++程序,肯定是要采取c++方式编译,所以觉得这个没什么意义。操作很简单,用g++方式编译c程序即可,注意,g++会对语法、类型等更为严格的检查。
(2) c++程序用c语言方式编译
a. c方式编译和c++方式编译,其差异就在于符号表标识符。同一个函数名,在c方式编译的其函数名跟编译前的函数一致,c++方式编译的则是以函数名结合参数作为编译后的函数名。要确保文件以.c方式编译,可以利用__cplusplus,这个宏在c++编译的方式才会定义的宏,结合之前的extern C用法如下:

#ifdef __cplusplus
extern "C"{
#endif /* __cplusplus */

//用c方式编译的代码

#ifdef __cplusplus
}
#endif /* __cplusplus */

这样,对于一份.c文件,采用gcc编译时候没有定义__cplusplus,宏判断不起作用,且自是用c语言的方式编译,采用g++编译定义了_cplusplus,经过上面宏判断,所以还是会以c语言的方式编译。注意,extern “C”是g++才具有的关键字,gcc并没有,所以如果用gcc编译而不加以宏判断直接使用extern “C”那么就会出现语法错误。
用c方式去编译c++文件,还要注意重载函数。c方式编译的c++文件决定不能出现重载函数。尝试extern “C”中出现重载函数:

#ifdef __cplusplus
extern "C"{
#endif /* __cplusplus */
int func(int a, int b){
    return a + b;
}

void func(const char* str){
    printf("str = %s\n", str);
}
#ifdef __cplusplus
}
#endif /* __cplusplus */

int main(void)
{
    func("hello");
    return 0;
}

这里写图片描述
提示找不到func函数。在c语言中肯定不能出现同名函数,不然编译器怎么知道它要调用的是哪一个。去除extern “C”关键字,也就是使其采用c++方式编译,编译结果:
这里写图片描述
编译通过。通过nm命令可以查看可执行程序的符号表:
这里写图片描述
可见c++编译方式会将重载函数名结合参数形成唯一的函数名。但是c方式编译的可执行文件却并非如此,函数名经过编译后的符号还是之前的函数名,所以出现找不到调用函数的现象。
b. c程序要调用c++写的程序,涉及到的可能有:c程序调用c++的普通函数,c程序调用c++的重载函数,c程序调用c++的成员函数(包括虚函数)。c程序自然是采用c的方式去编译的,即是采用gcc编译器,然而c++是采用c++方式编译,所以要强制将c++代码以c的方式去编译。
(1) c程序调用c++的普通函数

add.h和add.cpp
//add.h
#ifndef _ADD_H_
#define _ADD_H_
int add(int, int);
#endif /* _ADD_H_ */
//add.cpp
#include <iostream>
extern "C" {    //以c的方式去编译
int add(int a, int b)
{
    std::cout<<"a + b = "<< a + b << std::endl;
    return 0;
}
}

注意不能在声明add函数的add.h中指定以c的方式去编译,因为add.h是要被main.c文件包含的,c方式编译时并不能认得extern “C”关键字。
main.c

//main.c
#include <stdio.h>
#include "add.h"
int main(void)
{
    add(4, 9);
    return 0;
}

编译运行:
这里写图片描述
因为add.cpp用到标准c++库,所以要手动链接该库。
(2)c调用c++的重载函数
前面已经知道,在extern “C”中是不允许出现重载函数的,因为c方式下的程序并不支持重载函数,所以需要对重载函数进行接口封装。使用extern “C”是要告诉编译器按照c的方式来编译封装接口,接口里面的函数还是按照c++的语法和c++方式编译。
add.h和add.cpp

//add.h
#ifndef _ADD_H_
#define _ADD_H_
int add_ii(int, int);
int add_c(char);
#endif /* _ADD_H_ */

上面两个函数是供c程序调用的,自然需要用c方式编译。而这两个函数的实现体可以去调用c++的重载函数,这就完美契合了。

//add.cpp
#include <iostream>
int add(int a, int b)
{
    std::cout<< "a + b = " << a + b << std::endl;
    return 0;
}

int add(char c)
{
    std::cout << "c = " << c << std::endl;
}

extern "C" int add_ii(int a, int b)  //供c程序调用,用c方式编译
{
    add(a, b);
}

extern "C" int add_c(char c)
{
    add(c);
}

main.cpp

//main.cpp
#include <stdio.h>
#include "add.h"

int main(void)
{
    add_ii(4, 9);
    add_c('d');
    return 0;
}

编译运行:
这里写图片描述
(3)c程序调用c++的成员函数
这个比较复杂,先不说。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_29344757/article/details/73332501

智能推荐

使用nginx解决浏览器跨域问题_nginx不停的xhr-程序员宅基地

文章浏览阅读1k次。通过使用ajax方法跨域请求是浏览器所不允许的,浏览器出于安全考虑是禁止的。警告信息如下:不过jQuery对跨域问题也有解决方案,使用jsonp的方式解决,方法如下:$.ajax({ async:false, url: 'http://www.mysite.com/demo.do', // 跨域URL ty..._nginx不停的xhr

在 Oracle 中配置 extproc 以访问 ST_Geometry-程序员宅基地

文章浏览阅读2k次。关于在 Oracle 中配置 extproc 以访问 ST_Geometry,也就是我们所说的 使用空间SQL 的方法,官方文档链接如下。http://desktop.arcgis.com/zh-cn/arcmap/latest/manage-data/gdbs-in-oracle/configure-oracle-extproc.htm其实简单总结一下,主要就分为以下几个步骤。..._extproc

Linux C++ gbk转为utf-8_linux c++ gbk->utf8-程序员宅基地

文章浏览阅读1.5w次。linux下没有上面的两个函数,需要使用函数 mbstowcs和wcstombsmbstowcs将多字节编码转换为宽字节编码wcstombs将宽字节编码转换为多字节编码这两个函数,转换过程中受到系统编码类型的影响,需要通过设置来设定转换前和转换后的编码类型。通过函数setlocale进行系统编码的设置。linux下输入命名locale -a查看系统支持的编码_linux c++ gbk->utf8

IMP-00009: 导出文件异常结束-程序员宅基地

文章浏览阅读750次。今天准备从生产库向测试库进行数据导入,结果在imp导入的时候遇到“ IMP-00009:导出文件异常结束” 错误,google一下,发现可能有如下原因导致imp的数据太大,没有写buffer和commit两个数据库字符集不同从低版本exp的dmp文件,向高版本imp导出的dmp文件出错传输dmp文件时,文件损坏解决办法:imp时指定..._imp-00009导出文件异常结束

python程序员需要深入掌握的技能_Python用数据说明程序员需要掌握的技能-程序员宅基地

文章浏览阅读143次。当下是一个大数据的时代,各个行业都离不开数据的支持。因此,网络爬虫就应运而生。网络爬虫当下最为火热的是Python,Python开发爬虫相对简单,而且功能库相当完善,力压众多开发语言。本次教程我们爬取前程无忧的招聘信息来分析Python程序员需要掌握那些编程技术。首先在谷歌浏览器打开前程无忧的首页,按F12打开浏览器的开发者工具。浏览器开发者工具是用于捕捉网站的请求信息,通过分析请求信息可以了解请..._初级python程序员能力要求

Spring @Service生成bean名称的规则(当类的名字是以两个或以上的大写字母开头的话,bean的名字会与类名保持一致)_@service beanname-程序员宅基地

文章浏览阅读7.6k次,点赞2次,收藏6次。@Service标注的bean,类名:ABDemoService查看源码后发现,原来是经过一个特殊处理:当类的名字是以两个或以上的大写字母开头的话,bean的名字会与类名保持一致public class AnnotationBeanNameGenerator implements BeanNameGenerator { private static final String C..._@service beanname

随便推点

二叉树的各种创建方法_二叉树的建立-程序员宅基地

文章浏览阅读6.9w次,点赞73次,收藏463次。1.前序创建#include&lt;stdio.h&gt;#include&lt;string.h&gt;#include&lt;stdlib.h&gt;#include&lt;malloc.h&gt;#include&lt;iostream&gt;#include&lt;stack&gt;#include&lt;queue&gt;using namespace std;typed_二叉树的建立

解决asp.net导出excel时中文文件名乱码_asp.net utf8 导出中文字符乱码-程序员宅基地

文章浏览阅读7.1k次。在Asp.net上使用Excel导出功能,如果文件名出现中文,便会以乱码视之。 解决方法: fileName = HttpUtility.UrlEncode(fileName, System.Text.Encoding.UTF8);_asp.net utf8 导出中文字符乱码

笔记-编译原理-实验一-词法分析器设计_对pl/0作以下修改扩充。增加单词-程序员宅基地

文章浏览阅读2.1k次,点赞4次,收藏23次。第一次实验 词法分析实验报告设计思想词法分析的主要任务是根据文法的词汇表以及对应约定的编码进行一定的识别,找出文件中所有的合法的单词,并给出一定的信息作为最后的结果,用于后续语法分析程序的使用;本实验针对 PL/0 语言 的文法、词汇表编写一个词法分析程序,对于每个单词根据词汇表输出: (单词种类, 单词的值) 二元对。词汇表:种别编码单词符号助记符0beginb..._对pl/0作以下修改扩充。增加单词

android adb shell 权限,android adb shell权限被拒绝-程序员宅基地

文章浏览阅读773次。我在使用adb.exe时遇到了麻烦.我想使用与bash相同的adb.exe shell提示符,所以我决定更改默认的bash二进制文件(当然二进制文件是交叉编译的,一切都很完美)更改bash二进制文件遵循以下顺序> adb remount> adb push bash / system / bin /> adb shell> cd / system / bin> chm..._adb shell mv 权限

投影仪-相机标定_相机-投影仪标定-程序员宅基地

文章浏览阅读6.8k次,点赞12次,收藏125次。1. 单目相机标定引言相机标定已经研究多年,标定的算法可以分为基于摄影测量的标定和自标定。其中,应用最为广泛的还是张正友标定法。这是一种简单灵活、高鲁棒性、低成本的相机标定算法。仅需要一台相机和一块平面标定板构建相机标定系统,在标定过程中,相机拍摄多个角度下(至少两个角度,推荐10~20个角度)的标定板图像(相机和标定板都可以移动),即可对相机的内外参数进行标定。下面介绍张氏标定法(以下也这么称呼)的原理。原理相机模型和单应矩阵相机标定,就是对相机的内外参数进行计算的过程,从而得到物体到图像的投影_相机-投影仪标定

Wayland架构、渲染、硬件支持-程序员宅基地

文章浏览阅读2.2k次。文章目录Wayland 架构Wayland 渲染Wayland的 硬件支持简 述: 翻译一篇关于和 wayland 有关的技术文章, 其英文标题为Wayland Architecture .Wayland 架构若是想要更好的理解 Wayland 架构及其与 X (X11 or X Window System) 结构;一种很好的方法是将事件从输入设备就开始跟踪, 查看期间所有的屏幕上出现的变化。这就是我们现在对 X 的理解。 内核是从一个输入设备中获取一个事件,并通过 evdev 输入_wayland

推荐文章

热门文章

相关标签