C++基础知识3:extern "C"关键字
本文目的
简单的介绍一下extern "C"是干嘛用的,以及常见使用方式
前置与关联知识点
- extern关键字
- C和C++语言基础
- ifdef的作用
extern "C"关键字简介
1. C++和C的编译方式有所区别
C++支持函数重载,因此编译器编译函数的过程中会将函数的参数类型也加到编译后的代码中,而不仅仅是函数名。
C不支持函数重载,因此编译C语言代码的函数时不会带上函数的参数类型,一般只包括函数名。
ps: extern发生在编译时的链接阶段,"C"则是告知编译器以C的方式编译链接某段代码,可以拆开理解。
由此可见,在C++里面外部声明C文件实现的函数的时候,如果简单的extern会出现报错,因为C++和C编译出来的函数名不同,会显示找不到函数的实现。
示例代码如下,main.cpp里面调用myclib.c的函数fun_c出现报错
//myclib.c
#include"stdio.h"
void fun_c(void)
{
printf("这是一个C语言编译成的函数\n");
}
//main.cpp
#include<iostream>
extern void fun_c(void);//声明外部函数
int main(int argc, char *argv[])
{
fun_c();
return 0;
}
make(这里是cmake的build,一样的)的时候链接过程报错,显示找不到这个函数
[main] Building folder: draft
[build] Starting build
[proc] Executing command: /usr/bin/cmake --build /home/monchell/Mon_workspace/project/draft/build --config Debug --target all -- -j 10
[build] [ 25%] Linking CXX executable bin/heldlo
[build] CMakeFiles/heldlo.dir/Src/main.cpp.o:在函数‘main’中:
[build] main.cpp:(.text+0x10):对‘fun_c()’未定义的引用
[build] collect2: error: ld returned 1 exit status
2. extern "C"的作用
extern "C"用于告知编译器使用C的方式编译该部分代码,这个时候就能找到对应实现,不会再报错
//main.cpp
#include<iostream>
extern "C" void fun_c(void);//声明外部函数,且告知编译器这个外部函数以C的方式编译
int main(int argc, char *argv[])
{
fun_c();
return 0;
}
extern "C"在工程代码的使用方式
一般不会使用extern 函数的方式去在cpp文件内外部声明函数,而是之间include一个包含函数声明的头文件
步骤如下:
1. 建立一个C语言实现的函数库
//myclib.c文件
#include "myclib.h"//这里一般都要带这个同名头文件的,因为往往头文件声明了一些专用结构体或者宏定义,如果只是简单的函数接口的话可以不带
#include"stdio.h"
void fun1_c(void)
{
printf("这是一个C语言编译成的函数1\n");
fun2_c();
}
void fun2_c(void)
{
printf("这是一个C语言编译成的函数2\n");
}
void fun3_c(void)
{
printf("这是一个C语言编译成的函数3\n");
}
2. 头文件声明接口函数以及相关的结构体宏定义等
//myclib.h文件
#ifndef _MYCLIB_
#define _MYCLIB_
#ifdef __cplusplus
extern "C"{
#endif
void fun1_c(void);
void fun2_c(void);
void fun3_c(void);
#ifdef __cplusplus
}
#endif
#endif
- extern "C"写在这个地方,所有的CPP文件调用的时候都会包括这个extern “C”
- #ifdef __cplusplus的作用,使得这部分代码只对cpp文件有效,当myclib.c包含这个头文件的时候就不会编译(myclib调用extern "C"会报错)
3. CPP文件include头文件并且进行函数调用
//main.cpp文件
#include<iostream>
#include<myclib>
int main(int argc, char *argv[])
{
fun1_c();
return 0;
}