深入理解C/C++预处理器指令#pragma once以及与ifndef的比较
#pragma once用法总结
为了防止重复引用造成二义性
在C/C++中,在使用预编译指令#include的时候,为了防止重复引用造成二义性,通常有两种方式
第一种是#ifndef指令防止代码块重复引用,比如说
#ifndef _CODE_BLOCK #define _CODE_BLOCK// code#endif// _CODE_BLOCK
第二种就是#pragma once指令,在想要保护的文件开头写入
#pragma once
pragma once 的底层实现
#pragma once
是C和C++编程语言中的预处理器指令。预处理器是编译器的一个组成部分,它在编译过程的最初阶段处理源代码。当预处理器遇到#pragma
指令时,它会执行特殊的操作,具体取决于紧跟在#pragma
后面的文本。在#pragma once
的情况下,预处理器将记录包含此指令的文件,并确保在单次编译过程中不会多次包含该文件。具体来说,当预处理器遇到
#pragma once
时,它通常会做以下几件事:
预处理器将在内部维护一个列表,记录已经处理过的头文件。每当遇到
#pragma once
,预处理器都会检查这个列表。如果头文件已经在列表中,预处理器就会跳过这个头文件,不再处理它。这就防止了同一个头文件在单次编译中被重复包含。
如果头文件还不在列表中,预处理器就会将其添加到列表中,然后正常处理这个头文件。
需要注意的是,
#pragma once
并不是标准C++的一部分,尽管大多数现代C++编译器都支持它。因此,其具体行为可能会因编译器而异。在一些情况下,可能会存在问题,比如在文件系统链接(例如软链接或硬链接)或者网络文件系统等复杂场景下,编译器可能无法正确识别文件的唯一性,从而导致#pragma once
不能正常工作。在很多情况下,使用传统的头文件保护宏(header guard)是更安全、更标准的做法,例如:
#ifndef HEADER_FILE_H #define HEADER_FILE_H// ... header file content ...#endif // HEADER_FILE_H
在这种情况下,预处理器将检查
HEADER_FILE_H
是否已经定义。如果已经定义,那么在#ifndef
和#endif
之间的内容将被忽略。如果尚未定义,预处理器将定义HEADER_FILE_H
,并处理在#ifndef
和#endif
之间的内容。这是一种手动实现#pragma once
功能的方法,而且是C++标准支持的。
#pragma once和#ifndef区别
在C/C++编程中,为了防止头文件的重复引用,我们通常使用两种预处理器指令:
#ifndef
和#pragma once
。这两种指令都有各自的优点和缺点,选择使用哪一种主要取决于你的具体需求和环境。以下是对这两种预处理器指令的比较:
指令 描述 优点 缺点 #ifndef
依赖于自定义的宏名,可以保证同一份文件或内容相同的不同文件不会被包含两次。 1. 是C/C++语言的标准支持,兼容性好。 2. 可以针对一个文件中的部分代码。 3. 更加灵活。 1. 如果自定义的宏名重名,可能导致编译器找不到声明的情况。 2. 编译器每次都需要打开头文件才能判定是否有重复定义,可能使得编译大型项目时的时间较长。 #pragma once
由编译器提供保证,同一个文件不会被包含多次。 1. 不会出现宏名碰撞引发的问题。 2. 可以提高大型项目的编译速度。 3. 操作简单,效率高。 1. 是非标准的,有些编译器不支持,如GCC 3.4版本之前不支持。 2. 只能针对整个文件。 3. 如果某个头文件有多份拷贝,不能保证它们不被重复包含。 总的来说,
#ifndef
和#pragma once
都是用于防止头文件的重复引用的有效工具,但它们在实际使用中的效果可能会因项目的具体需求和环境而异。对于#pragma once和#ifndef在C++中的使用,你已经给出了非常详细的解释。这两种预处理器指令都用于防止头文件的重复包含,但它们的工作方式和兼容性有所不同。
编译器差异:#pragma once是一个非标准的预处理器指令,这意味着它可能不被所有的编译器支持。然而,大多数现代编译器(包括GCC,Clang,MSVC等)都支持这个指令。另一方面,#ifndef是一个标准的预处理器指令,因此它在所有的C++编译器中都是被支持的。
文件系统差异:#pragma once的行为可能会受到文件系统的影响。例如,在某些文件系统中,同一文件可能有多个有效的路径,这可能会导致编译器错误地将同一文件视为不同的文件。这是因为#pragma once是通过文件路径来判断文件是否已经被包含的。相比之下,#ifndef不受这种问题的影响,因为它是通过宏名来判断的。
代码可读性:#pragma once指令的语法比#ifndef更简洁,这可能会使代码更易于阅读和理解。然而,这也可能会使得一些复杂的条件编译场景变得更难处理,因为#ifndef提供了更多的灵活性。
性能:在某些情况下,#pragma once可能会比#ifndef更快。这是因为编译器可以立即知道一个文件是否已经被包含,而不需要扫描整个文件。然而,这个性能优势可能并不显著,特别是对于较小的项目。
总的来说,#pragma once和#ifndef都有各自的优点和缺点,选择使用哪一个主要取决于你的具体需求和环境。
相关文章:
深入理解C/C++预处理器指令#pragma once以及与ifndef的比较
#pragma once用法总结 为了防止重复引用造成二义性 在C/C中,在使用预编译指令#include的时候,为了防止重复引用造成二义性,通常有两种方式 第一种是#ifndef指令防止代码块重复引用,比如说 #ifndef _CODE_BLOCK #define _CODE_BLO…...

