分析以下一段代码:
/*=======sum.h=========*/ #ifndef SUM_H #define SUM_H #include <stdio.h> int sum(int a,int b); #endif; /*=======sum.c=========*/ #include "sum.h" int sum(int a,int b) { int c=a+b; return c; } /*====main.cpp======*/ #include "sum.h" void mian(){ cout << sum(1,2)<<endl; }
调用以上三个文件,编译通过,但是执行是出现以下问题:
obj : error LNK2001: 无法解析的外部符号 "int __cdecl sum(int,int)" (?sum@@YAHHH@Z)
E:\Programming\Grapic\test\Debug\test.exe : fatal error LNK1120: 1 个无法解析的外部命令
问题出在哪里呢? 在main.cpp里调用了sum.c,也就是说在C++程序里调用了C程序,此时如果没有作相应处理将会出现链接错误。
extern "C"表示编译生成的内部符号名使用C约定。C++支持函数重载,而C不支持,两者的编译规则也不一样。函数被C++编译后在符号库中的名字与C语言的不同。例如,假设某个函数的原型为:void foo( int x, int y ); 该函数被C编译器编译后在符号库中的名字可能为_foo,而C++编译器则会产生像_foo_int_int之类的名字(不同的编译器可能生成的名字不同,但是都采用了相同的机制,生成的新名字称为“mangled name”)。_foo_int_int这样的名字包含了函数名、函数参数数量及类型信息,C++就是靠这种机制来实现函数重载的。
那么如果在C中调用C++代码,以及如何在C++中调用C的代码呢?
extern "C"表示编译生成的内部符号名使用C约定。
1. 如何在C++中调用C呢?
C++调用C,extern "C" 的作用是:让C++连接器找调用函数的符号时采用C的方式
本文开头提出的笔试题可以这样修改:
/*=======sum.h=========*/ #ifndef SUM_H #define SUM_H #include <stdio.h> int sum(int a,int b); #endif; /*=======sum.c=========*/ #include "sum.h" int sum(int a,int b) { int c=a+b; return c; } /*====main.cpp======*/ extern "C" { #include "sum.h" } void mian(){ cout << sum(1,2)<<endl; }
执行成功
相信到这里差不多明白了
2. 怎样在C里调用C++呢?
在C中引用C++函数(C调用C++,使用extern "C"则是告诉编译器把cpp文件中extern "C"定义的函数依照C的方式来编译封装接口,当然接口函数里面的C++语法还是按C++方式编译)
执行:test1.obj : error LNK2019: 无法解析的外部符号 _sum,该符号在函数 _main 中被引用
E:\Programming\Grapic\test\Debug\test.exe : fatal error LNK1120: 1 个无法解析的外部命令
/*=======sum.h=========*/ #ifndef SUM_H #define SUM_H #include <stdio.h> int sum(int a,int b); #endif; /*=======sum.cpp=========*/ #include "sum.h" extern "C" { int sum(int a,int b) { int c=a+b; return c; } } /*====main.c======*/ #include "sum.h" void mian(){ cout << sum(1,2)<<endl; }
3. 标准规范写法
一般我们都将函数声明放在头文件,当我们的函数有可能被C或C++使用时,我们无法确定被谁调用,使得不能确定是否要将函数声明在extern "C"里,所以,我们可以添加
#ifdef __cplusplus extern "C" { #endif //函数声明 #ifdef __cplusplus } #endif
利用以上声明形式就可以综合运用了。
在C中引用C++语言中的函数和变量时,C++的函数或变量要声明在extern "C"{}里,但是在C语言中不能使用extern "C",否则编译出错。(出现错误: error C2059: syntax error : 'string',这个错误在网上找了很久,国内网站没有搜到直接说明原因的,原因是extern "C"是C++中的关键词,不是C的,所有会出错。
/*=======sum.h=========*/ #ifndef SUM_H #define SUM_H #include <stdio.h> int sum(int a,int b); #endif; /*=======sum.cpp=========*/ #include "sum.h" int sum(int a,int b) { int c=a+b; return c; } /*====main.c======*/ #include "sum.h" void mian(){ cout << sum(1,2)<<endl; }
FAQs in section [32]: [32.1] 混合C和C++编程时我需要知道什么? [32.2] 如何在C++代码中包含标准的C头文件? [32.3] 如何在C++代码中包含非系统的C头文件? [32.4] 如何修改我自己的C头文件 , 以便更容易的在C++代码中包含他们? [32.5] 如何从C++代码中调用非系统C 函数f(int,char和float)? from my C++
本文向大家介绍python和C语言混合编程实例,包括了python和C语言混合编程实例的使用技巧和注意事项,需要的朋友参考一下 最近为了测试网速情况怎么样,由于部分业务服务器需要关闭icmp,这样的话采用普通的ping就无法适应我的需求,于是自己简单的写了一个基于tcp端口的ping的程序,由于c执行效率比较的不错,但是开发效率低下,而python是开发效率高,但是执行效率不如C,由于需要大规模的
本文向大家介绍.NET Core3.1编写混合C++程序,包括了.NET Core3.1编写混合C++程序的使用技巧和注意事项,需要的朋友参考一下 前言 随着 .NET Core 3.1 的第二个预览版本发布,微软正式将 C++/CLI 移植到 .NET Core 上,从此可以使用 C++ 编写 .NET Core 的程序了。 由于目前仅有 MSVC 支持编译此类混合代码,并且由于涉及到非托管代码
本文向大家介绍Swift、Objective-C、Cocoa混合编程设置指南,包括了Swift、Objective-C、Cocoa混合编程设置指南的使用技巧和注意事项,需要的朋友参考一下 Swift 被设计用来无缝兼容 Cocoa 和 Objective-C 。在 Swift 中,你可以使用 Objective-C 的 API(包括系统框架和你自定义的代码),你也可以在 Objective-C中
C++编程规范 C/C++编程规范 C/C++编程规范--北京软通动力信息技术有限公司 高质量C/C++编程规范指南 C语言编程规范--华为技术有限公司 Google C++ 编程规范 C++编程规范:101条规则、准则与最佳实践 PDF扫描版.pdf C++ 编程规范专栏
亲爱的读者,这些C Programming Interview Questions专门设计用于让您熟悉在C Programming主题面试中可能遇到的问题的本质。 根据我的经验,很好的面试官在你的面试中几乎不打算问任何特定的问题,通常问题从这个主题的一些基本概念开始,然后他们继续基于进一步的讨论和你回答的问题 - 指针上的指针是什么? 它是一个指针变量,可以保存另一个指针变量的地址。 它取消引用两