`
eriol
  • 浏览: 409956 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

extern “C" 揭秘

    博客分类:
  • C++
阅读更多

 

extern是C/C++语言中表明函数和全局变量作用范围(可见性)的关键字,该关键字告诉编译器,其声明的函数和变量可以在本模块或其它模块中使用。被extern "C"限定的函数或变量是extern类型的。

 

对于如下语句:

  extern int a;

 

仅仅是一个变量的声明,其并不是在定义变量a,并未为a分配内存空间。变量a在所有模块中作为一种全局变量只能被定义一次,否则会出现连接错误。

 

通常,在模块的头文件中对本模块提供给其它模块引用的函数和全局变量以关键字extern声明。例如,如果模块B欲引用该模块A中定义的全局变量和函数时只需包含模块A的头文件即可。这样,模块B中调用模块A中的函数时,在编译阶段,模块B虽然找不到该函数,但是并不会报错;它会在连接阶段中从模块A编译生成的目标代码中找到此函数。

 

此外,与extern对应的关键字是static,被它修饰的全局变量和函数只能在本模块中使用。因此,一个函数或变量只可能被本模块使用时,其不可能被extern “C”修饰。

 

被extern "C"修饰的变量和函数是按照C语言方式编译和连接的。

 

 

未加extern “C”声明时的编译方式

 

首先看看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++中,函数void foo( int x, int y )与void foo( int x, float y )编译生成的符号是不相同的,后者为_foo_int_float。

同样地,C++中的变量除支持局部变量外,还支持类成员变量和全局变量。用户所编写程序的类成员变量可能与全局变量同名,我们以"."来区分。而本质上,编译器在进行编译时,与函数的处理相似,也为类中的变量取了一个独一无二的名字,这个名字与用户程序中同名的全局变量名字不同。

 

 

未加extern "C"声明时的连接方式

 

假设在C++中,模块A的头文件如下:

 

// 模块A头文件 moduleA.h

#ifndef MODULE_A_H

#define MODULE_A_H

int foo( int x, int y );

#endif

 

 

在模块B中引用该函数:

 

// 模块B实现文件 moduleB.cpp

#include "moduleA.h"

foo(2,3);

 

 

实际上,在连接阶段,连接器会从模块A生成的目标文件moduleA.obj中寻找_foo_int_int这样的符号!

 

 

加extern "C"声明后的编译和连接方式

 

加extern "C"声明后,模块A的头文件变为:

 

// 模块A头文件 moduleA.h

#ifndef MODULE_A_H

#define MODULE_A_H

extern "C" int foo( int x, int y );

#endif

 

 

在模块B的实现文件中仍然调用foo( 2,3 ),其结果是:

(1) 模块A编译生成foo的目标代码时,没有对其名字进行特殊处理,采用了C语言的方式;

(2) 连接器在为模块B的目标代码寻找foo(2,3)调用时,寻找的是未经修改的符号名_foo。

 

如果在模块A中函数声明了foo为extern "C"类型,而模块B中包含的是extern int foo( int x, int y ) ,则模块B找不到模块A中的函数;反之亦然。

 

所以,可以用一句话概括extern “C”这个声明的真实目的:实现C++与C及其它语言的混合编程。

 

明白了C++中extern "C"的设立动机,我们下面来具体分析extern "C"通常的使用技巧。

 

 

extern "C"的惯用法

 

(1) 在C++中引用C语言中的函数和变量,在包含C语言头文件(假设为cExample.h)时,需进行下列处理:

 

 

extern "C"
{
    #include "cExample.h"
}

 

 

而在C语言的头文件中,对其外部函数只能指定为extern类型,C语言中不支持extern "C"声明,在.c文件中包含了extern "C"时会出现编译语法错误。

 

这是一个C++引用C函数的例子,工程中包含的三个文件的源代码如下:

 

/* c语言头文件:cExample.h */
#ifndef C_EXAMPLE_H
#define C_EXAMPLE_H
extern int add(int x,int y);
#endif

/* c语言实现文件:cExample.c */
#include "cExample.h"
int add( int x, int y )
{
    return x + y;
}

// c++实现文件,调用add:cppFile.cpp
extern "C" 
{
    #include "cExample.h"
}
int main(int argc, char* argv[])
{
    add(2,3); 
    return 0;
}

 

如果C++调用一个C语言编写的.DLL时,当包括.DLL的头文件或声明接口函数时,应加extern "C" { }。

 

(2) 在C中引用C++语言中的函数和变量时,C++的头文件需添加extern "C",但是在C语言中不能直接引用声明了extern "C"的该头文件,应该仅在C文件中将C++中定义的extern "C"函数声明为extern类型。

 

//C++头文件 cppExample.h
#ifndef CPP_EXAMPLE_H
#define CPP_EXAMPLE_H
extern "C" int add( int x, int y );
#endif

//C++实现文件 cppExample.cpp
#include "cppExample.h"
int add( int x, int y )
{
    return x + y;
}

/* C实现文件 cFile.c
/* 这样会编译出错:#include "cExample.h" */
extern int add( int x, int y );
int main( int argc, char* argv[] )
{
    add( 2, 3 ); 
    return 0;
}
 
分享到:
评论

相关推荐

    extern C详细解释

    `extern "C"`用于告诉C++编译器,在处理其后的代码时,应采用C语言的链接约定。这意味着编译器不会对函数名进行修饰,而是保持其原始形式。这对于在C++代码中调用C语言编写的库或函数至关重要。 在实际应用中,`...

    关于externC的超详解

    在C++编程中,`extern "C"`是一个特殊的声明,用于告诉编译器按照C语言的规则处理特定的函数和变量。...在阅读“关于externC.txt”这个文件时,你可以期待深入学习到更多关于`extern "C"`的具体细节和实际应用案例。

    extern c 用法解析

    在C++编程中,`extern "C"`是一个特殊的声明,用于指示编译器按照C语言的规则处理函数和全局变量,而不是C++的标准方式。由于C++支持函数重载和名称修饰(name mangling),这可能导致C++编译的代码与C编译的代码在...

    extern "C"的详细用法+demo

    在C++编程中,`extern "C"`是一个特殊的声明,用于告诉编译器按照C语言的规则处理特定的函数或变量。这是因为C++支持名称修饰(Name Mangling),即编译器为了支持重载和其他特性,会改变函数和变量的原始名称。而...

    C语言中的extern关键字详细讲述

    分组成:“extern”和“C”,我们分别对这两部分进行详细的解释。 ### 1. extern关键字 `extern`是C语言中的一个关键字,主要用于声明一个变量或者函数的定义存在于当前编译单元之外的地方。当我们使用`extern`...

    C语言之extern声明辨析

    C语言之extern声明辨析 extern关键字是C语言中的一种修饰符,用于声明变量或函数的定义在其他文件中。它可以置于变量或者函数前,以标示变量或者函数的定义在别的文件中,提示编译器遇到此变量和函数时在其他模块中...

    浅析extern “C”的作用

    extern "C"是C++编程语言中的一个关键字,它用于告诉C++编译器按照C语言的方式处理接下来的函数声明。这样做的原因主要是因为C++和C语言在编译过程中对于函数名的处理方式存在差异,从而导致C++代码中不能直接调用...

    C语言extern使用方法总结.doc

    C语言extern使用方法总结 在C语言中,extern关键字是一个非常重要的概念,它可以用来声明全局变量和函数,但是很多人却不太了解它的使用方法。在本文中,我们将详细介绍extern关键字的使用方法和注意事项。 声明和...

    C语言中extern用法总结

    这是因为C++支持函数重载,导致函数名可能发生变化,而`extern "C"`可以确保函数名与C语言中的保持一致。 #### 四、总结 `extern`关键字是C语言中用于实现多个源文件间通信的重要手段之一。正确理解和使用`extern`...

    extern 'C' 详解

    B)在 C 中引用 C++ 语言中的函数和变量时,C++ 的头文件需添加 extern "C",但是在 C 语言中不能直接引用声明了 extern "C" 的头文件,应该仅将 C 文件中将 C++ 中定义的 extern "C" 函数声明为 extern 类型。...

    iOS extern使用教程

    在C/C++语言中,`extern`是用来告知编译器变量或函数是在其他地方定义的。当你在某个文件中声明了一个全局变量或函数,但在其他文件中需要使用它时,就需要用`extern`来声明这个外部变量或函数的存在。 例如,假设...

    extern “C”使用详解.doc

    总结来说,`extern "C"`是C++为兼容C语言而设计的关键字,它让C++代码可以正确链接到C库,同时也使得C库可以被C++代码调用。理解并合理使用`extern "C"`,能够帮助我们在C++中无缝地集成C代码,提高代码的复用性和...

    C++中extern “C”含义深层探索.doc

    在C++编程中,`extern "C"`是一个关键的语法特性,它的主要目的是为了实现C++与C语言的兼容性。C++作为一种更强大的、面向对象的语言,它引入了诸如类、模板、重载等高级特性,这些特性使得C++在编译后的符号表示与...

    extern在C和C++中的作用

    - **`extern "C"`的作用**:当在C++环境中使用C语言库时,需要通过`extern "C"`来避免名字修饰的发生,从而确保函数能够正确地被链接器识别。 #### 使用示例 假设有一个C语言定义的函数`void myFunc(int x);`,...

    extern “C”(让C++程序调用C函数的声明方法)

    此时,通过使用`extern "C"`可以告知C++编译器按照C语言的链接约定处理特定的函数或变量,从而避免链接错误。 #### 如何使用`extern "C"` 在C++中声明C函数时,可以使用`extern "C"`关键字包裹起来: ```cpp ...

    C语言extern关键字的用法

    ### C语言extern关键字的用法 #### 一、引言 在C语言中,`extern`关键字主要用于在多个源文件之间共享变量或函数定义。它允许程序员在一个源文件中声明一个变量或函数,并且该变量或函数的实际定义位于另一个...

    C项目文件组织规则(extern用法)

    extern在C语言项目文件组织中的用法,非常经典

    探索C++的秘密之详解extern

    总之,extern "C"是C++语言提供的一种链接指定,它允许程序员在C++代码中调用C语言编写的功能,或者在C++的库中使用C语言编写的部分,是C++语言为了兼容和链接C语言代码提供的一种机制。通过使用extern "C",可以...

    extern c的用法

    `extern "C"` 是 C++ 语言中一种特殊的声明方式,用于桥接 C 和 C++ 之间的兼容性问题。在 C++ 中,由于支持函数重载和类成员等特性,编译器会对函数和变量的名字进行修饰(也叫名称修饰或名称 mangling),产生不同...

Global site tag (gtag.js) - Google Analytics