extern "C"

本贴最后更新于 2461 天前,其中的信息可能已经时移俗易

原文

本以为很简单,仔细阅读了一下 C++ 标准,发现内容还不少。总结了一下。

要点:函数类型,函数名,变量名具有语言链接性,language linkage。

  • 语言链接性可能会影响到名字以及调用约定等,由实现决定。

  • C++ 默认的语言连接性是 C++ 语言链接性。

  • 语言链接性仅作用于函数类型,函数名,变量名。

  • 不同语言链接性的函数类型是不同的类型,即便其余的地方都相同。

  • 语言链接性用链接性规格(linkage-specification)来声明,分有无大括号两种形式。

        extern string-literal { declaration-seq opt }
        extern string-literal declaration;
  • 所有的实现都必须支持 "C" 和 "C++" 链接性。

  • 链接性规格允许嵌套,此时最内层的那个起作用,但是并不建立作用域。因为不是作用域,B 可以看到 A

        extern "C" { extern "C++" { class A{}; } }
        extern "C" { class B:A{};}
  • 如果 C 链接性施加到 C++ 类成员和成员函数的类型,忽略 C 链接性,但是其余的地方依然有效,比如成员函数的参数。比如 extern "C" { class A{ void f(void (*p)()){} }; }

    • A 不是函数和变量,没有语言链接性
    • f 是 C++ 成员函数,忽略所指定的 C 链接性,如果在外层没指定别的,就是
    • C++ 链接性。
    • p 具有 C 链接性。
  • 除了 C++ 链接性的函数外,同一个函数不带链接性规格的声明的函数不能早于带的。如果前面声明了带链接性规格的形式,后面又出现了不带的形式,不受影响。

    • 比如 extern "C" void foo(); void foo(); 是可以的,反过来不行。
    • 但是 extern void foo(); extern "C++" void foo(); 可以。
  • 特定名字的 C 语言链接性的函数最多只能有一个,即便放在不同的 namespace 里,也是同一个函数,变量也是如此。这样的函数或变量也不能重复定义。两个 f 是同一个

        namespace A { extern "C" void f(); }
        namespace B { extern "C" void f(); }
  • 有链接性规格的函数具有外部链接性。因此下面的代码是错误的。
        extern "C" void f();
        static void f();
  • 链接性规格,带大括号的形式,不影响里面的声明是声明还是定义。单个声明的形式,视为 extern 限定符。
        extern "C" int i;       // 是声明;
        extern "C" { int i; }   // 是定义;
        extern "C" { extern int i; } // 又是声明。
  • 单个声明形式的连接性规格,不能再指定存储类别。
        extern "C" static void f(); // error
  • 因为语言链接性只跟链接性有关,因此 C 语言链接性的函数也可以有默认参数,函数的接口可以有引用。

  • 标准 C++ 库中的符号默认是 C++ 链接性。C++ 用的标准 C 库中的外部链接性符号可以是 C 或者 C++ 语言链接性,推荐后者。任何带有连续两个下划线的名字保留给实现用作同时具有 C 和 C++ 链接性的名字。

  • 标准 C 库中的任何函数签名都保留给实现用来做同时具有 C 和 C++ 链接性的名字。

  • 因为不同语言连接性的相同签名的函数类型算作不同类型,所以可以相互重载,不算同一个函数。类似的有两个原型的函数还有 bsearch,qsort 等带函数指针类型的参数的函数。

// commented by ctrlz
// 这里应该是ill-formed. 
        extern "C" int atexit(void (*f)(void))   // 这里的 f 是 C 链接性
        extern "C++" int atexit(void (*f)(void)) // 这里的 f 是 C++ 链接性
  • 以上都是 C++ 标准里的理论,落实到实践上,为了实现方便,很多编译器都把 C 和 C++ 链接性的函数的类型视为相同的类型,此时的链接性仅仅影响生成的名字,或者还可能影响异常规格,比如把 extern "C" 的函数默认视为 throw() 的等,并不影响调用约定。同样也为了实现的方便,C++ 所用的 C 库中的符号也都是 C 语言链接性的,因此这样的情况下,atexit, bsearch, qsort 等在这些编译器搭配的库中也就只有一种原型了。这种情况下,同样签名的 C 和 C++ 链接性的函数类型就算做同种类型了,下边的程序编译就会失败(VC 和 gcc 在这一点上都不符合标准):
