注意这里的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. FreeImage裁剪示例

    //截图 int cropImage(const char* file, int left, int top, int right, int bottom, BYTE* &dstData, D ...

  2. mybatis常用jdbcType数据类型

    MyBatis 通过包含的jdbcType类型 BIT         FLOAT      CHAR           TIMESTAMP       OTHER       UNDEFINED ...

  3. ORACLE CASE函数 .

    Case具有两种格式.简单Case函数和Case搜索函数. --简单Case函数 CASE sex WHEN '1' THEN '男' WHEN '2' THEN '女' ELSE '其他' END ...

  4. make clean与make distclean的区别

    make clean仅仅是清除之前编译的可执行文件及配置文件. 而make distclean要清除所有生成的文件. Makefile 在符合GNU Makefiel惯例的Makefile中,包含了一 ...

  5. php 数组 array_values () array_key()

    <?php // array_unique($array) 去除重复 // array_unshif()向数组的顶部追加函数 // array_shif($a,"ss")向数 ...

  6. 怎么查看mysql执行过的sql。

    有些时候当程序做了更新,数据库负载突然上来,或者并发翻了几倍.这个时候如果用show full processlist; 根本看不到完全的sql.怎么才能看是哪些sql导致的呢,在网上查了资料,有一下 ...

  7. locate命令的安装

    linux中locate命令可以快速定位我们需要查找的文件,但是在yum中,locate的安装包名为mlocate(yum list | grep locate可以查看),安装方法: yum -y i ...

  8. C语言学习second--C语言基础学习

    1.标准C语言 C语言诞生于20世纪70年代,年龄比我们自己还要大,期间产生了很多标准,但是各种编译器对标准的支持不尽相同. ANSI C是使用的最广泛的一个标准,也是第一个正式标准,被称为“标准C语 ...

  9. SQL Server identity种子

    背景: 用identity修饰列可以使它自动增长 例了: create table T(ID int not null identity(1,1),      Data nvarchar(32)); ...

  10. linux上应用随机启动

    这是个go项目,其他的可以参考. 首先要有个脚本比如demo #!/bin/bash # # etcd This shell script takes care of starting and sto ...