extern “C“

extern "C",经常使用的东西,但是在不同场景下含义有一些区别:

extern "C" 在 C 调用 C++ 和 C++ 调用 C 的场景下,确实有不同的含义,尽管它们的核心概念都是告诉编译器如何处理符号(即函数名)的方式。

1. C 调用 C++:

在这种情况下,C 代码想要调用 C++ 的函数。由于 C++ 使用名称修饰(name mangling)来支持函数重载,C 编译器无法识别 C++ 中修饰过的名称。为了让 C 代码能够调用 C++ 函数,我们需要告诉 C++ 编译器 按照 C 的调用约定和符号规则来导出函数,这样 C 编译器就能识别并调用它。

cpp 复制代码
// C++ 函数
extern "C" void foo() {
    // C++ 实现
}
  • extern "C" 在这里的作用是告诉 C++ 编译器按照 C 的链接规则来导出函数。

  • C 代码调用时,会按 C 的规则来查找 foo 函数,而 C++ 会确保 foo 函数没有被名称修饰,C 编译器就能正确识别它。

2. C++ 调用 C:

在这种情况下,C++ 代码想要调用 C 函数。C 编译器不会做名称修饰,因此 C++ 编译器需要告知编译器使用 C 的链接规则,防止它对符号进行修饰,使得 C++ 代码能够正确地链接到 C 函数。

cpp 复制代码
// C++ 代码
extern "C" {
    void bar();  // 声明 C 中的函数
}

void call_c() {
    bar();  // 调用 C 函数
}
  • extern "C" 在这里的作用是告诉 C++ 编译器 不做名称修饰,而是按照 C ABI(应用二进制接口)来导出符号,这样 C++ 编译器就不会修改函数名,让 C++ 代码能够正确调用 C 函数。

主要区别:

  • C 调用 C++: C 编译器无法处理 C++ 的名称修饰,使用 extern "C" 是为了告诉 C++ 编译器按照 C 的链接规范导出符号,确保 C 可以调用。

  • C++ 调用 C: C++ 编译器需要遵循 C 的链接规范,防止 C++ 对函数名做修饰,使用 extern "C" 是为了让 C++ 编译器遵循 C 的 ABI 规则来查找和链接 C 函数。

结论:

extern "C" 在这两种场景下的作用是相同的------告诉编译器按照 C 的符号规则来处理符号。在 C++ 调用 C 时,extern "C" 确保符号名称不会被修饰在 C 调用 C++ 时,extern "C" 确保 C++ 的符号名称符合 C 的规则以便 C 编译器可以识别并链接

C调用C++的例子:

(1)C++ 实现文件(player.cpp

cpp 复制代码
#include "player_c_api.h"
#include <iostream>

class Player {
public:
    void start() {
        std::cout << "Player start\n";
    }
};

extern "C" {

struct PlayerHandle {
    Player* impl;
};

PlayerHandle* player_create() {
    PlayerHandle* h = new PlayerHandle;
    h->impl = new Player();
    return h;
}

void player_start(PlayerHandle* h) {
    if (h && h->impl) {
        h->impl->start();
    }
}

void player_destroy(PlayerHandle* h) {
    if (h) {
        delete h->impl;
        delete h;
    }
}

}

(2)C 头文件(player_c_api.h

cpp 复制代码
#ifndef PLAYER_C_API_H
#define PLAYER_C_API_H

#ifdef __cplusplus
extern "C" {
#endif

typedef struct PlayerHandle PlayerHandle;

PlayerHandle* player_create(void);
void player_start(PlayerHandle* handle);
void player_destroy(PlayerHandle* handle);

#ifdef __cplusplus
}
#endif

#endif

(3)C 调用方(main.c

cpp 复制代码
#include "player_c_api.h"

int main() {
    PlayerHandle* p = player_create();
    player_start(p);
    player_destroy(p);
    return 0;
}
相关推荐
GoWjw3 小时前
在C&C++指针的惯用方法
c语言·开发语言·c++
Coding Peasant3 小时前
GD32E230 I2C从机功能深度解析与实现指南
c语言·stm32·单片机·mcu·arm
superman超哥3 小时前
仓颉语言中错误恢复策略的深度剖析与工程实践
c语言·开发语言·c++·python·仓颉
玖剹3 小时前
记忆化搜索题目(二)
c语言·c++·算法·leetcode·深度优先·剪枝·深度优先遍历
搬砖的kk3 小时前
Lycium++ - OpenHarmony PC C/C++ 增强编译框架
c语言·开发语言·c++
fantasy_arch5 小时前
pd_process.c 文件源码分析
c语言·数据库·视频编解码·av1
黎雁·泠崖9 小时前
C 语言指针进阶教程:const 修饰、野指针规避与传址调用
c语言·开发语言
历程里程碑10 小时前
C++ 17异常处理:高效捕获与精准修复
java·c语言·开发语言·jvm·c++
Dillon Dong11 小时前
从C到Simulink:用Counter模块玩转嵌入式定时器
c语言·stm32·simulink