C++ 中名字的作用域、概念、嵌套与实践(十八)
1. 名字的作用域基本概念
作用域(scope) 指的是程序中的一个区域(通常被花括号 {}
包围),在这里一个名字(如变量名、函数名、类名等)有其特定含义。
- 在 同一个作用域 中,一个名字只能绑定到唯一的实体(变量、函数或类型)上;否则会产生重定义错误。
- 同一个名字 可以在程序的不同作用域中定义,可能指向不同的实体。这就引出了作用域的层次性和嵌套。
1.1 作用域的生效范围
当你在某处声明并定义了一个名字,它通常在当前作用域开始处一直到该作用域结束时都可用。例如:
int main() {int sum = 0; // sum 的作用域从这里开始,一直到 main 函数结束{int x = 100; // x 的作用域仅限于当前这个花括号内sum += x;} // x 在这里“消亡”,后续无法访问 xreturn 0;
}
sum
一直可用到main
函数结束。x
只在它所在的内部块生效,离开这个块就“超出作用域”了。
2. 不同类型的作用域
在 C++ 中,我们可以根据作用域的定义方式和所在位置,大致分为以下几种常见的作用域类型:
-
全局作用域(global scope)
- 定义在所有函数体和命名空间之外的名字,比如全局变量、函数(如
int main()
)等。 - 一旦声明,全局作用域内的名字可在整个程序中被访问(如果在多文件工程中,需要借助
extern
等机制进行声明引用)。 - 例:
int globalCount = 0;
- 定义在所有函数体和命名空间之外的名字,比如全局变量、函数(如
-
命名空间作用域(namespace scope)
- C++ 提供了命名空间(如
namespace std { }
)来组织和区分名字。 - 命名空间中的实体只在该命名空间的作用域内有效。访问时可加
std::
之类的前缀限定符,也可以在当前作用域通过using namespace std;
或者using std::cout;
引入。
- C++ 提供了命名空间(如
-
块作用域(block scope)
- 也称局部作用域,通常由花括号围成的区域,如函数体、
if
/for
/while
语句块等。 - 例:在
main()
函数中定义的变量sum
就只有在main
的范围内可见。
- 也称局部作用域,通常由花括号围成的区域,如函数体、
-
类作用域(class scope)
- 在类内部声明和定义的成员(成员函数、成员变量)只在类中可见;可以通过公有(public)、保护(protected)或私有(private)等访问说明符控制可见性。
- 同样也算一种“花括号”的作用域,稍微更特殊,因为它涉及到访问权限修饰符。
-
函数作用域与函数原型作用域(相对较少单独提及)
- 函数作用域主要指函数体内部。
- 函数原型作用域指参数列表(形参)的可见范围,通常只在原型声明时可见。
通常,前 3 类是我们日常编程中最常遇到和最需关注的。
3. 嵌套作用域与隐藏规则
3.1 嵌套作用域(Inner & Outer Scope)
一个作用域可以嵌套在另一个作用域之内:
- 外层作用域(outer scope):套在外面的大的作用域;
- 内层作用域(inner scope):被包含在内的较小范围的作用域。
在 C++ 中:
- 如果一个名字在外层作用域中声明,则在其所有内层作用域中都是可见的(前提是没有被隐藏);
- 可以在内层作用域使用相同的名字“重新定义”一个变量,此时 内层的定义 会隐藏(shadow)外层的同名实体。
3.2 隐藏与作用域操作符
-
隐藏(shadowing):当内层作用域中定义了一个与外层作用域同名的实体时,内层实体会覆盖外层对该名字的引用。例如:
int reused = 42; // 全局变量 reused(全局作用域) int main() {int reused = 0; // 局部变量 reused(块作用域,隐藏全局的 reused)// ... }
在
main()
函数中,使用reused
会引用 局部变量 而非全局变量。 -
作用域操作符
::
:可以显式指定访问某个特定的作用域中的名字。例如::reused
表示全局命名空间中的reused
变量(如果存在),而非局部定义的那个。
4. 示例分析
以下示例展示了全局变量与局部变量的隐藏现象,以及如何通过作用域操作符来访问全局变量:
#include <iostream>// 全局变量 reused 拥有全局作用域
int reused = 42;int main() {int unique = 0; // unique 拥有块作用域// #1:此时还没有局部的 reused,所以访问的是全局 reusedstd::cout << reused << " " << unique << std::endl; // 输出:42 0// 定义一个与全局同名的局部变量 reusedint reused = 0; // #2:此时在 main() 的块作用域中,reused 指代局部变量std::cout << reused << " " << unique << std::endl; // 输出:0 0// #3:显式访问全局作用域中的 reusedstd::cout << ::reused << " " << unique << std::endl; // 输出:42 0return 0;
}
总结:同名的局部变量会“隐藏”全局变量。若要访问被隐藏的外层名字,可用作用域操作符(
::
)来强行指定要访问的作用域。
5. 实践建议与注意事项
-
避免不必要的同名隐藏
- 在实践中,并不推荐 定义与全局变量同名的局部变量;这样做可能导致阅读者(包括自己在内)感到困惑。
- 若确有需要(如某些极端情况、或在模板元编程、元数据注入的特殊场景),也要在注释或命名上予以说明。
-
优先使用局部变量
- 相比全局变量,局部变量的作用域更小,方便管理和控制,减少命名冲突风险,也有助于写出更可维护的代码。
-
明确命名
- 如果你打算区分全局变量与局部变量,可以在命名上加前缀(例如
g_
表示全局,m_
表示成员变量等),但要与团队达成一致约定。
- 如果你打算区分全局变量与局部变量,可以在命名上加前缀(例如
-
命名空间
- 对于较大的项目,将相关的函数、类等放在一个命名空间下,避免全局命名污染。
- 当在头文件和源文件之间共享全局变量,借助
extern
和命名空间可以使结构更清晰,也能避免重名冲突。
-
善用作用域来防止变量滥用
- 只在需要的范围内定义变量,作用域越小越好,方便后续维护和安全检查。
- 在 C++17 之后,甚至可以在
if
或switch
语句的初始化中声明变量,仅在该语句内可见。
6. 结语
- 作用域 是 C++ 中非常重要的概念,它决定了一个名字的可见范围以及它指向的具体实体。
- 嵌套作用域 使得外层名字可以被内层使用,但也带来了同名隐藏的可能性;合理地使用和避开这种隐藏是编写清晰代码的一大关键。
- 当你在阅读或维护他人代码时,如果发现同名变量行为诡异,别忘了考虑是不是作用域在作怪。借助
::
作用域操作符可以进行调试或明确访问外层名字。 - 在实际项目中,为了维持代码可读性,建议避免在内层作用域中定义与外层同名的变量。如果有使用全局变量,也要尽可能地减少、或者明确区分其含义与命名。
希望这篇文章能让你更加明确地理解 C++ 中名字的作用域,帮助你在编码中更好地组织与管理命名,写出可读性强、可维护的优秀代码!
参考资料
- cppreference.com (离线亦可查阅)对作用域、命名空间及隐藏规则的介绍
- Modern C++ 编程实战系列书籍与视频课程
相关文章:
C++ 中名字的作用域、概念、嵌套与实践(十八)
1. 名字的作用域基本概念 作用域(scope) 指的是程序中的一个区域(通常被花括号 {} 包围),在这里一个名字(如变量名、函数名、类名等)有其特定含义。 在 同一个作用域 中,一个名字只…...
Go语言nil原理深度解析:底层实现与比较规则
Go语言nil原理深度解析:底层实现与比较规则 引言 在Go语言中,nil 是一个特殊的关键字,用于表示引用类型的“零值”。它在指针、切片、映射、通道、接口和函数等类型中广泛使用。本文将从 底层实现、比较规则、与其他语言的对比 等角度&#…...
使用Java操作Redis
文章目录 常用Redis键操作命令使用Java操作Redis配置 Redis 连接信息创建 Maven 项目结构 操作字符串操作列表操作集合操作散列操作有序集合总结 报错解决办法: 当使用redis-cli连接客户端出现not connected时 使用redis-cli -h 192.168.222.101 -p 6379即可解决 …...
开源的CMS建站系统可以随便用吗?有什么需要注意的?
开源CMS建站系统虽然具有许多优点,但并非完全“随便用”。无论选哪个CMS系统,大家在使用的时候,可以尽可能地多注意以下几点: 1、版权问题 了解开源许可证:不同的开源CMS系统采用不同的开源许可证,如GPL、…...
数据结构:探秘AVL树
本节重点 理解AVL树的概念掌握AVL树正确的插入方法利用_parent指针正确更新平衡因子掌握并理解四种旋转方式:左单旋,右单旋,左右双旋,右左双旋 一、AVL树的概念 AVL树得名于它的发明者G. M. Adelson-Velsky和E. M. Landis&…...
C++ 变量与初始化详解(十五)
1. 变量定义 在 C 中,定义变量的基本形式通常是先写出 类型说明符(type specifier),后面紧跟由逗号分隔的一个或多个变量名,最后以分号结束。简单示例如下: int sum 0, value, units_sold 0; Sales_ite…...
【网络协议详解】—— STP 、RSTP、MSTP技术(学习笔记)
一、STP技术工作原理 STP(Spanning Tree Protocol)生成树协议(IEEE 802.1D)是一种网络协议,用于在网络拓扑中防止环路的产生。在二层交换网络中,逻辑上阻塞部分接口,实现从根交换机到所有节点的…...
C++中将记录集的数据复制到Excel工作表中的CRange类CopyFromRecordset函数异常怎么捕获
文章目录 一、异常类型及捕获逻辑二、完整代码示例三、关键错误场景与解决方案1. CopyFromRecordset 返回空数据2. COM错误 0x800A03EC3. Excel进程残留4. 内存不足 四、调试与日志记录1. 启用详细日志2. 捕获错误描述3. 调试断点 五、最佳实践 在C中使用 CRange::CopyFromReco…...
综述速读|086.04.24.Retrieval-Augmented Generation for AI-Generated Content A Survey
论文题目:Retrieval-Augmented Generation for AI-Generated Content: A Survey 论文地址:https://arxiv.org/abs/2402.19473 bib引用: misc{zhao2024retrievalaugmentedgenerationaigeneratedcontent,title{Retrieval-Augmented Generation…...
对内核fork进程中写时复制的理解记录
前言 文章写于学习Redis时对aof后台重写中写时复制的疑问 一、感到不理解的歧义 在部分技术文档中(以小林的文章为例),对写时复制后的内存权限存在如歧义: ! 二、正确技术表述 根据Linux内核实现(5.15版本&#x…...
【新手初学】SQL注入getshell
一、引入 木马介绍: 木马其实就是一段程序,这个程序运行到目标主机上时,主要可以对目标进行远程控制、盗取信息等功能,一般不会破坏目标主机,当然,这也看黑客是否想要搞破坏。 木马类型: 按照功…...
【湖北工业大学2025年ACM校赛(同步赛)】题解
比赛链接 A. 蚂蚁上树 题目大意 给定一棵 n n n 个结点的树,根结点为 1 1 1。每个 叶结点 都有一只蚂蚁,每过 1 1 1 秒钟,你可以选一些蚂蚁往其 父结点 走一步,但是要求任意两只蚂蚁都不能在同一个 非根结点 上。 问至少要…...
FPGA Verilog/VHDl 中的锁存latch
目录 一、前言二、锁存器定义三、verilog中锁存的产生四、verilog中锁存的影响和消除五、FPGA中的锁存器资源 一、前言 在做FPGA设计时,我们要求在组合逻辑设计时,case或者if-else条件要完整,否则会产生锁存。本文主要介绍锁存产生的原因和影…...
Ubuntu24.04 配置远程桌面服务
一:安装 sudo apt update sudo apt install vino 二:设置 gsettings set org.gnome.Vino require-encryption false # 关闭加密(某些 VNC 客户端不支持加密) gsettings set org.gnome.Vino prompt-enabled false # 关闭连接…...
【二刷代码随想录】螺旋矩阵求解方法、推荐习题
一、求解方法 (1)按点模拟路径 在原有坐标的基准上,叠加 横纵坐标 的变化值,求出下一位置,并按题完成要求。但需注意转角的时机判断,特别是最后即将返回上一出发点的位置。 (2)按层…...
Python基础教程:从格式化到项目管理
一、Typora代码块支持格式 在Typora中编写代码时,支持多种语言的语法高亮显示: 二、代码格式化 1. %格式化(传统方式) 第一种方式:正规方式 示例代码: name "张三" age 10 print("我的…...
Python爬虫:开启数据抓取的奇幻之旅(一)
目录 一、爬虫初印象:揭开神秘面纱 二、工欲善其事:前期准备 (一)Python 环境搭建 1.下载 Python 安装包: 2.运行安装程序: 3.配置环境变量(若自动添加失败)&#x…...
分布式ID服务实现全面解析
分布式ID生成器是分布式系统中的关键基础设施,用于在分布式环境下生成全局唯一的标识符。以下是各种实现方案的深度解析和最佳实践。 一、核心需求与设计考量 1. 核心需求矩阵 需求 重要性 实现难点 全局唯一 必须保证 时钟回拨/节点冲突 高性能 高并发场景…...
浏览器与网络模块实践
浏览器渲染步骤 浏览器渲染大致分为以下四个步骤: 1. 构建 DOM 树 • 过程:当浏览器接收到 HTML 文档后,会从上到下依次解析 HTML 代码。每遇到一个开始标签,就会创建一个对应的 DOM 节点,并根据标签的嵌套关系将这些…...
谈谈Minor GC、Major GC和Full GC
目录 一、背景 二、三者之间的区分 1、Minor GC 2、Major GC (1)老年代空间不足: (2)晋升(Promotion)失败: (3)空间分配担保失败: &#x…...
基于SpringBoot实现的高校实验室管理平台功能四
一、前言介绍: 1.1 项目摘要 随着信息技术的飞速发展,高校实验室的管理逐渐趋向于信息化、智能化。传统的实验室管理方式存在效率低下、资源浪费等问题,因此,利用现代技术手段对实验室进行高效管理显得尤为重要。 高校实验室作为…...
梯度裁剪(Gradient Clipping)
梯度裁剪(Gradient Clipping)是一种用于防止梯度爆炸(Gradient Explosion)的技术,具体来说: 1. 梯度裁剪的作用 问题背景:在训练深度神经网络(尤其是RNN/LSTM)时&#x…...
联合办公空间WeWork的创新模式与私域流量时代的品牌温度——兼论开源AI大模型AI智能名片S2B2C商城小程序源码的潜在价值
摘要:本文聚焦于联合办公空间WeWork的成功模式,深入剖析其如何让创业用户摆脱传统租赁的束缚,打破空间与社交限制,为创业带来新的可能性与趣味性,并有效降低创业成本与风险。同时探讨了WeWork在私域流量时代所建立的平…...
Git配置
为什么要用:下载zip只是当前分支,不能进行仓库push、pull、checkout 1. 下载Git 先判断是否已经下过Git: git --version若没有版本号出来,就去下载:https://git-scm.com/downloads (Windows、linux、mac…...
Protobuf 的快速使用(二)
这个部分会对通讯录进⾏多次升级,使⽤ 2.x 表⽰升级的版本,最终将会升级如下内容: 不再打印联系⼈的序列化结果,⽽是将通讯录序列化后并写⼊⽂件中。 从⽂件中将通讯录解析出来,并进⾏打印。 新增联系⼈属性ÿ…...
网页设计思路
CSS实现思路: 用一个div直接父级继承 在这里插入图片描述 一LOGO结构 h1>a>搜索关键字 二导航栏结构 结构:ul>li>a 三搜索框结构 div>input/a 四用户头像结构 div>a>imgspan 处理行内块和行内垂直对齐方向使用 vertical-align...
Vue3 配合 fullPage.js 打造高效全屏滚动网页
引言 在现代网页设计中,整屏滚动(Full-page Scrolling)已成为展示内容的一种流行方式。通过将内容分成若干个全屏页面,并配合流畅的过渡动画,可以为用户带来身临其境的浏览体验。本文将介绍如何使用 fullPage.js 插件来…...
全排列 II:去重的技巧与实现
全排列 II:去重的技巧与实现 1. 引言:排列问题的坑 你有没有遇到过这样的问题? 当我们在做全排列(Permutation)的时候,如果输入的数组中包含重复元素,生成的排列中就会出现大量重复项。这样不…...
微型导轨和普通导轨有哪些区别?
微型导轨和普通导轨都是常用的工业机械传动装置,目前,市场上有各种各样的导轨产品。那么微型导轨和普通导轨有哪些区别呢? 1、尺寸:微型导轨尺寸较小,滑座宽度最小可达 8MM,长度最小可达 11MM 左右…...
Java 输入流到输出流
Java 输入流到输出流的复制方法主要有以下六种实现方式,根据性能、适用场景和实现原理可分为不同类别: 一、基础字节流方式 实现原理:通过 FileInputStream 和 FileOutputStream 逐字节或块读取数据并写入。 代码示例: try (In…...
Anaconda安装-Ubuntu-Linux
1、进入Anaconda官网,以下载最新版本,根据自己的操作系统选择适配的版本。 2、跳过注册: 3、选择适配的版本: 4、cd ~/anaconda_download 5、bash Anaconda3-2024.10-1-Linux-x86_64.sh 6、按Enter或PgDn键滚动查看协议&…...
每日一题之既约分数
题目描述 本题为填空题,只需要算出结果后,在代码中使用输出语句将所填结果输出即可。 如果一个分数的分子和分母的最大公约数是 1,这个分数称为既约分数。 例如 3/4,1/8,7/1, 都是既约分数。 请问,有多少个既约分…...
诠视科技MR眼镜如何使用VLC 进行RTSP投屏到电脑
文章目录 一、应用开发部分(1)基础场景构建(2)添加XvCameraManager(3)添加XvMRVideoCaptureManager(4)添加XvRTSPStreamerManager(5)打包测试 二、VLC media …...
“头”里有什么——HTML 元信息
2025/3/28 向全栈工程师迈进! 一、看基本HTML <!doctype html> <html lang"zh-CN"><head><meta charset"utf-8" /><title>我的测试页面</title></head><body><p>这是我的页面</p&g…...
【Kafka】从理论到实践的深度解析
在当今数字化转型的时代,企业面临着数据量呈指数级增长、业务系统愈发复杂的挑战。在这样的背景下,高效的数据传输与处理技术成为了关键。Kafka,作为一款分布式消息队列系统,凭借其卓越的性能和丰富的特性,在众多企业的…...
Debezium系列之:使用Debezium和Apache Iceberg构建数据湖
Debezium系列之:使用Debezium和Apache Iceberg构建数据湖 Debezium Server Iceberg“Debezium Server Iceberg” 消费者设置数据复制Upsert 模式保留已删除的记录使用Upsert模式追加模式优化批处理大小在数据分析的世界中,数据湖是存储和管理大量数据以满足数据分析、报告或机…...
resnet网络迁移到昇腾执行(OM上篇)
目录 总体介绍 pytorch迁移OM模型 原始代码详细介绍 模型加载和初始化 初始化统计变量 数据推理及归一化 统计每个样本的结果 基本概念 Softmax(归一化指数函数) 作用 代码示例 应用场景 argmax取最大值索引 作用 代码示例 两者配合使用…...
RHCA核心课程技术解析5:红帽高可用性集群架构与深度实践
一、红帽高可用集群架构全景 1.1 核心组件交互逻辑 graph TD A[节点1] -->|Corosync 心跳| B[节点2] A -->|Pacemaker 资源管理| C[共享存储] B --> C D[Fencing设备] -->|STONITH| A D -->|STONITH| B C -->|GFS2锁管理| A C -->|GFS2锁管理| B 1.2 集…...
Display Serializer、Camera Deserializer(Camera Des)和SerDes 加解串应用
1. 概述:三者的核心定位 (1) SerDes(Serializer/Deserializer) 定义:通用高速数据传输技术,实现并行↔串行双向转换。角色:数据链路的“翻译官”,解决并行传输的带宽与距…...
vue3+bpmn.js基本使用
一、案例使用依赖 // 必填"bpmn-js": "^7.3.1", "bpmn-js-properties-panel": "^0.37.2","bpmn-moddle":"^7.1.3","camunda-bpmn-moddle": "^7.0.1",// 可选"element-plus/icons-vue&qu…...
《数据结构:单链表》
“希望就像星星,或许光芒微弱,但永不熄灭。” 博主的个人gitee:https://gitee.com/friend-a188881041351 一.概念与结构 链表是一种物理存储上非连续、非顺序的存储结构,数据元素的顺序逻辑是通过链表中的指针链接次序实现的。 单…...
RedHatLinux(2025.3.22)
1、创建/www目录,在/www目录下新建name和https目录,在name和https目录下分别创建一个index.htm1文件,name下面的index.html 文件中包含当前主机的主机名,https目录下的index.htm1文件中包含当前主机的ip地址。 (1&…...
C++异常处理完全指南:从原理到实战
文章目录 异常的基本概念基本异常抛出与捕获多类型异常捕获异常重新抛出异常安全异常规范(noexcept)栈展开与析构标准库异常总结 异常的基本概念 异常是程序运行时发生的非预期事件(如除零、内存不足)。C通过try、catch和throw提…...
Oracle 19C 备份
在 Oracle 19c 中,备份数据库通常使用 RMAN(Recovery Manager) 工具,它是 Oracle 提供的官方备份和恢复工具。以下是通过 RMAN 备份 Oracle 19c 数据库的详细步骤和命令。 一、RMAN 基本概念 RMAN 是 Oracle 的备份和恢复工具&am…...
深入理解MySQL聚集索引与非聚集索引
在数据库管理系统中,索引是提升查询性能的关键。MySQL支持多种类型的索引,其中最基础也是最重要的两种是聚集索引和非聚集索引。本文将深入探讨这两种索引的区别,并通过实例、UML图以及Java代码示例来帮助您更好地理解和应用它们。 一、概念…...
用Python打造智能宠物:强化学习的奇妙之旅
友友们好! 我是Echo_Wish,我的的新专栏《Python进阶》以及《Python!实战!》正式启动啦!这是专为那些渴望提升Python技能的朋友们量身打造的专栏,无论你是已经有一定基础的开发者,还是希望深入挖掘Python潜力的爱好者,这里都将是你不可错过的宝藏。 在这个专栏中,你将会…...
OGG故障指南:OGG-01163 Bad column length (xxx) specified for column
报错 OGG-01163 Bad column length (xxx) specified for column AAA in table OWNER.TABLE, maximum allowable length is yyy原因 源端修改了字段长度。 虽然源端和目标端的长度已经通过DDL语句修改到一致,在extract进程未重启的情况下,生成的trail文…...
XML标签格式转换为YOLO TXT格式
针对的是多边形(<polygon>)来描述对象的边界,而不是传统的矩形框(<bndbox>) import xml.etree.ElementTree as ET import os from pathlib import Path# 解析VOC格式的XML文件,提取目标框的标…...
Java的string默认值
在Java中,String类型的默认值取决于其定义和实例化的方式。 以下是关于String默认值的详细说明 未实例化的String变量 如果定义一个String变量但未对其进行实例化(即未使用new关键字或直接赋值),其默认值为:ml-search[null]。这…...
侯捷 C++ 课程学习笔记:C++ 中引用与指针的深度剖析
目录 一、引言 二、引用与指针的基本概念 (一)引用 (二)指针 三、引用与指针的区别 (一)定义与初始化 (二)内存空间与 NULL 值 (三)自增操作 …...