注意这里的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. java 简单的文件上传

    一.文件上传原理: 1.文件上传的前提: a.form表单的method必须是post b.form表单的enctype必须是multipart/form-data(决定了POST请求方式,请求正文的 ...

  2. Android Studio默认产生Fragment

    package com.edaixi.fragment; import android.content.Context;import android.net.Uri;import android.os ...

  3. SemaphoreFullException when checking user role via ASP.NET membership

    将指定的计数添加到该信号量中会导致其超过最大计数 This issue was fixed by restarting ASP.NET Development Server on windows ta ...

  4. Python httplib学习

    httplib是python中http协议的客户端实现,可以使用该模块与HTTP服务器进行交互. 如示例1: import httplib url = "www.126.com"c ...

  5. QTDesigner的QVBoxLayout自动随窗口拉伸

    在MainWindow的构造函数中添加如下代码://设置Uiui.setupUi(this); //使Ui可自适应父窗口大小QVBoxLayout* mainLayout = new QVBoxLay ...

  6. QT多重继承的时候,要把QObject放在最前面,否则报错——C++认为人性本恶,默认都是私有的,这点和Delphi的世界观不一样

    在买来的控件(没有源码)的基础上,想加入QObject的一些特性,不得不多继承: class MyProgress : public CProgress, public QObject 但总是报错: ...

  7. QT:使用“状态模式”绘制界面

    QT与很多GUI库不同(如MFC),它不能随时随地地在界面上画图,只能在界面类的painterEvent中画图,如此一来,想在绘制QT界面时使用状态模式(GOF的23种设计模式之一)就有点困难了,作为 ...

  8. 【N年前的文章脑补:HttpHandler HttpModule入门篇】

    HttpHandler HttpModule入门篇 ASP.Net处理Http Request时,使用Pipeline(管道)方式,由各个HttpModule对请求进行处理,然后到达 HttpHand ...

  9. 关于k-means聚类算法的matlab实现

    在数据挖掘中聚类和分类的原理被广泛的应用. 聚类即无监督的学习. 分类即有监督的学习. 通俗一点的讲就是:聚类之前是未知样本的分类.而是根据样本本身的相似性进行划分为相似的类簇.而分类 是已知样本分类 ...

  10. 【DSA MOOC】起泡排序的原理及常数优化

    根据学堂在线TsinghuaX: 30240184X 数据结构(2015秋)这门课的内容,对bubblesort做了一些总结. 1. bubblesort(起泡排序),原理来自这样一个观察规律:若序列 ...