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

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

[[428009]]

本文转载自微信公众号「编程学习基地」,作者deroy 。转载本文请联系编程学习基地公众号。

前言

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

C++调用C文件

一个C语言文件test.c

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

一个头文件test.h

  1. #ifndef _TEST_H 
  2. #define _TEST_H 
  3.  
  4. void print(int a,int b); 
  5.  
  6. #endif 

C++文件调用C函数

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

执行命令

  1. gcc -c test.c 
  2. 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文件.

  1. $ gcc -c test.c 
  2. $ nm test.o  
  3.                  U _GLOBAL_OFFSET_TABLE_ 
  4. 0000000000000000 T print 
  5.                  U printf 

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

  1. $ g++ -c test.c 
  2. $ ls 
  3. main.cpp  makefile  test.c  test.h  test.o 
  4. $ nm test.o 
  5.                  U _GLOBAL_OFFSET_TABLE_ 
  6.                  U printf 
  7. 0000000000000000 T _Z5printii 

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

总结

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

解决调用失败问题

修改test.h文件

  1. #ifndef _TEST_H 
  2. #define _TEST_H 
  3. extern "C"
  4. void print(int a,int b); 
  5. #endif 

修改后再次执行命令

  1. gcc -c test.c 
  2. g++ -o main main.cpp test.o 
  3. ./main 

运行无报错

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

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

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

重新执行命令如下

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

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

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

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

将test.h修改如下

  1. #ifndef __TEST_H__ 
  2. #define __TEST_H__ 
  3.  
  4. #ifdef __cplusplus 
  5. #if __cplusplus 
  6. extern "C"
  7. #endif 
  8. #endif /* __cplusplus */ 
  9.  
  10. extern void print(int a,int b); 
  11.  
  12. #ifdef __cplusplus 
  13. #if __cplusplus 
  14. #endif 
  15. #endif /* __cplusplus */ 
  16. #endif /* __TEST_H__ */ 

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

再次执行命令

  1. gcc -c test.c 
  2. gcc -o main main.c test.o 
  3. ./main 

结果示意:

 

责任编辑:武晓燕 来源: 编程学习基地
相关推荐

2020-07-31 18:33:56

C++编程语言

2019-08-28 14:21:39

C++C接口代码

2010-01-28 13:35:41

调用C++函数

2019-06-10 19:00:23

Cmain函数编程语言

2010-01-26 15:51:06

C++变量

2023-11-09 23:31:02

C++函数调用

2010-01-20 14:35:55

C++调用

2014-01-02 10:46:35

PostgreSQLC++

2011-04-08 09:52:44

C++C#DLL

2014-05-15 16:33:05

C++CLI调用C#

2010-01-26 14:10:22

Visual C++

2010-01-20 09:54:27

C++数据类型

2010-01-28 10:33:10

C++开发程序

2010-01-21 11:23:58

C++函数调用

2010-01-14 17:13:53

C++接口

2010-01-21 14:07:14

CC++声明

2010-01-21 09:34:57

C++语法

2010-01-27 16:05:06

C++堆栈

2023-03-15 15:58:11

Python动态库C++

2010-01-28 13:45:06

C++数组
点赞
收藏

51CTO技术栈公众号