注意这里的c调用c++或者c++调用c的意思是.c文件中调用.cpp文件中的代码,或者相反

集成开发环境如vc++6.0或者vs都是通过文件后缀来区别当前要编译的是C代码还是C++代码,然后采用相应的编译,调用协议等

使用extern "C"主要是因为C编译器编译函数时不带参数的类型信息,只包含函数的符号名字,例如int foo(int x);编译后_foo的符号,C连接器只要找到了调用函数的符号,就认为连接成功,。

但是C++编译器为了实现函数重载,会在编译时带上函数的参数信息,如它可以把上面的函数编译成类似于_foo _int 符号

所以,C调用C++,使用extern "C"是告诉编译器依照C的方式来进行编译封装接口,当然接口函数里面的C++语法还是按C++方式编译。

如:1.普通函数

//c++

extern "C" int foo(int x);

int foo(int x){};

这里编译器会将foo函数编译成_foo符号,而不会编译成类似_foo _int

那么C可以这样调用C++函数:

int  foo(int i);

void CCC(int x){

foo(x);

}

2.如果想调用重载的C++函数,则须封装单独的接口供C使用,也就是每一个重载函数必须起一个不同的名字

如:

//C++代码

void foo(int x);

void foo(float x);

extern "C" void foo_i(int x){

foo(x);

}

extern "C" void foo_f(float x){

foo(x);

}

那么在C中可这样调用

void foo_i(int x);

void foo_f(float x);

void cc(int x,float x1){

foo_i(x);

foo_f(x1);

}

3.C中想调用C++中的成员函数(包括虚函数),则提供一个简单的包装(wrapper)

例如:

//c++ code

class C{

vitual double f(int);

};

extern "C" double call_C_f(C *p,int i){  //wrapper function

return *p->f(i);

}

然后,你就可以这样调用C::f():

//C code

double call_C_f(struct C* p,int i);  //声明

void ccc(struct C* p,int i){

double d=call_C_f(p,i);

}

问题:struct C* p从哪里来?也就是说怎么在C中定义C++对象,上面只是说了思想,真实的C中使用C++类需要把原来的类都封装一下,可以参考http://blog.csdn.net/caspiansea/article/details/9676153

在C++调用C函数,extern "C"的作用就是:让C++连接器找调用函数的符号时采用c的方式

如:

//c code

void foo(int i);

c++中这样调用C函数

//c++ code

extern "C" void foo(int x);

就是让C++连接器能通过_foo而不是用_foo _int这样的符号

时常在cpp的代码之中看到这样的代码:特别是C++中引入C的头文件,这些C头文件出现很多如下代码。

#ifdef _cplusplus extern "C"{ #endif

//一段代码

#ifdef _cplusplus} #endif

其中_cplusplus是c++编译器的保留宏定义,就是说c++编译器认为这个宏已经定义了。

所以关键是 extern "C"{};

extern "C"是告诉C++编译器括号里的代码是按照c的obj文件格式编译的,要链接的话按照C的命名规则去找

要明白为何使用extern "C",还得从cpp中对函数的重载处理说起,在C++中为了支持重载,编译生成汇编码的时候,要对函数的名字进行一些处理,加入了函数的返回类型等等东西,而在C中,只是简单的函数名字而已,没有加入任何的其他信息,也就是说C++和C对产生的函数的名字处理不一样

现在我们知道了c和c++对函数名字编译时采取了不同的处理,但是为什么要使用extern "C"呢,原因是这样的,C++之父再设计C++的时候,已经有了大量C的代码,为了支持原来的C代码和已经写好的C库,需要在C++中尽可能地支持C,extern "C"就是其中的一个策略。

试想这样的情况,一个库文件已经用C写好了而且运行得很好,这个时候我们需要使用这个库文件,但是我们需要用C++文件来重写这个代码,如果这个代码使用的是c++的方式连接这个C文件的话,那么就会出现链接错误。

现在我们有了一个C库文件,它的头文件是f.h,产生的lib文件是f.lib,那么我们如果要在C++中使用这个库文件,我们需要这样写:

extern "C" {

#include "f.h"

}

