0

0

C++中的__has_include是什么?(如何探测头文件是否存在以实现兼容性)

尼克

尼克

发布时间:2026-02-16 15:51:11

|

332人浏览过

|

来源于php中文网

原创

__has_include 是 c++17 起支持的预处理器运算符,用于编译期判断头文件是否存在,返回 1 或 0,仅可在 #if 等条件编译中使用,支持尖括号和双引号形式,但不展开宏、不保证可成功包含或特性可用。

c++中的__has_include是什么?(如何探测头文件是否存在以实现兼容性)

__has_include 是什么,它能干什么

它是一个 C++17 起支持的预处理器运算符,用来在编译期判断某个头文件是否存在。不是运行时检查,也不是 #include 失败后兜底——而是让你提前“问一句”,再决定要不要 #include、用哪个版本的接口、甚至跳过某段代码。典型场景是跨平台或跨标准库版本写兼容代码,比如想用 <span></span> 但又得兼容没实现它的旧 libstdc++

怎么用:语法和基本套路

必须在预处理阶段用,只能出现在 #if / #ifdef 等条件编译中,不能放在函数体里或作为变量值。返回 1 表示存在,0 表示不存在(注意:不保证可成功包含,只表示路径能被找到)。

  • #if __has_include(<optional>)</optional> —— 检查系统头文件,尖括号形式
  • #if __has_include("my_header.h") —— 检查用户头文件,双引号形式
  • 不能混用:#if __has_include("optional") 是错的,不会匹配 <optional></optional>
  • 不能带宏展开:#define HEAD "optional" 然后 #if __has_include(HEAD) 不合法,预处理器不展开字符串字面量

常见踩坑:为什么写了却没生效

最容易忽略的是编译器支持和标准模式。它不是所有“C++17”编译就自动开——有些老版本 clang/gcc 需要显式开启,而且行为有差异:

  • gcc 5.1+ 支持,但默认不启用;需加 -std=c++17 或更高,且不能用 -std=gnu++17(某些旧版会禁用)
  • clang 3.9+ 支持,同样依赖 -std=c++17
  • MSVC 直到 19.29(VS 2019 16.11)才完整支持,之前版本可能返回 0 即使头文件存在
  • 就算 __has_include(<filesystem>)</filesystem> 返回 1,也不代表 std::filesystem 可用——可能只是头文件存在但未实现,还得配合 __cpp_lib_filesystem 宏判断

一个真实兼容场景:optional 与 boost::optional 共存

你想优先用 std::optional,没有就退到 boost::optional。光靠 try-include 会触发编译错误,而 __has_include 能干净分叉:

Pixlr Remove BG
Pixlr Remove BG

几秒钟删除图片背景

下载

立即学习C++免费学习笔记(深入)”;

#if __has_include(<optional>)
#  include <optional>
#  define MY_OPTIONAL_NS std
#else
#  include <boost/optional.hpp>
#  define MY_OPTIONAL_NS boost
#endif
<p>using my_opt = MY_OPTIONAL_NS::optional<int>;

注意这里没用 #ifdef __cpp_lib_optional——因为那个宏检测的是语言特性是否启用,不是头文件是否存在;而你真正需要的是“我能不能 #include 这个头”,否则可能 #include <optional></optional> 失败直接中断编译。

真正难的是组合判断:头文件存在 + 特性宏定义 + 编译器实际支持。单独靠 __has_include 只解决第一个环节,漏掉后面两个,照样崩。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
typedef和define区别
typedef和define区别

typedef和define区别在类型检查、作用范围、可读性、错误处理和内存占用等。本专题为大家提供typedef和define相关的文章、下载、课程内容,供大家免费下载体验。

116

2023.09.26

define的用法
define的用法

define用法:1、定义常量;2、定义函数宏:3、定义条件编译;4、定义多行宏。更多关于define的用法的内容,大家可以阅读本专题下的文章。

355

2023.10.11

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1553

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

236

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

127

2025.10.17

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

814

2023.08.22

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

552

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

216

2023.09.04

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

283

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 9.6万人学习

C 教程
C 教程

共75课时 | 4.8万人学习

C++教程
C++教程

共115课时 | 18万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号