站长网 语言 C++怎样调用写好的C接口?

C++怎样调用写好的C接口?

前言如何在C++代码中调用写好的C接口?你可能会奇怪,C++不是兼容C吗?直接调用不就可以了,那么我们来测试一下,先看看C++如何调用C代码接口的。C++调用C文件一个C语言文件test.c#include stdio.hvoid print(int a,int b){ printf(这里调用的是C语言的函数:%

前言

如何在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 

$ ls 

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 

#endif /* __cplusplus */ 

#endif /* __TEST_H__ */ 

ps:下期介绍一个Source Insight的插件,快速生成上面的代码

 

再次执行命令

 

gcc -c test.c 

gcc -o main main.c test.o 

./main 

本文来自网络,不代表站长网立场,转载请注明出处:https://www.tzzz.com.cn/html/biancheng/yuyan/2021/1104/20032.html

作者: dawei

【声明】:站长网内容转载自互联网,其相关言论仅代表作者个人观点绝非权威,不代表本站立场。如您发现内容存在版权问题,请提交相关链接至邮箱:bqsm@foxmail.com,我们将及时予以处理。
联系我们

联系我们

0577-28828765

在线咨询: QQ交谈

邮箱: xwei067@foxmail.com

工作时间:周一至周五,9:00-17:30,节假日休息

返回顶部