深入解析C++中的extern关键字:跨文件共享变量与函数的终极指南
🚀 C++ extern 关键字深度解析:跨文件编程的终极指南
📅 更新时间:2025年6月5日
🏷️ 标签:C++ | extern关键字 | 多文件编程 | 链接与声明 | 现代C++
文章目录
- 前言
- 🔥一、extern 是什么?
- 🎯 关键词解释:
- 👇声明 VS 定义
- 声明
- 定义
- 总结
- 🚨 二、如何利用extern解决重定义问题
- 🚨产生原因
- 🧠 原理解析:
- 💡解决办法
- 📁三、extern 跨文件共享变量
- ❌四、extern 常见错误分析
- ❶ 多重定义错误(multiple definition)
- ❷ undefined reference 错误
- 总结
前言
提示:这里可以添加本文要记录的大概内容:
在 C++ 的世界里,extern 是一个看似简单却极其重要的关键字。它在跨文件变量访问、函数声明、内外部链接管理中扮演了关键角色。初学者容易忽视它的重要性,甚至误用它导致编译错误、链接错误如 undefined reference。
本文将从以下几个方面,带你深入理解 extern 的本质与用法:
提示:以下是本篇文章正文内容,下面案例可供参考
🔥一、extern 是什么?
简单地说,extern 用于声明一个变量或函数在其他文件中定义。它告诉编译器:“这个标识符不是在当前文件中定义的,去别的地方找吧”
🎯 关键词解释:
extern int count; // 声明一个整型变量 count 在其他地方定义
它并不分配内存,而只是“声明”变量存在于别处,由链接器去解析实际定义
👇声明 VS 定义
在深入extern之前,必须明确 声明(Declaration)和定义(Definition) 的区别:
声明
声明(Declaration):告诉编译器某个变量/函数的存在,但不分配内存
extern int global_var; // 声明,不分配内存
void foo(); // 函数声明
函数声明前面默认是有extern的 所以可以省略
定义
定义(Definition):为变量/函数分配存储空间,并给出具体实现
int global_var = 42; // 定义,分配内存
void foo() { ... } // 函数定义
extern仅用于声明,不能用于定义!
总结
注意:extern int a = 10; 实际上也是定义,容易误导新手!
所以基本上不会这样写
🚨 二、如何利用extern解决重定义问题
🚨产生原因
我们说过头文件只能用来做文件的声明,源文件用来做文件的实现
如果我们在头文件中定义变量会怎么样呢?
假设我们在global.h中定义了两个变量
//global.h
int global_age = 10;//这是定义 不是声明
std::string global_name = "llfc";//这是定义 不是声明
然后我们创建global.cpp包含global.h
//global.cpp
#include "global.h"
然后我们在main.cpp中包含global.h
//main.cpp
#include <iostream>
#include "global.h"
int main() {std::cout << "Hello, World!" << std::endl;std::cout << "globbal name is" << global_name << std::endl;std::cout << "global age is " << global_age << std::endl;return 0;
}
我们运行上述代码, 程序编译并未产生问题,但是链接产生问题了
[1/3] Building CXX object CMakeFiles/day05_extern.dir/global.cpp.obj
[2/3] Building CXX object CMakeFiles/day05_extern.dir/main.cpp.obj
[3/3] Linking CXX executable day05_extern.exe
FAILED: day05_extern.exe
multiple definition of `global_age'; CMakeFiles/day05_extern.dir/main.cpp.obj: /global.h:8: first defined here
multiple definition of `global_name[abi:cxx11]'; CMakeFiles/day05_extern.dir/main.cpp.obj:/global.h:9: first defined here
上述报错的意思是在链接main.cpp.obj时发现global_age重定义,第一次定义在global.h这里
同样链接mian.cpp.obj时发现global_name重定义,第一次定义在global.h
因为global.h中定义了global_age,根据我们之前学习的预编译知识,只要是包含global.h的文件都会展开global.h,main.cpp展开了一次,global.cpp展开了一次
🧠 原理解析:
🚨 问题的本质在于:变量定义只能出现一次
头文件中定义变量(错误示例)
// global.h
int global_age = 20; // ❌ 定义:分配了存储空间
// main.cpp
#include "global.h"
// another.cpp
#include "global.h"
🔴 错误原因:
global.h 被两个 .cpp 文件包含。
每个 .cpp 文件都复制了一份 int global_age = 20;。
链接阶段报错:变量多重定义(LNK2005 / LNK1169)
💡解决办法
可以采用extern关键字声明变量,然后将变量的定义放在global.cpp中, 这样main.cpp包含global.h就只会展开声明。声明可以重复声明,不会有问题
在global.h中用extern声明两个变量
//global.h
extern int global_age ;
extern std::string global_name ;
在global.cpp中定义这些变量
//global.cpp
#include "global.h"
#include <string>
// 定义全局变量
std::string global_name = "John Doe";
int global_age = 30;
在main.cpp中包含global.h
//main.cpp
#include <iostream>
#include "global.h"
int main() {std::cout << "Hello, World!" << std::endl;std::cout << "globbal name is" << global_name << std::endl;std::cout << "global age is " << global_age << std::endl;return 0;
}
再次编译,运行成功
Hello, World!
globbal name isJohn Doe
global age is 30
📁三、extern 跨文件共享变量
假设我们有两个 .cpp 文件:main.cpp 和 utils.cpp,我们希望在 utils.cpp 中定义一个全局变量 count,并在 main.cpp 中使用它
📁 utils.cpp
#include <iostream>
int count = 10; // 实际定义
📁 main.cpp
#include <iostream>extern int count; // 声明,而非定义int main() {std::cout << "Count is: " << count << std::endl;return 0;
}
这样就不会报错,main.cpp中的这个extern就会告诉编译器,这个count已经在其他地方定义了,你自己去找就好了
❌四、extern 常见错误分析
❶ 多重定义错误(multiple definition)
// file1.cpp
int count = 10;// file2.cpp
int count = 20; // ❌ error: multiple definition of 'count'
✅ 解决方式:
只在一个 .cpp 文件中定义变量,在其他文件用 extern 声明
❷ undefined reference 错误
// main.cpp
extern int num;int main() {return num; // ❌ error: undefined reference to `num`
}
这个错误表示你声明了变量,但链接阶段找不到定义。
✅ 解决方式:
确保有一个源文件里有 int num = 0; 的定义
总结
extern 是你理解多文件组织、链接过程、变量生命周期的桥梁。不掌握它,你永远会被 undefined reference 等链接错误困扰。
如果你觉得本文对你有帮助,不妨点赞 + 收藏 + 关注,更多 C++ 系列教程将持续更新 🔥!
相关文章:

深入解析C++中的extern关键字:跨文件共享变量与函数的终极指南
🚀 C extern 关键字深度解析:跨文件编程的终极指南 📅 更新时间:2025年6月5日 🏷️ 标签:C | extern关键字 | 多文件编程 | 链接与声明 | 现代C 文章目录 前言🔥一、extern 是什么?&…...
在鸿蒙HarmonyOS 5中使用DevEco Studio实现录音机应用
1. 项目配置与权限设置 1.1 配置module.json5 {"module": {"requestPermissions": [{"name": "ohos.permission.MICROPHONE","reason": "录音需要麦克风权限"},{"name": "ohos.permission.WRITE…...

UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)
UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中,UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化…...

SpringCloudGateway 自定义局部过滤器
场景: 将所有请求转化为同一路径请求(方便穿网配置)在请求头内标识原来路径,然后在将请求分发给不同服务 AllToOneGatewayFilterFactory import lombok.Getter; import lombok.Setter; import lombok.extern.slf4j.Slf4j; impor…...
uniapp中使用aixos 报错
问题: 在uniapp中使用aixos,运行后报如下错误: AxiosError: There is no suitable adapter to dispatch the request since : - adapter xhr is not supported by the environment - adapter http is not available in the build 解决方案&…...

(转)什么是DockerCompose?它有什么作用?
一、什么是DockerCompose? DockerCompose可以基于Compose文件帮我们快速的部署分布式应用,而无需手动一个个创建和运行容器。 Compose文件是一个文本文件,通过指令定义集群中的每个容器如何运行。 DockerCompose就是把DockerFile转换成指令去运行。 …...

多种风格导航菜单 HTML 实现(附源码)
下面我将为您展示 6 种不同风格的导航菜单实现,每种都包含完整 HTML、CSS 和 JavaScript 代码。 1. 简约水平导航栏 <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport&qu…...
06 Deep learning神经网络编程基础 激活函数 --吴恩达
深度学习激活函数详解 一、核心作用 引入非线性:使神经网络可学习复杂模式控制输出范围:如Sigmoid将输出限制在(0,1)梯度传递:影响反向传播的稳定性二、常见类型及数学表达 Sigmoid σ ( x ) = 1 1 +...

用docker来安装部署freeswitch记录
今天刚才测试一个callcenter的项目,所以尝试安装freeswitch 1、使用轩辕镜像 - 中国开发者首选的专业 Docker 镜像加速服务平台 编辑下面/etc/docker/daemon.json文件为 {"registry-mirrors": ["https://docker.xuanyuan.me"] }同时可以进入轩…...
爬虫基础学习day2
# 爬虫设计领域 工商:企查查、天眼查短视频:抖音、快手、西瓜 ---> 飞瓜电商:京东、淘宝、聚美优品、亚马逊 ---> 分析店铺经营决策标题、排名航空:抓取所有航空公司价格 ---> 去哪儿自媒体:采集自媒体数据进…...
Caliper 配置文件解析:config.yaml
Caliper 是一个区块链性能基准测试工具,用于评估不同区块链平台的性能。下面我将详细解释你提供的 fisco-bcos.json 文件结构,并说明它与 config.yaml 文件的关系。 fisco-bcos.json 文件解析 这个文件是针对 FISCO-BCOS 区块链网络的 Caliper 配置文件,主要包含以下几个部…...
【HTTP三个基础问题】
面试官您好!HTTP是超文本传输协议,是互联网上客户端和服务器之间传输超文本数据(比如文字、图片、音频、视频等)的核心协议,当前互联网应用最广泛的版本是HTTP1.1,它基于经典的C/S模型,也就是客…...

多模态大语言模型arxiv论文略读(108)
CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文标题:CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文作者:Sayna Ebrahimi, Sercan O. Arik, Tejas Nama, Tomas Pfister ➡️ 研究机构: Google Cloud AI Re…...
实现弹窗随键盘上移居中
实现弹窗随键盘上移的核心思路 在Android中,可以通过监听键盘的显示和隐藏事件,动态调整弹窗的位置。关键点在于获取键盘高度,并计算剩余屏幕空间以重新定位弹窗。 // 在Activity或Fragment中设置键盘监听 val rootView findViewById<V…...

QT: `long long` 类型转换为 `QString` 2025.6.5
在 Qt 中,将 long long 类型转换为 QString 可以通过以下两种常用方法实现: 方法 1:使用 QString::number() 直接调用 QString 的静态方法 number(),将数值转换为字符串: long long value 1234567890123456789LL; …...
[Java恶补day16] 238.除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...

华为云Flexus+DeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建
华为云FlexusDeepSeek征文|DeepSeek-V3/R1 商用服务开通全流程与本地部署搭建 前言 如今大模型其性能出色,华为云 ModelArts Studio_MaaS大模型即服务平台华为云内置了大模型,能助力我们轻松驾驭 DeepSeek-V3/R1,本文中将分享如何…...

智能仓储的未来:自动化、AI与数据分析如何重塑物流中心
当仓库学会“思考”,物流的终极形态正在诞生 想象这样的场景: 凌晨3点,某物流中心灯火通明却空无一人。AGV机器人集群根据实时订单动态规划路径;AI视觉系统在0.1秒内扫描包裹信息;数字孪生平台正模拟次日峰值流量压力…...
高防服务器能够抵御哪些网络攻击呢?
高防服务器作为一种有着高度防御能力的服务器,可以帮助网站应对分布式拒绝服务攻击,有效识别和清理一些恶意的网络流量,为用户提供安全且稳定的网络环境,那么,高防服务器一般都可以抵御哪些网络攻击呢?下面…...

自然语言处理——循环神经网络
自然语言处理——循环神经网络 循环神经网络应用到基于机器学习的自然语言处理任务序列到类别同步的序列到序列模式异步的序列到序列模式 参数学习和长程依赖问题基于门控的循环神经网络门控循环单元(GRU)长短期记忆神经网络(LSTM)…...

【论文阅读28】-CNN-BiLSTM-Attention-(2024)
本文把滑坡位移序列拆开、筛优质因子,再用 CNN-BiLSTM-Attention 来动态预测每个子序列,最后重构出总位移,预测效果超越传统模型。 文章目录 1 引言2 方法2.1 位移时间序列加性模型2.2 变分模态分解 (VMD) 具体步骤2.3.1 样本熵(S…...
根据万维钢·精英日课6的内容,使用AI(2025)可以参考以下方法:
根据万维钢精英日课6的内容,使用AI(2025)可以参考以下方法: 四个洞见 模型已经比人聪明:以ChatGPT o3为代表的AI非常强大,能运用高级理论解释道理、引用最新学术论文,生成对顶尖科学家都有用的…...

select、poll、epoll 与 Reactor 模式
在高并发网络编程领域,高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表,以及基于它们实现的 Reactor 模式,为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。 一、I…...

SpringTask-03.入门案例
一.入门案例 启动类: package com.sky;import lombok.extern.slf4j.Slf4j; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cache.annotation.EnableCach…...
什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南
文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果 自用
outline: none 用于移除元素获得焦点时默认的轮廓线 broder:0 用于移除边框 font-size:0 用于设置字体不显示 list-style: none 消除<li> 标签默认样式 margin: xx auto 版心居中 width:100% 通栏 vertical-align 作用于行内元素 / 表格单元格ÿ…...
【C++从零实现Json-Rpc框架】第六弹 —— 服务端模块划分
一、项目背景回顾 前五弹完成了Json-Rpc协议解析、请求处理、客户端调用等基础模块搭建。 本弹重点聚焦于服务端的模块划分与架构设计,提升代码结构的可维护性与扩展性。 二、服务端模块设计目标 高内聚低耦合:各模块职责清晰,便于独立开发…...

ios苹果系统,js 滑动屏幕、锚定无效
现象:window.addEventListener监听touch无效,划不动屏幕,但是代码逻辑都有执行到。 scrollIntoView也无效。 原因:这是因为 iOS 的触摸事件处理机制和 touch-action: none 的设置有关。ios有太多得交互动作,从而会影响…...

在WSL2的Ubuntu镜像中安装Docker
Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...