c代码中调用c++,c++代码中调用c代码的更多相关文章

  1. C# 5.0中使用CallerMemberName、CallerFilePath和CallerLineNumber获取代码的调用方信息(转载)

    很多时候,我们需要在运行过程中记录一些调测的日志信息,如下所示: public void DoProcessing() { TraceMessage("DoProcessing()被XXX调 ...

  2. 【优雅代码】深入浅出 妙用Javascript中apply、call、bind

    这篇文章实在是很难下笔,因为网上相关文章不胜枚举. 巧合的是前些天看到阮老师的一篇文章的一句话: “对我来说,博客首先是一种知识管理工具,其次才是传播工具.我的技术文章,主要用来整理我还不懂的知识.我 ...

  3. 【TypeScript】如何在TypeScript中使用async/await,让你的代码更像C#。

    [TypeScript]如何在TypeScript中使用async/await,让你的代码更像C#. async/await 提到这个东西,大家应该都很熟悉.最出名的可能就是C#中的,但也有其它语言也 ...

  4. 将PL/SQL代码封装在机灵的包中

    将代码封装在机灵的包中 http://www.oracle.com/technetwork/issue-archive/2013/13-jan/o13plsql-1872456.html 绝大多数基于 ...

  5. PC逆向之代码还原技术,第五讲汇编中乘法的代码还原

    目录 PC逆向之代码还原技术,第五讲汇编中乘法的代码还原 一丶简介乘法指令 1.乘法指令 2.代码还原注意问题 二丶乘法的汇编代码产生的格式 1.高级代码观看 2.乘法的汇编代码还原. 三丶乘法总结 ...

  6. 在vue中使用import()来代替require.ensure()实现代码打包分离

    最近看到一种router的写法 import Vue from 'vue' import Router from 'vue-router' Vue.use(Router) const login = ...

  7. java中的静态变量、静态方法与静态代码块详解与初始化顺序

      我们知道类的生命周期分为装载.连接.初始化.使用和卸载的五个过程.其中静态代码在类的初始化阶段被初始化. 而非静态代码则在类的使用阶段(也就是实例化一个类的时候)才会被初始化. 静态变量 可以将静 ...

  8. java中静态变量,静态代码块,静态方法,实例变量,匿名代码块等的加载顺序

    转自:http://blog.csdn.net/mrzhoug/article/details/51581994 一.在Java中,使用”{}”括起来的代码称为代码块,代码块可以分为以下四种: 1.普 ...

  9. C++派生类中如何初始化基类对象(五段代码)

    今天收到盛大的面试,问我一个问题,关于派生类中如何初始化基类对象,我在想派生类对于构造函数不都是先构造基类对象,然后在构造子类对象,但是如果我们在成员初始化列表先初始化派生类的私有成员,在函数内去调用 ...

  10. Python中的进程池与线程池(包含代码)

    Python中的进程池与线程池 引入进程池与线程池 使用ProcessPoolExecutor进程池,使用ThreadPoolExecutor 使用shutdown 使用submit同步调用 使用su ...

随机推荐

  1. discuz二次开发笔记(三)------discuz的安装步骤

    下载好discuz的安装包后,解压在自己定义的文件夹里面,将upload里面的文件拷贝出来放到和upload同级的地方,然后删除upload文件夹. 打开浏览器,输入你文件夹的地址:如:http:// ...

  2. ASP.net与SQLite数据库通过js和ashx交互(连接和操作)

    ASP.net与SQLite数据库通过js和ashx交互(连接和操作): 废话(也是思路):用的是VS2010,打算做网站前后台.由于不喜欢前台语言里加些与html和css和js的其他内容,想实现前后 ...

  3. PHP实现简单爬虫

    <?php /**  * 爬虫程序 -- 原型  *  * 从给定的url获取html内容  *  * @param string $url  * @return string  */ func ...

  4. 为TL-WR720N编译带mentohust和njit-client的openwrt固件

    openwrt的trunk版已经支持720N了.简单好多. 首先下载openwrt源码,我下的是trunk版 svn co svn://svn.openwrt.org/openwrt/trunk/ 然 ...

  5. xcode 5与ios 7的屏幕适配问题

    #define DEVICE_IS_IPHONE5 ([[UIScreen mainScreen] bounds].size.height == 568) float height = DEVICE_ ...

  6. Asp.Net MVC3 简单入门第一季(三)详解Controller之Filter

    前言 前面两篇写的比较简单,刚开始写这个系列的时候我面向的对象是刚开始接触Asp.Net MVC的朋友,所以写的尽量简单.所以写的没多少技术含量.把这些技术总结出来,然后一简单的方式让更多的人很好的接 ...

  7. iOS GCD详解

    前言 对初学者来说,GCD似乎是一道迈不过去的坎,很多人在同步.异步.串行.并行和死锁这几个名词的漩涡中渐渐放弃治疗.本文将使用图文表并茂的方式给大家形象地解释其中的原理和规律. 线程.任务和队列的概 ...

  8. leetcode_question_102 Binary Tree Level Order Traversal

    Given a binary tree, return the level order traversal of its nodes' values. (ie, from left to right, ...

  9. DevExpress Util HelpV3

    using System; using System.Drawing; using DevExpress.XtraCharts; namespace DevExpressUtilHelpV3 { pu ...

  10. 对js中prototype的理解

    一直不理解child.prototype = new Parent()和child.prototype =Parent.prototype的区别,到现在为止,我觉得它俩最大的区别就是:前者共享构造器里 ...