extern "C"

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

原文

本以为很简单,仔细阅读了一下 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思源笔记

    1083 引用 • 3461 回帖 • 286 关注
  • C++

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

    106 引用 • 152 回帖 • 1 关注
  • 技术

    到底什么才是技术呢?

    88 引用 • 179 回帖 • 4 关注

相关帖子

回帖

欢迎来到这里!

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

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

推荐标签 标签

  • 音乐

    你听到信仰的声音了么?

    59 引用 • 509 回帖
  • 前端

    前端技术一般分为前端设计和前端开发,前端设计可以理解为网站的视觉设计,前端开发则是网站的前台代码实现,包括 HTML、CSS 以及 JavaScript 等。

    247 引用 • 1347 回帖
  • 星云链

    星云链是一个开源公链,业内简单的将其称为区块链上的谷歌。其实它不仅仅是区块链搜索引擎,一个公链的所有功能,它基本都有,比如你可以用它来开发部署你的去中心化的 APP,你可以在上面编写智能合约,发送交易等等。3 分钟快速接入星云链 (NAS) 测试网

    3 引用 • 16 回帖
  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖 • 2 关注
  • PWL

    组织简介

    用爱发电 (Programming With Love) 是一个以开源精神为核心的民间开源爱好者技术组织,“用爱发电”象征开源与贡献精神,加入组织,代表你将遵守组织的“个人开源爱好者”的各项条款。申请加入:用爱发电组织邀请帖
    用爱发电组织官网:https://programmingwithlove.stackoverflow.wiki/

    用爱发电组织的核心驱动力:

    • 遵守开源守则,体现开源&贡献精神:以分享为目的,拒绝非法牟利。
    • 自我保护:使用适当的 License 保护自己的原创作品。
    • 尊重他人:不以各种理由、各种漏洞进行未经允许的抄袭、散播、洩露;以礼相待,尊重所有对社区做出贡献的开发者;通过他人的分享习得知识,要留下足迹,表示感谢。
    • 热爱编程、热爱学习:加入组织,热爱编程是首当其要的。我们欢迎热爱讨论、分享、提问的朋友,也同样欢迎默默成就的朋友。
    • 倾听:正确并恳切对待、处理问题与建议,及时修复开源项目的 Bug ,及时与反馈者沟通。不抬杠、不无视、不辱骂。
    • 平视:不诋毁、轻视、嘲讽其他开发者,主动提出建议、施以帮助,以和谐为本。只要他人肯努力,你也可能会被昔日小看的人所超越,所以请保持谦虚。
    • 乐观且活跃:你的努力决定了你的高度。不要放弃,多年后回头俯瞰,才会发现自己已经成就往日所仰望的水平。积极地将项目开源,帮助他人学习、改进,自己也会获得相应的提升、成就与成就感。
    1 引用 • 487 回帖 • 9 关注
  • 黑曜石

    黑曜石是一款强大的知识库工具,支持本地 Markdown 文件编辑,支持双向链接和关系图。

    A second brain, for you, forever.

    10 引用 • 85 回帖
  • FFmpeg

    FFmpeg 是一套可以用来记录、转换数字音频、视频,并能将其转化为流的开源计算机程序。

    22 引用 • 31 回帖 • 1 关注
  • Pipe

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

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

    131 引用 • 1114 回帖 • 152 关注
  • Spark

    Spark 是 UC Berkeley AMP lab 所开源的类 Hadoop MapReduce 的通用并行框架。Spark 拥有 Hadoop MapReduce 所具有的优点;但不同于 MapReduce 的是 Job 中间输出结果可以保存在内存中,从而不再需要读写 HDFS,因此 Spark 能更好地适用于数据挖掘与机器学习等需要迭代的 MapReduce 的算法。

    74 引用 • 46 回帖 • 550 关注
  • JSON

    JSON (JavaScript Object Notation)是一种轻量级的数据交换格式。易于人类阅读和编写。同时也易于机器解析和生成。

    51 引用 • 190 回帖 • 1 关注
  • SVN

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

    29 引用 • 98 回帖 • 692 关注
  • SQLite

    SQLite 是一个进程内的库,实现了自给自足的、无服务器的、零配置的、事务性的 SQL 数据库引擎。SQLite 是全世界使用最为广泛的数据库引擎。

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

    Bug 本意是指臭虫、缺陷、损坏、犯贫、窃听器、小虫等。现在人们把在程序中一些缺陷或问题统称为 bug(漏洞)。

    77 引用 • 1741 回帖 • 1 关注
  • Netty

    Netty 是一个基于 NIO 的客户端-服务器编程框架,使用 Netty 可以让你快速、简单地开发出一个可维护、高性能的网络应用,例如实现了某种协议的客户、服务端应用。

    49 引用 • 33 回帖 • 22 关注
  • Webswing

    Webswing 是一个能将任何 Swing 应用通过纯 HTML5 运行在浏览器中的 Web 服务器,详细介绍请看 将 Java Swing 应用变成 Web 应用

    1 引用 • 15 回帖 • 635 关注
  • 生活

    生活是指人类生存过程中的各项活动的总和,范畴较广,一般指为幸福的意义而存在。生活实际上是对人生的一种诠释。生活包括人类在社会中与自己息息相关的日常活动和心理影射。

    228 引用 • 1450 回帖
  • sts
    2 引用 • 2 回帖 • 148 关注
  • FlowUs

    FlowUs.息流 个人及团队的新一代生产力工具。

    让复杂的信息管理更轻松、自由、充满创意。

    1 引用
  • 服务器

    服务器,也称伺服器,是提供计算服务的设备。由于服务器需要响应服务请求,并进行处理,因此一般来说服务器应具备承担服务并且保障服务的能力。

    124 引用 • 580 回帖
  • PostgreSQL

    PostgreSQL 是一款功能强大的企业级数据库系统,在 BSD 开源许可证下发布。

    22 引用 • 22 回帖
  • 以太坊

    以太坊(Ethereum)并不是一个机构,而是一款能够在区块链上实现智能合约、开源的底层系统。以太坊是一个平台和一种编程语言 Solidity,使开发人员能够建立和发布下一代去中心化应用。 以太坊可以用来编程、分散、担保和交易任何事物:投票、域名、金融交易所、众筹、公司管理、合同和知识产权等等。

    34 引用 • 367 回帖 • 2 关注
  • 负能量

    上帝为你关上了一扇门,然后就去睡觉了....努力不一定能成功,但不努力一定很轻松 (° ー °〃)

    85 引用 • 1201 回帖 • 451 关注
  • Mobi.css

    Mobi.css is a lightweight, flexible CSS framework that focus on mobile.

    1 引用 • 6 回帖 • 696 关注
  • SMTP

    SMTP(Simple Mail Transfer Protocol)即简单邮件传输协议,它是一组用于由源地址到目的地址传送邮件的规则,由它来控制信件的中转方式。SMTP 协议属于 TCP/IP 协议簇,它帮助每台计算机在发送或中转信件时找到下一个目的地。

    4 引用 • 18 回帖 • 587 关注
  • Ruby

    Ruby 是一种开源的面向对象程序设计的服务器端脚本语言,在 20 世纪 90 年代中期由日本的松本行弘(まつもとゆきひろ/Yukihiro Matsumoto)设计并开发。在 Ruby 社区,松本也被称为马茨(Matz)。

    7 引用 • 31 回帖 • 177 关注
  • IPFS

    IPFS(InterPlanetary File System,星际文件系统)是永久的、去中心化保存和共享文件的方法,这是一种内容可寻址、版本化、点对点超媒体的分布式协议。请浏览 IPFS 入门笔记了解更多细节。

    20 引用 • 245 回帖 • 230 关注
  • InfluxDB

    InfluxDB 是一个开源的没有外部依赖的时间序列数据库。适用于记录度量,事件及实时分析。

    2 引用 • 55 关注