C++ include头文件的顺序以及双引号““和尖括号<>的区别
本文章进一步详细解释 #include 的头文件包含机制,包括搜索路径的处理、双引号 "" 和尖括号 <> 在不同环境中的使用差异,以及它们的底层机制。
1. 头文件包含机制和搜索路径详解
#include 是一个预处理指令,用于在编译前将指定的头文件的内容插入到当前源文件中。在 C++ 编译过程中,头文件的查找顺序和搜索路径的选择会影响项目的编译成功与否。
双引号 "" 的搜索路径机制
当使用双引号 #include "filename" 包含头文件时,编译器通常按照以下顺序查找头文件:
- 当前文件所在的目录:
- 编译器首先查找与当前源文件同目录的头文件。这是因为我们通常使用双引号来包含本地定义的头文件(如类定义的
.h文件)。
- 编译器首先查找与当前源文件同目录的头文件。这是因为我们通常使用双引号来包含本地定义的头文件(如类定义的
- 包含该文件的文件夹:
- 如果该源文件本身是通过
#include被其他文件引用的,编译器会在最先引用该文件的文件夹内查找。
- 如果该源文件本身是通过
- 指定的项目包含路径(通过编译器参数指定的路径,例如
-I选项):- 编译器会查找通过编译器命令行参数指定的头文件路径(
-I/path/to/include)。这个路径可以是多个目录,依赖于构建系统的设置。
- 编译器会查找通过编译器命令行参数指定的头文件路径(
- 全局系统路径:
- 如果前面的路径中没有找到文件,编译器会在全局头文件目录中查找。通常这些路径是操作系统或者 IDE 预先配置好的系统路径,比如
/usr/include(Linux 系统)或者C:\Program Files\(Windows 系统)。
- 如果前面的路径中没有找到文件,编译器会在全局头文件目录中查找。通常这些路径是操作系统或者 IDE 预先配置好的系统路径,比如
尖括号 <> 的搜索路径机制
当使用尖括号 #include <filename> 包含头文件时,编译器仅在全局或系统路径中查找头文件,忽略当前目录。这是为了更高效地查找标准库和第三方库中的头文件。查找路径如下:
-
标准库路径:
- 编译器首先在 C++ 标准库的路径中查找头文件。这些路径可能包含
<iostream>,<vector>,<string>等标准库文件。
- 编译器首先在 C++ 标准库的路径中查找头文件。这些路径可能包含
-
系统库路径:
- 然后,编译器会查找系统头文件路径。比如,在 Linux 系统中,常见路径为
/usr/include和/usr/local/include,而在 Windows 系统中,可能是C:\Program Files (x86)\Microsoft Visual Studio\。
- 然后,编译器会查找系统头文件路径。比如,在 Linux 系统中,常见路径为
-
指定的头文件路径:
- 依赖于编译器设置,可能还会根据用户指定的第三方库路径(如通过
-I指定的路径)来查找第三方库的头文件,例如 Boost、Qt 等。
- 依赖于编译器设置,可能还会根据用户指定的第三方库路径(如通过
双引号和尖括号的选择与差异
- 使用双引号
""更适合引用 项目内部 或者 本地定义的头文件,因为它会首先查找当前文件目录,确保本地头文件的优先级较高。这种方式适用于项目间的模块化开发。 - 使用尖括号
<>通常用于 标准库 和 外部库 的头文件,避免在当前目录下无意中引用到同名的头文件。
更详细的搜索过程比较
| 使用符号 | 查找顺序 | 适用场景 |
|---|---|---|
双引号 "" | 1. 当前目录 2. 引用文件所在目录 3. 编译器指定的项目路径(如 -I)4. 系统路径 | 用于引用本地的头文件 |
尖括号 <> | 1. 标准库路径 2. 系统路径 3. 第三方路径 | 用于标准库和外部库的头文件 |
编译器选项与头文件路径
在使用编译器时,可以通过选项配置头文件的查找路径。例如:
-
GCC/Clang:
-I:指定额外的头文件路径。例如,g++ -I/path/to/headers main.cpp会告诉编译器在/path/to/headers中查找头文件。
-
Visual Studio:
- 在项目属性的 "C/C++" > "附加包含目录" 中,可以指定额外的包含目录。编译器会在这些目录中查找双引号
""和尖括号<>所包含的头文件。
- 在项目属性的 "C/C++" > "附加包含目录" 中,可以指定额外的包含目录。编译器会在这些目录中查找双引号
-
CMake:
- 使用
include_directories()或target_include_directories()指定头文件的搜索路径。
- 使用
2. 双引号和尖括号的深层机制
双引号和尖括号的主要区别在于它们告诉编译器如何查找文件的路径。尖括号中的头文件往往是一些不常修改的外部依赖,比如标准库文件、系统级文件或者第三方库,而双引号中的头文件是用户自定义的文件,更容易发生变化。因此,双引号和尖括号的查找路径和优先级设计不同。
双引号 vs 尖括号查找行为
双引号 "filename":
1. 查找当前目录/源文件目录
2. 查找编译器路径(通过 -I 指定)
3. 查找标准系统路径
尖括号 <filename>:
1. 查找标准系统路径
2. 查找第三方路径
3. #include 的例子与实践
包含自定义头文件的实例
#include "MyClass.h" // 当前目录查找
#include "subfolder/Helper.h" // 指定子文件夹查找
#include <iostream> // 系统库查找,标准C++库
#include <cmath> // 系统路径下的数学库
在这个例子中,"MyClass.h" 和 "subfolder/Helper.h" 都是项目内的文件,编译器会从源文件所在的目录或项目结构中的路径查找。<iostream> 和 <cmath> 是标准库的头文件,因此编译器直接在标准路径中查找。
4. 如何正确组织头文件
为了避免潜在的查找冲突和提高编译速度,建议遵循以下头文件组织原则:
-
使用前向声明:
- 尽量在头文件中使用前向声明来减少包含不必要的头文件。例如,在类成员中引用其他类的指针时,可以通过前向声明避免引入整个类的定义。
class MyClass; // 前向声明
class AnotherClass {
MyClass* ptr; // 使用指针或引用即可避免包含整个 MyClass 头文件
};
2.防止多次包含(Include Guard):
- 为了避免头文件的多次包含,常用的做法是使用包含防护,即
#ifndef、#define、#endif组合,或者在现代 C++ 中使用#pragma once。 -
// MyClass.h
#ifndef MYCLASS_H
#define MYCLASS_Hclass MyClass {
// Class Definition
};#endif // MYCLASS_H
-
确保头文件独立:
- 每个头文件应该可以独立包含,不依赖其他头文件的顺序。通过包含防护和适当的前向声明,避免头文件之间的复杂依赖关系。
5. 总结
- 双引号
"":优先用于包含项目中的本地文件,编译器会先从当前目录查找,再到系统路径查找。 - 尖括号
<>:用于包含标准库或外部库文件,编译器直接从系统路径查找。 - 头文件包含顺序:优先包含与当前实现文件相关的头文件,然后是标准库头文件,接着是第三方库,最后是项目内部其他模块的头文件。
相关文章:
C++ include头文件的顺序以及双引号““和尖括号<>的区别
本文章进一步详细解释 #include 的头文件包含机制,包括搜索路径的处理、双引号 "" 和尖括号 <> 在不同环境中的使用差异,以及它们的底层机制。 1. 头文件包含机制和搜索路径详解 #include 是一个预处理指令,用于在编译前将…...
Flutter鸿蒙版本灵活使用方法间的回调处理复杂化的逻辑
目录 写在前面 示例代码 main.dart: one.dart: 代码解析 1. 主入口 main 函数 2. MyApp 类 3. CallbackExample 类 4. onok 函数 5. one 函数 写在后面 写在前面 在 Flutter 开发中,灵活使用函数之间的回调带来了多种好处,包括提高可重用性、…...
视频号直播自动回复与循环发送话术-自动化插件
我们在做视频号直播的时候,会有这种自动回复咨询问题的功能 唯一客服浏览器插件现在就支持,在视频号直播后台,自动化回复用户问题,以及循环发送我们的介绍话术...
springcloud之服务集群注册与发现 Eureka
前言 1:对于能提供完整领域服务接口功能的RPC而言,例如;gRPC、Thrift、Dubbo等,服务的注册与发现都是核心功能中非常重要的一环,使得微服务得到统一管理。 2:在分布式领域中有个著名的CAP理论;…...
C++:模拟实现list
目录 节点 迭代器 整体框架 构造函数 empty_init 拷贝构造 赋值重载 析构函数 clear insert erase push_back和push_front pop_back和push_front size empty Print_Container 节点 对于链表节点,我们需要一个数据、一个前驱指针、一个后继指针来维护…...
解锁5 大无水印热门短视频素材库
想让你的抖音视频更出彩吗?想知道那些爆款视频的素材源头吗?快来了解以下 5 个超棒的视频素材下载平台。 蛙学网 国内的视频素材佼佼者,有大量 4K 高清且无水印的素材,自然风光、情感生活等类别任你选,不少还免费&…...
【电商购物管理系统】Python+Django网页界面平台+商品管理+数据库
一、介绍 电商购物管理系统,本系统前端使用HTML、CSS、BootStrap等技术搭建前端界面,后端使用Django框架处理用户的逻辑请求。主要功能有: 管理员登录与管理:管理员可以登录后台,对用户和商品进行增删改查的操作。用…...
AD9248驱动的简易示波器设计——FPGA学习笔记21
一、原理 我们这里设计的是显示 1024 个波形数据点, 在绘制每一行的图像的时候, 比对每一个数据和 VS 的 Y 坐标是否相等, 如果相等就绘制这个波形点。 这样我们就能完成 1024 个波形点在整个屏幕的显示。 二、乒乓操作 可见FPGA实现双口RAM…...
微软十月补丁星期二发现了 118 个漏洞
微软将在2024 年 10 月补丁星期二解决 118 个漏洞,并且有证据表明发布的 5 个漏洞被野蛮利用和/或公开披露,尽管微软尚未将其中任何一个漏洞评定为严重漏洞。 在这五个漏洞中,微软列出了两个已被利用的漏洞,这两个漏洞现在都已列…...
到底是微服务,还是SOA?
引言:大概正式工作有5年了,换了三个大厂【也是真特么世道艰难,中国互联网人才饱和了】。基本上每个公司有的架构都不太相同,干过TOC和TOB的业务,但是大家用的架构都不太相同。有坚持ALL in one的SB,最后服务…...
JDK17常用新特性
目前国内大部分开发人员都是在使用jdk8,甚至是jdk6,但是随着jdk的更新迭代,jdk8我觉得可能就会慢慢的淡出舞台,随着目前主流框架最新版推出明确说明了不再支持jdk8,也促使我不得不抓紧学习了解一波jdk17的新特性&#…...
【分布式微服务云原生】探索负载均衡的艺术:深入理解与实践指南
探索负载均衡的艺术:深入理解与实践指南 摘要: 在本文中,我们将深入探讨负载均衡的概念、重要性以及实现负载均衡的多种算法。通过详细的技术解析、Java代码示例、流程图和对比表格,您将了解如何选择合适的负载均衡策略来优化资源…...
拥抱云原生
专题七:云原生实战72课时 专题简介: 云原生正在改变世界,新一代架构思想ServiceMesh、Serverless改变传统软件架构模式,本专题基于完全云上架构实战,结合微服务架构和云计算平台两者的优势,属于架构师必备…...
关于使用若依并快速构建系统的操作指南
准备阶段--下载源码(脚手架) 1.1 若依官网地址:https://www.ruoyi.vip/ 1.2 选择“前后端分离版本进行下载”,如下图所示 1.3 跳转gitee后,直接按如下步骤进行下载。 前后端模块分离 解压,并打开到项目…...
【分布式微服务云原生】 选择SOAP还是RESTful API?深入探讨与实践指南
🌐 选择SOAP还是RESTful API?深入探讨与实践指南 摘要: 在构建现代Web服务时,开发者常常面临一个关键决策:是选择SOAP还是RESTful API?本文将为您提供一个全面的比较,包括两者的适用场景、安全…...
HarmonyOS NEXT 应用开发实战(五、页面的生命周期及使用介绍)
HarmonyOS NEXT是华为推出的最新操作系统,arkUI是其提供的用户界面框架。arkUI的页面生命周期管理对于开发者来说非常重要,因为它涉及到页面的创建、显示、隐藏、销毁等各个阶段。以下是arkUI页面生命周期的介绍及使用举例。 页面的生命周期的作用 页面…...
C# 比较两个集合和比较对象
1、比较集合 /// <summary> /// 比较两个集合 /// </summary> /// <typeparam name"T"></typeparam> /// <param name"list1"></param> /// <param name"list2"></param> /// <returns>&…...
Spark高级用法-自定义函数
用户可以根据需求自己封装计算的逻辑,对字段数据进行计算 内置函数,是spark提供的对字段操作的方法 ,split(字段) 对字段中的数进行切割,F.sum(字段) 会将该字段下的数据进行求和 实际业务中又能内置函数不满足计算需求࿰…...
『Mysql进阶』Mysql explain详解(五)
目录 Explain 介绍 Explain分析示例 explain中的列 1. id 列 2. select_type 列 3. table 列 4. partitions 列 5. type 列 6. possible_keys 列 7. key 列 8. key_len 列 9. ref 列 10. rows 列 11. filtered 列 12. Extra 列 Explain 介绍 EXPLAIN 语句提供有…...
【工具】音视频翻译工具基于Whisper+ChatGPT
OpenAI推出的开源语音识别工具Whisper,以其卓越的语音识别能力,在音频和视频文件处理领域大放异彩。与此同时,ChatGPT也在翻译领域崭露头角,其强大的翻译能力备受赞誉。因此,一些字幕制作团队敏锐地捕捉到了这两者的结…...
rknn优化教程(二)
文章目录 1. 前述2. 三方库的封装2.1 xrepo中的库2.2 xrepo之外的库2.2.1 opencv2.2.2 rknnrt2.2.3 spdlog 3. rknn_engine库 1. 前述 OK,开始写第二篇的内容了。这篇博客主要能写一下: 如何给一些三方库按照xmake方式进行封装,供调用如何按…...
【JavaEE】-- HTTP
1. HTTP是什么? HTTP(全称为"超文本传输协议")是一种应用非常广泛的应用层协议,HTTP是基于TCP协议的一种应用层协议。 应用层协议:是计算机网络协议栈中最高层的协议,它定义了运行在不同主机上…...
大型活动交通拥堵治理的视觉算法应用
大型活动下智慧交通的视觉分析应用 一、背景与挑战 大型活动(如演唱会、马拉松赛事、高考中考等)期间,城市交通面临瞬时人流车流激增、传统摄像头模糊、交通拥堵识别滞后等问题。以演唱会为例,暖城商圈曾因观众集中离场导致周边…...
SCAU期末笔记 - 数据分析与数据挖掘题库解析
这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...
从零开始打造 OpenSTLinux 6.6 Yocto 系统(基于STM32CubeMX)(九)
设备树移植 和uboot设备树修改的内容同步到kernel将设备树stm32mp157d-stm32mp157daa1-mx.dts复制到内核源码目录下 源码修改及编译 修改arch/arm/boot/dts/st/Makefile,新增设备树编译 stm32mp157f-ev1-m4-examples.dtb \stm32mp157d-stm32mp157daa1-mx.dtb修改…...
【git】把本地更改提交远程新分支feature_g
创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...
[Java恶补day16] 238.除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...
html css js网页制作成品——HTML+CSS榴莲商城网页设计(4页)附源码
目录 一、👨🎓网站题目 二、✍️网站描述 三、📚网站介绍 四、🌐网站效果 五、🪓 代码实现 🧱HTML 六、🥇 如何让学习不再盲目 七、🎁更多干货 一、👨…...
c++第七天 继承与派生2
这一篇文章主要内容是 派生类构造函数与析构函数 在派生类中重写基类成员 以及多继承 第一部分:派生类构造函数与析构函数 当创建一个派生类对象时,基类成员是如何初始化的? 1.当派生类对象创建的时候,基类成员的初始化顺序 …...
永磁同步电机无速度算法--基于卡尔曼滤波器的滑模观测器
一、原理介绍 传统滑模观测器采用如下结构: 传统SMO中LPF会带来相位延迟和幅值衰减,并且需要额外的相位补偿。 采用扩展卡尔曼滤波器代替常用低通滤波器(LPF),可以去除高次谐波,并且不用相位补偿就可以获得一个误差较小的转子位…...
