相关文章推荐
逆袭的红酒  ·  倚天屠龙:Github Copilot ...·  1 年前    · 
性感的夕阳  ·  享年91岁!图灵奖得主、软件工程圣经《人月神 ...·  2 年前    · 
没读研的小蝌蚪  ·  Spark ...·  2 年前    · 
温文尔雅的皮蛋  ·  内核该怎么学?Linux内核进程管理工作原理 ...·  2 年前    · 
率性的大熊猫  ·  音视频开发技术(19)FFmpeg命令过滤器 ...·  2 年前    · 
Code  ›  C++如何调用写好的C接口?开发者社区
https://cloud.tencent.com/developer/article/1888537
活泼的领结
2 年前
作者头像
DeROy
0 篇文章

C++如何调用写好的C接口?

前往专栏
腾讯云
开发者社区
文档 意见反馈 控制台
首页
学习
活动
专区
工具
TVP
文章/答案/技术大牛
发布
首页
学习
活动
专区
工具
TVP
返回腾讯云官网
社区首页 > 专栏 > 编程学习基地 > C++如何调用写好的C接口?

C++如何调用写好的C接口?

作者头像
DeROy
发布 于 2021-10-13 15:32:30
565 0
发布 于 2021-10-13 15:32:30
举报

前言

如何在C++代码中调用写好的C接口?你可能会奇怪,C++不是兼容C吗?直接调用不就可以了,那么我们来测试一下,先看看C++如何调用C代码接口的。

C++调用C文件

一个C语言文件test.c

#include <stdio.h>
void print(int a,int b)
    printf("这里调用的是C语言的函数:%d,%d\n",a,b);

一个头文件test.h

#ifndef _TEST_H
#define _TEST_H
void print(int a,int b);
#endif

C++文件调用C函数

#include <iostream>
using namespace std;
#include "test.h"
int main()
   cout<<"现在调用C语言函数\n";
   print(3,4);
   return 0;

执行命令

gcc -c test.c
g++ -o main main.cpp test.o

编译后链接出错:main.cpp对print(int, int)未定义的引用。

那么g++编译器为什么找不到print(int,int)呢,其实在我们学C++重载的时候就提到过C++底层的编译原理。

原因分析

test.c我们使用的是C语言的编译器gcc进行编译的,其中的函数print编译之后,在符号表中的名字为 print,通过nm查看.o文件.

$ gcc -c test.c
$ nm test.o 
                 U _GLOBAL_OFFSET_TABLE_
0000000000000000 T print
                 U printf

我们链接的时候采用的是 g++ 进行链接,也就是 C++ 链接方式,程序在运行到调用 print 函数的代码时,会在符号表中寻找 _Z5printii (是按照C++的链接方法来寻找的,所以是找 _Z5printii 而不是找 print )的名字,发现找不到,所以会提示“未定义的引用”

$ g++ -c test.c
main.cpp  makefile  test.c  test.h  test.o
$ nm test.o
                 U _GLOBAL_OFFSET_TABLE_
                 U printf
0000000000000000 T _Z5printii

此时如果我们在对print的声明中加入 extern “C” ,这个时候,g++编译器就会按照C语言的链接方式进行寻找,也就是在符号表中寻找print(这才是C++兼容C),这个时候是可以找到的,是不会报错的。

总结

编译后底层解析的符号不同,C语言是 _print ,C++是 __Z5printii

解决调用失败问题

修改test.h文件

#ifndef _TEST_H
#define _TEST_H
extern "C"{
void print(int a,int b);
#endif

修改后再次执行命令

gcc -c test.c
g++ -o main main.cpp test.o
./main

运行无报错

思考:那C语言能够调用C接口吗

实验:定义main.c函数如下

#include <stdio.h>
#include "test.h"
int main()
    printf("现在调用C语言函数\n");
    print(3,4);
    return 0;

重新执行命令如下

gcc -c test.c
gcc -o mian main.c test.o

报错:C语言里面没有extern “C“这种写法

C接口既能被C++调用又能被C调用

为了使得test.c代码既能被C++调用又能被C调用

将test.h修改如下

#ifndef __TEST_H__
#define __TEST_H__
#ifdef __cplusplus
#if __cplusplus
extern "C"{
#endif
#endif /* __cplusplus */
extern void print(int a,int b);
#ifdef __cplusplus
#if __cplusplus
#endif
 
推荐文章
逆袭的红酒  ·  倚天屠龙:Github Copilot vs Cursor_cursor copilot对比-CSDN博客
1 年前
性感的夕阳  ·  享年91岁!图灵奖得主、软件工程圣经《人月神话》作者Fred Brooks逝世 - 知乎
2 年前
没读研的小蝌蚪  ·  Spark Sql实战--合并数据_spark sql 合并列_dqz_nihao的博客-CSDN博客
2 年前
温文尔雅的皮蛋  ·  内核该怎么学?Linux内核进程管理工作原理(代码演示) - 知乎
2 年前
率性的大熊猫  ·  音视频开发技术(19)FFmpeg命令过滤器(裁剪、水印、画中画、多宫格) - 知乎
2 年前
今天看啥   ·   Py中国   ·   codingpro   ·   小百科   ·   link之家   ·   卧龙AI搜索
删除内容请联系邮箱 2879853325@qq.com
Code - 代码工具平台
© 2024 ~ 沪ICP备11025650号