目录
extern "C"是什么?
前面一章节简单的介绍了extern "C"的基本概念就是有时候在C++工程中可能需要将某些函数按照C的风格来编译,在函数前加extern "C",意思是告诉编译器,将该函数按照C语言规则来编译。
这一章将进行详细介绍其用法,为了更好地了解,下面将举个OJ题实例:
使用场景:
题目: 有效括号
解这道题需要我们自己写个栈,但是这里为了讲解extern "C"的用法,将把栈一系列操作写好编译到静态库中然后进行调用:
bool isValid(char* s) {
ST st;
StackInit(&st);
int i = 0;
while (s[i])
{
//左括号就插入
if (s[i] == '(')
{
StackPush(&st, ')');
}
else if (s[i] == '{')
{
StackPush(&st, '}');
}
else if (s[i] == '[')
{
StackPush(&st, ']');
}
else
{
//if (st.empty())//栈里为空,且现在为右括号就直接返回false
// return false;
if (!StackEmpty(&st) && StackTop(&st) == s[i])
{
StackPop(&st);
}
else
{
StackDestroy(&st);
return false;
}
}
i++;
}
if (StackEmpty(&st))
{
StackDestroy(&st);
return true;
}
else
{
StackDestroy(&st);
return false;
}
}
int main()
{
char str1[] = "{[]}";
char str2[] = "([)]";
printf("%d\n", isValid(str1));
printf("%d\n", isValid(str2));
return 0;
}
静态库的创建
静态库,目前我还不太清楚,后面会在linux中跟大家一起探讨,现在咱们就理解成别人写好给你用的,被编译好的程序.
先创建一个空项目,把之前写的栈放进去,然后打开项目属性,把.exe属性改为静态库(lib)属性即可:
编译后运行,静态库就完成了。
C程序调用C写的静态库,显然没有任何阻碍可以正常操作,同理C++程序调用C++写的静态库也不会有问题,这里就不带大家一起操作了。下面我将讲解C++调用C写的静态库 和C调用C++写的静态库的操作。
C++程序调用C写的静态库
调用库之前我们要做如下操作:
第一步:
第二步:
然后下面就可以操作起来了.
C++调用C写的静态库
首先需要包含头文件(创建栈静态库的路径)
#include "../stack/stack/stack.h"
.表示当前目录
..表示上一层目录
然后我们来运行一下:
这里出现了链接错误,通过前一章函数重载的学习我们很快会发现问题的所在:C++修饰函数名的方法与C修饰函数名方法不一样,导致在静态库找不到这些函数,因此我们引出了:extern "C"
这段代码的意思是:告诉编译器在调用这个静态库时,调用函数使用C的风格去调用(即调用函数时不使用函数名修饰规则)
这样我们就可以正常运行了:
C程序调用C++写的静态库
首先我们把先前C++的项目后缀改成 .c,把静态库的c文件改成cpp,我们试着反着来调用,会发生什么。
很显然C程序无法调用C++静态库中函数名被修饰的函数
所以我们需要在C++静态库中使用extern "C",因为c程序中无法识别extern "c"
我们这里介绍两种方法来修改:
第一种:
#ifdef __cplusplus
extern "C"
{
#endif
//初始化栈
void StackInit(Stack* ps);
//销毁栈
void StackDestroy(Stack* ps);
//压栈
void StackPush(Stack* ps, STDatatype x);
//出栈
void StackPop(Stack* ps);
//取出栈顶元素
STDatatype StackTop(Stack* ps);
//栈的大小
int StackSize(Stack* ps);
//判断栈是否为空
bool StackEmpty(Stack* ps);
#ifdef __cplusplus
}
#endif
第二种:
#ifdef __cplusplus
#define EXTERN_C extern "C"
#else
#define EXTERN_C
#endif
//初始化栈
EXTERN_C void StackInit(Stack* ps);
//销毁栈
EXTERN_C void StackDestroy(Stack* ps);
//压栈
EXTERN_C void StackPush(Stack* ps, STDatatype x);
//出栈
EXTERN_C void StackPop(Stack* ps);
//取出栈顶元素
EXTERN_C STDatatype StackTop(Stack* ps);
//栈的大小
EXTERN_C int StackSize(Stack* ps);
//判断栈是否为空
EXTERN_C bool StackEmpty(Stack* ps);
现在就可以正常运行了:
以上是extern "C" 的详细操作,有不足的地方或者对代码有更好的见解,欢迎评论区留言共同商讨,共同进步!!