extern "C"

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

原文

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

    1062 引用 • 3455 回帖 • 139 关注
  • C++

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

    108 引用 • 153 回帖 • 2 关注
  • 技术

    到底什么才是技术呢?

    88 引用 • 179 回帖 • 4 关注

相关帖子

回帖

欢迎来到这里!

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

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

推荐标签 标签

  • 程序员

    程序员是从事程序开发、程序维护的专业人员。

    594 引用 • 3541 回帖
  • FreeMarker

    FreeMarker 是一款好用且功能强大的 Java 模版引擎。

    23 引用 • 20 回帖 • 473 关注
  • Notion

    Notion - The all-in-one workspace for your notes, tasks, wikis, and databases.

    10 引用 • 79 回帖
  • Kubernetes

    Kubernetes 是 Google 开源的一个容器编排引擎,它支持自动化部署、大规模可伸缩、应用容器化管理。

    119 引用 • 54 回帖 • 11 关注
  • H2

    H2 是一个开源的嵌入式数据库引擎,采用 Java 语言编写,不受平台的限制,同时 H2 提供了一个十分方便的 web 控制台用于操作和管理数据库内容。H2 还提供兼容模式,可以兼容一些主流的数据库,因此采用 H2 作为开发期的数据库非常方便。

    11 引用 • 54 回帖 • 678 关注
  • CSDN

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

    14 引用 • 155 回帖
  • 架构

    我们平时所说的“架构”主要是指软件架构,这是有关软件整体结构与组件的抽象描述,用于指导软件系统各个方面的设计。另外还有“业务架构”、“网络架构”、“硬件架构”等细分领域。

    142 引用 • 442 回帖
  • Chrome

    Chrome 又称 Google 浏览器,是一个由谷歌公司开发的网页浏览器。该浏览器是基于其他开源软件所编写,包括 WebKit,目标是提升稳定性、速度和安全性,并创造出简单且有效率的使用者界面。

    63 引用 • 289 回帖
  • BND

    BND(Baidu Netdisk Downloader)是一款图形界面的百度网盘不限速下载器,支持 Windows、Linux 和 Mac,详细介绍请看这里

    107 引用 • 1281 回帖 • 44 关注
  • abitmean

    有点意思就行了

    38 关注
  • 知乎

    知乎是网络问答社区,连接各行各业的用户。用户分享着彼此的知识、经验和见解,为中文互联网源源不断地提供多种多样的信息。

    10 引用 • 66 回帖 • 1 关注
  • 阿里巴巴

    阿里巴巴网络技术有限公司(简称:阿里巴巴集团)是以曾担任英语教师的马云为首的 18 人,于 1999 年在中国杭州创立,他们相信互联网能够创造公平的竞争环境,让小企业通过创新与科技扩展业务,并在参与国内或全球市场竞争时处于更有利的位置。

    43 引用 • 221 回帖 • 29 关注
  • Pipe

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

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

    134 引用 • 1128 回帖 • 112 关注
  • SEO

    发布对别人有帮助的原创内容是最好的 SEO 方式。

    36 引用 • 200 回帖 • 46 关注
  • 宕机

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

    13 引用 • 82 回帖 • 81 关注
  • ReactiveX

    ReactiveX 是一个专注于异步编程与控制可观察数据(或者事件)流的 API。它组合了观察者模式,迭代器模式和函数式编程的优秀思想。

    1 引用 • 2 回帖 • 190 关注
  • Kotlin

    Kotlin 是一种在 Java 虚拟机上运行的静态类型编程语言,由 JetBrains 设计开发并开源。Kotlin 可以编译成 Java 字节码,也可以编译成 JavaScript,方便在没有 JVM 的设备上运行。在 Google I/O 2017 中,Google 宣布 Kotlin 成为 Android 官方开发语言。

    19 引用 • 33 回帖 • 91 关注
  • AWS
    11 引用 • 28 回帖 • 4 关注
  • 禅道

    禅道是一款国产的开源项目管理软件,她的核心管理思想基于敏捷方法 scrum,内置了产品管理和项目管理,同时又根据国内研发现状补充了测试管理、计划管理、发布管理、文档管理、事务管理等功能,在一个软件中就可以将软件研发中的需求、任务、bug、用例、计划、发布等要素有序的跟踪管理起来,完整地覆盖了项目管理的核心流程。

    10 引用 • 15 回帖 • 2 关注
  • SSL

    SSL(Secure Sockets Layer 安全套接层),及其继任者传输层安全(Transport Layer Security,TLS)是为网络通信提供安全及数据完整性的一种安全协议。TLS 与 SSL 在传输层对网络连接进行加密。

    70 引用 • 193 回帖 • 417 关注
  • 微服务

    微服务架构是一种架构模式,它提倡将单一应用划分成一组小的服务。服务之间互相协调,互相配合,为用户提供最终价值。每个服务运行在独立的进程中。服务于服务之间才用轻量级的通信机制互相沟通。每个服务都围绕着具体业务构建,能够被独立的部署。

    97 引用 • 155 回帖 • 1 关注
  • Mobi.css

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

    1 引用 • 6 回帖 • 781 关注
  • DNSPod

    DNSPod 建立于 2006 年 3 月份,是一款免费智能 DNS 产品。 DNSPod 可以为同时有电信、网通、教育网服务器的网站提供智能的解析,让电信用户访问电信的服务器,网通的用户访问网通的服务器,教育网的用户访问教育网的服务器,达到互联互通的效果。

    6 引用 • 26 回帖 • 536 关注
  • 心情

    心是产生任何想法的源泉,心本体会陷入到对自己本体不能理解的状态中,因为心能产生任何想法,不能分出对错,不能分出自己。

    59 引用 • 369 回帖
  • 创造

    你创造的作品可能会帮助到很多人,如果是开源项目的话就更赞了!

    189 引用 • 1021 回帖 • 1 关注
  • OpenResty

    OpenResty 是一个基于 NGINX 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态 Web 应用、Web 服务和动态网关。

    17 引用 • 46 关注
  • 服务

    提供一个服务绝不仅仅是简单的把硬件和软件累加在一起,它包括了服务的可靠性、服务的标准化、以及对服务的监控、维护、技术支持等。

    41 引用 • 24 回帖