git 环境配置 + gitee拉取代码
好嘛 配环境的时候 老是忘记这个命令行 干脆自己写一个记录一下 也不用搜了 1.先从git官网下载git 安装 2.然后从gitee拉取代码的时候提示 这是因为换了新电脑没有加入新的公钥啦 哎 所以老是记不住命令行 first : git config --global user.name “Your Name” …...

港联证券|港股拥抱特专科技企业 内资券商“修炼内功”蓄势而为
港股市场新一轮改革举措渐次落地。特别是港交所推出特专科技公司上市机制,吸引符合资格的科技企业申请赴港上市,成为这一轮港股市场改革的“重头戏”。 作为香港资本市场的重要参与者,内资券商立足香港、背靠内地、辐射全球,走出一…...

多项创新技术加持,实现零COGS的Microsoft Editor语法检查器
编者按:Microsoft Editor 是一款人工智能写作辅助工具,其中的语法检查器(grammar checker)功能不仅可以帮助不同水平、领域的用户在写作过程中检查语法错误,还可以对错误进行解释并给出正确的修改建议。神经语法检查器…...

Python编程环境搭建:Windows中如何安装Python
在 Windows 上安装 Python 和安装普通软件一样简单,下载安装包以后猛击“下一步”即可。 Python 安装包下载地址:https://www.python.org/downloads/ 打开该链接,可以看到有两个版本的 Python,分别是 Python 3.x 和 Python 2.x&…...