//commented by ctrlz.
//ill-formed
void f(void (*p)())
{
}
extern "C" void f(void (*q)())
{
}
  • 链接性不应该影响函数的类型,就像普通函数的返回类型不影响函数的类型一样,从重载解析的角度看,如果接受以上的行为,势必影响重载解析的过程。
  • B3log

    B3log 是一个开源组织,名字来源于“Bulletin Board Blog”缩写,目标是将独立博客与论坛结合,形成一种新的网络社区体验,详细请看 B3log 构思。目前 B3log 已经开源了多款产品:SymSoloVditor思源笔记

    1063 引用 • 3454 回帖 • 189 关注
  • C++

    C++ 是在 C 语言的基础上开发的一种通用编程语言,应用广泛。C++ 支持多种编程范式,面向对象编程、泛型编程和过程化编程。

    107 引用 • 153 回帖
  • 技术

    到底什么才是技术呢?

    88 引用 • 179 回帖 • 4 关注

相关帖子

回帖

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • Postman

    Postman 是一款简单好用的 HTTP API 调试工具。

    4 引用 • 3 回帖 • 7 关注
  • ZooKeeper

    ZooKeeper 是一个分布式的,开放源码的分布式应用程序协调服务,是 Google 的 Chubby 一个开源的实现,是 Hadoop 和 HBase 的重要组件。它是一个为分布式应用提供一致性服务的软件,提供的功能包括:配置维护、域名服务、分布式同步、组服务等。

    59 引用 • 29 回帖 • 14 关注
  • 学习

    “梦想从学习开始,事业从实践起步” —— 习近平

    171 引用 • 512 回帖
  • JWT

    JWT(JSON Web Token)是一种用于双方之间传递信息的简洁的、安全的表述性声明规范。JWT 作为一个开放的标准(RFC 7519),定义了一种简洁的,自包含的方法用于通信双方之间以 JSON 的形式安全的传递信息。

    20 引用 • 15 回帖 • 6 关注
  • C++

    C++ 是在 C 语言的基础上开发的一种通用编程语言,应用广泛。C++ 支持多种编程范式,面向对象编程、泛型编程和过程化编程。

    107 引用 • 153 回帖
  • Love2D

    Love2D 是一个开源的, 跨平台的 2D 游戏引擎。使用纯 Lua 脚本来进行游戏开发。目前支持的平台有 Windows, Mac OS X, Linux, Android 和 iOS。

    14 引用 • 53 回帖 • 538 关注
  • 宕机

    宕机,多指一些网站、游戏、网络应用等服务器一种区别于正常运行的状态,也叫“Down 机”、“当机”或“死机”。宕机状态不仅仅是指服务器“挂掉了”、“死机了”状态,也包括服务器假死、停用、关闭等一些原因而导致出现的不能够正常运行的状态。

    13 引用 • 82 回帖 • 59 关注
  • 机器学习

    机器学习(Machine Learning)是一门多领域交叉学科,涉及概率论、统计学、逼近论、凸分析、算法复杂度理论等多门学科。专门研究计算机怎样模拟或实现人类的学习行为,以获取新的知识或技能,重新组织已有的知识结构使之不断改善自身的性能。

    83 引用 • 37 回帖
  • Swagger

    Swagger 是一款非常流行的 API 开发工具,它遵循 OpenAPI Specification(这是一种通用的、和编程语言无关的 API 描述规范)。Swagger 贯穿整个 API 生命周期,如 API 的设计、编写文档、测试和部署。

    26 引用 • 35 回帖 • 5 关注
  • 房星科技

    房星网,我们不和没有钱的程序员谈理想,我们要让程序员又有理想又有钱。我们有雄厚的房地产行业线下资源,遍布昆明全城的 100 家门店、四千地产经纪人是我们坚实的后盾。

    6 引用 • 141 回帖 • 584 关注
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    91 引用 • 384 回帖 • 2 关注
  • 自由行
    4 关注
  • Sym

    Sym 是一款用 Java 实现的现代化社区(论坛/BBS/社交网络/博客)系统平台。

    下一代的社区系统,为未来而构建

    524 引用 • 4601 回帖 • 700 关注
  • iOS

    iOS 是由苹果公司开发的移动操作系统,最早于 2007 年 1 月 9 日的 Macworld 大会上公布这个系统,最初是设计给 iPhone 使用的,后来陆续套用到 iPod touch、iPad 以及 Apple TV 等产品上。iOS 与苹果的 Mac OS X 操作系统一样,属于类 Unix 的商业操作系统。

    85 引用 • 139 回帖
  • Log4j

    Log4j 是 Apache 开源的一款使用广泛的 Java 日志组件。

    20 引用 • 18 回帖 • 29 关注
  • TensorFlow

    TensorFlow 是一个采用数据流图(data flow graphs),用于数值计算的开源软件库。节点(Nodes)在图中表示数学操作,图中的线(edges)则表示在节点间相互联系的多维数据数组,即张量(tensor)。

    20 引用 • 19 回帖 • 1 关注
  • NGINX

    NGINX 是一个高性能的 HTTP 和反向代理服务器,也是一个 IMAP/POP3/SMTP 代理服务器。 NGINX 是由 Igor Sysoev 为俄罗斯访问量第二的 Rambler.ru 站点开发的,第一个公开版本 0.1.0 发布于 2004 年 10 月 4 日。

    313 引用 • 547 回帖 • 1 关注
  • Hprose

    Hprose 是一款先进的轻量级、跨语言、跨平台、无侵入式、高性能动态远程对象调用引擎库。它不仅简单易用,而且功能强大。你无需专门学习,只需看上几眼,就能用它轻松构建分布式应用系统。

    9 引用 • 17 回帖 • 612 关注
  • SVN

    SVN 是 Subversion 的简称,是一个开放源代码的版本控制系统,相较于 RCS、CVS,它采用了分支管理系统,它的设计目标就是取代 CVS。

    29 引用 • 98 回帖 • 694 关注
  • Latke

    Latke 是一款以 JSON 为主的 Java Web 框架。

    71 引用 • 535 回帖 • 789 关注
  • WordPress

    WordPress 是一个使用 PHP 语言开发的博客平台,用户可以在支持 PHP 和 MySQL 数据库的服务器上架设自己的博客。也可以把 WordPress 当作一个内容管理系统(CMS)来使用。WordPress 是一个免费的开源项目,在 GNU 通用公共许可证(GPLv2)下授权发布。

    66 引用 • 114 回帖 • 223 关注
  • Solo

    Solo 是一款小而美的开源博客系统,专为程序员设计。Solo 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    1435 引用 • 10056 回帖 • 489 关注
  • CSDN

    CSDN (Chinese Software Developer Network) 创立于 1999 年,是中国的 IT 社区和服务平台,为中国的软件开发者和 IT 从业者提供知识传播、职业发展、软件开发等全生命周期服务,满足他们在职业发展中学习及共享知识和信息、建立职业发展社交圈、通过软件开发实现技术商业化等刚性需求。

    14 引用 • 155 回帖
  • Git

    Git 是 Linux Torvalds 为了帮助管理 Linux 内核开发而开发的一个开放源码的版本控制软件。

    209 引用 • 358 回帖 • 1 关注
  • 分享

    有什么新发现就分享给大家吧!

    248 引用 • 1795 回帖
  • 百度

    百度(Nasdaq:BIDU)是全球最大的中文搜索引擎、最大的中文网站。2000 年 1 月由李彦宏创立于北京中关村,致力于向人们提供“简单,可依赖”的信息获取方式。“百度”二字源于中国宋朝词人辛弃疾的《青玉案·元夕》词句“众里寻他千百度”,象征着百度对中文信息检索技术的执著追求。

    63 引用 • 785 回帖 • 164 关注
  • BookxNote

    BookxNote 是一款全新的电子书学习工具,助力您的学习与思考,让您的大脑更高效的记忆。

    笔记整理交给我,一心只读圣贤书。

    1 引用 • 1 回帖