0

0

c++中如何判断整数加法是否产生溢出_c++11标准数值安全检查【详解】

穿越時空

穿越時空

发布时间:2026-01-20 11:20:03

|

425人浏览过

|

来源于php中文网

原创

C++23起标准库提供std::add_overflow安全检测整数加法溢出,定义于,支持有符号/无符号类型;C++11至C++17不支持,需C++23兼容编译器(如GCC 13+、Clang 16+)。

c++中如何判断整数加法是否产生溢出_c++11标准数值安全检查【详解】

std::add_overflow 判断加法是否溢出(C++23 起)

标准库直到 C++23 才正式提供 std::add_overflow,它在 中定义,可安全检测有符号/无符号整数加法溢出。C++11 / C++14 / C++17 均不支持该函数,强行使用会编译失败。

如果你实际环境是 C++23 或更高(如 GCC 13+、Clang 16+),可直接用:

int a = INT_MAX;
int b = 1;
int result;
if (std::add_overflow(a, b, &result)) {
    // 溢出发生,result 值未定义
    std::cout << "overflow!\n";
}
  • std::add_overflow 返回 bool:true 表示溢出,false 表示成功,且 *result 被写入正确和值
  • 支持所有整型(char, short, int, long long, unsigned int 等),但两个操作数与结果类型必须相同
  • 对无符号类型,它检测的是数学意义上的“超出表示范围”,而非“回绕”——这符合大多数安全场景的预期

兼容 C++11 的手动溢出检测(推荐方案)

C++11 没有标准溢出检测机制,但可通过算术关系判断。核心思路是:对有符号加法 a + b,溢出只可能发生在同号相加时;对无符号,则检查结果是否“变小”。

以下为通用、无 UB、可内联的模板函数(适用于 C++11 起):

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

Frase
Frase

Frase是一款出色的长篇 AI 写作工具,快速创建seo优化的内容。

下载
template 
bool add_would_overflow(T a, T b) {
    static_assert(std::is_integral::value, "");
    if constexpr (std::is_signed::value) {
        if (b > 0) return a > std::numeric_limits::max() - b;
        else return a < std::numeric_limits::min() - b;
    } else {
        return b > std::numeric_limits::max() - a;
    }
}
  • 不执行实际加法,避免未定义行为(UB);所有比较均在合法范围内进行
  • std::numeric_limits::max()::min() 是 C++11 就支持的,无需额外头文件(只需
  • 注意:不能写成 a + b 这类表达式——对有符号类型,a + b 本身已是 UB,编译器可能优化掉整个判断

为什么 __builtin_add_overflow 不是跨平台解法

GCC 和 Clang 提供 __builtin_add_overflow(a, b, &result),它在 C++11 下可用且高效,但它是编译器扩展,不是标准 C++。MSVC 不支持,Intel ICC 支持有限,交叉编译时易出问题。

  • 返回 bool,语义与 std::add_overflow 一致,但名字带双下划线,属于保留标识符,不应在用户命名空间中复用
  • 即使你只用 GCC/Clang,也建议将其封装在条件编译中,作为底层 fallback,而非直接暴露在业务逻辑里
  • 例如:#ifdef __has_builtin(__builtin_add_overflow) 可用于运行时检测,但需注意宏定义时机

别忽略类型提升带来的隐式溢出风险

C++ 整型提升(integer promotion)会让小整型(如 int8_t)先转为 int 再运算,此时溢出检测对象其实是 int,而非原始类型。若你真要检查 int8_t x + int8_t y 是否在 int8_t 范围内溢出,必须显式 cast 回目标类型再判断。

  • 错误示范:int8_t a = 100, b = 100; bool ov = add_would_overflow(a, b); —— 此时 T 推导为 int,检测的是 int 溢出,不是 int8_t
  • 正确做法:add_would_overflow(a, b),强制指定模板参数
  • 更安全:统一用固定宽度类型(如 int32_t)并显式指定模板实参,避免依赖推导

实际项目中,最易被忽略的是「检测类型与运算类型不一致」和「有符号加法误用无符号逻辑」。只要确保模板参数显式、不依赖隐式提升、不执行未检查的中间加法,C++11 完全可以写出健壮的溢出判断逻辑。

相关专题

更多
mysql标识符无效错误怎么解决
mysql标识符无效错误怎么解决

mysql标识符无效错误的解决办法:1、检查标识符是否被其他表或数据库使用;2、检查标识符是否包含特殊字符;3、使用引号包裹标识符;4、使用反引号包裹标识符;5、检查MySQL的配置文件等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

182

2023.12.04

Python标识符有哪些
Python标识符有哪些

Python标识符有变量标识符、函数标识符、类标识符、模块标识符、下划线开头的标识符、双下划线开头、双下划线结尾的标识符、整型标识符、浮点型标识符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

280

2024.02.23

java标识符合集
java标识符合集

本专题整合了java标识符相关内容,想了解更多详细内容,请阅读下面的文章。

254

2025.06.11

c++标识符介绍
c++标识符介绍

本专题整合了c++标识符相关内容,阅读专题下面的文章了解更多详细内容。

121

2025.08.07

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

540

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

53

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

19

2026.01.20

热门下载

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

精品课程

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

共94课时 | 7.1万人学习

C 教程
C 教程

共75课时 | 4.1万人学习

C++教程
C++教程

共115课时 | 12.9万人学习

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

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