Sui Builder House首尔站倒计时!
Sui主网上线后的第一场Builder House活动即将在韩国首尔举行,同期将举办首场线下面对面的黑客松。活动历时两天,将为与会者提供独特的学习、交流和娱乐的机会。活动详情请查看:Sui Builder House首尔站|主网上线后首次亮相。 Sui…...
Java设计模式-状态模式
简介 在软件开发领域,设计模式是一组经过验证的、被广泛接受的解决问题的方案。其中之一是状态模式,它提供了一种优雅的方式来管理对象的不同状态。 状态模式是一种行为型设计模式,它允许对象在内部状态发生改变时改变其行为。状态模式将对…...
智慧社区用什么技术开发
智慧社区是指利用信息技术和先进的管理理念,将社区内的各种公共服务进行整合和优化,提高社区居民的生活品质和社区管理的效率。为了实现智慧社区的建设,需要采用多种技术,包括但不限于以下几种: 1.物联网技术…...
多线程 线程池饱和策略
RejectedExecutionHandler(饱和策略):当队列和线程池都满了,说明线程池处于饱和状态,那么必须采取一种策略处理提交的新任务。 这个策略默认情况下是AbortPolicy,表示无法处理新任务时抛出异常。 在JDK 1…...
进程间通信之信号
进程间通信之信号 1. 信号2. 信号由谁产生?3. 有哪些信号4. 信号的安装5. 信号的发送1) 使用kill函数2)使用alarm函数3) 使用raise6.发送多个信号7. 信号集1. 信号 什么是信号? 信号是给程序提供一种可以处理异步事件的方法,它利用软件中断来实现。不能自定义信号,所有信号…...
二分查找三道题
二分查找 两种写法:左闭右闭[left,right]、左闭右开[left,right) 主要有几点不同:1. right是从num.length开始还是从num.length-1开始。2.left<还是<right。3.rightmid还是mid1 左闭右闭写法: public int search(int[] nums, int targ…...

MyBatis 框架
MyBatis 框架 MyBatis 简介搭建 MyBatis 开发环境核心配置文件详解mapper 映射文件(实现增删改查)MyBatis获取参数值的两种方式MyBatis的各种查询功能特殊SQL的执行自定义映射resultMapresultMap 字段和属性的映射多对一映射处理一对多映射处理 动态SQLM…...

【C++】虚表和虚基表到底有哪些区别?
虚表和虚基表 虚表虚基表虚拟继承和虚函数都存在时的对象模型 虚表 我们知道,如果类中声明了的方法是用virtual进行修饰的,则说明当前这个方法要作为虚函数,而虚函数的存储和普通函数的存储是有区别的 当有虚函数声明时,编译器会…...

剑指 Offer 04. 二维数组中的查找解题思路
文章目录 标题解题思路优化 标题 在一个 n * m 的二维数组中,每一行都按照从左到右 非递减 的顺序排序,每一列都按照从上到下 非递减 的顺序排序。请完成一个高效的函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整…...

冯诺依曼体系结构详解
一.冯诺伊曼体系结构的概念: 约翰冯诺依曼(John von Neumann,1903.1.28-1957.2.8),美籍匈牙利数学家,计算机科学家,物理学家。是20世纪最重要的数学家之一,后来被称为计算机之父。 后…...

ISO证书“带标”与“不带标”的区别是什么?
ISO9001质量管理体系认证是企业产品获得“通行绿卡”的最直接最有效的途径。 通过认证在打破贸易壁垒,提高产品知名度,降低生产成本,提高经济效益,维护消费者权益,减少重复审核负担等方面的作用越来越为企业界所共知。…...

RocketMQ 领域模型概述
本文为您介绍 Apache RocketMQ 的领域模型。 Apache RocketMQ 是一款典型的分布式架构下的中间件产品,使用异步通信方式和发布订阅的消息传输模型。通信方式和传输模型的具体说明,请参见下文通信方式介绍和消息传输模型介绍。 Apache RocketMQ 产品具备…...

黄河千年清一回与人类健康
黄河千年清一回奏响一曲曲让人类走进幸福新时代的壮丽凯歌。疫情之后的首届全世界健康产业发展大会 5 月28 日上午 9 时在中国首都北京召开 The Yellow River has played a magnificent song of triumph in the millennium, ushering humanity into a new era of happiness. T…...

Android java层hook------xposed框架的使用
xposed曾经是android平台上最好的java层hook和调试工具,由于已经不再更新,当前支持的android系统版本比较老旧,目前只能支持到android6.0,故已经逐渐落伍,目前android上最广泛使用的hook工具是frida,这是另…...

css奇淫巧计
1.input文本内容替换 -webkit-text-security:通过用形状替换字符,仅影响那些字段不是的typepassword 可选值:none (无),circle (圆圈),disc (圆形),square &a…...

龙虎榜——20250610
上证指数放量收阴线,个股多数下跌,盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型,指数短线有调整的需求,大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的:御银股份、雄帝科技 驱动…...

解决Ubuntu22.04 VMware失败的问题 ubuntu入门之二十八
现象1 打开VMware失败 Ubuntu升级之后打开VMware上报需要安装vmmon和vmnet,点击确认后如下提示 最终上报fail 解决方法 内核升级导致,需要在新内核下重新下载编译安装 查看版本 $ vmware -v VMware Workstation 17.5.1 build-23298084$ lsb_release…...

【第二十一章 SDIO接口(SDIO)】
第二十一章 SDIO接口 目录 第二十一章 SDIO接口(SDIO) 1 SDIO 主要功能 2 SDIO 总线拓扑 3 SDIO 功能描述 3.1 SDIO 适配器 3.2 SDIOAHB 接口 4 卡功能描述 4.1 卡识别模式 4.2 卡复位 4.3 操作电压范围确认 4.4 卡识别过程 4.5 写数据块 4.6 读数据块 4.7 数据流…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...

MySQL 8.0 OCP 英文题库解析(十三)
Oracle 为庆祝 MySQL 30 周年,截止到 2025.07.31 之前。所有人均可以免费考取原价245美元的MySQL OCP 认证。 从今天开始,将英文题库免费公布出来,并进行解析,帮助大家在一个月之内轻松通过OCP认证。 本期公布试题111~120 试题1…...
06 Deep learning神经网络编程基础 激活函数 --吴恩达
深度学习激活函数详解 一、核心作用 引入非线性:使神经网络可学习复杂模式控制输出范围:如Sigmoid将输出限制在(0,1)梯度传递:影响反向传播的稳定性二、常见类型及数学表达 Sigmoid σ ( x ) = 1 1 +...

QT: `long long` 类型转换为 `QString` 2025.6.5
在 Qt 中,将 long long 类型转换为 QString 可以通过以下两种常用方法实现: 方法 1:使用 QString::number() 直接调用 QString 的静态方法 number(),将数值转换为字符串: long long value 1234567890123456789LL; …...
今日学习:Spring线程池|并发修改异常|链路丢失|登录续期|VIP过期策略|数值类缓存
文章目录 优雅版线程池ThreadPoolTaskExecutor和ThreadPoolTaskExecutor的装饰器并发修改异常并发修改异常简介实现机制设计原因及意义 使用线程池造成的链路丢失问题线程池导致的链路丢失问题发生原因 常见解决方法更好的解决方法设计精妙之处 登录续期登录续期常见实现方式特…...

【分享】推荐一些办公小工具
1、PDF 在线转换 https://smallpdf.com/cn/pdf-tools 推荐理由:大部分的转换软件需要收费,要么功能不齐全,而开会员又用不了几次浪费钱,借用别人的又不安全。 这个网站它不需要登录或下载安装。而且提供的免费功能就能满足日常…...

GO协程(Goroutine)问题总结
在使用Go语言来编写代码时,遇到的一些问题总结一下 [参考文档]:https://www.topgoer.com/%E5%B9%B6%E5%8F%91%E7%BC%96%E7%A8%8B/goroutine.html 1. main()函数默认的Goroutine 场景再现: 今天在看到这个教程的时候,在自己的电…...