0

0

c++如何使用 sanitizers 发现未定义行为_c++ UBSan使用教程【调试】

穿越時空

穿越時空

发布时间:2025-12-13 08:41:53

|

378人浏览过

|

来源于php中文网

原创

UBSan检测C++未定义行为需编译时加-fsanitize=undefined,运行时直接报错定位;推荐clang++ -fsanitize=undefined -O2 -g -fno-omit-frame-pointer,配合UBSAN_OPTIONS可全量报告,适用于CI和本地开发但不可用于发布。

c++如何使用 sanitizers 发现未定义行为_c++ ubsan使用教程【调试】

用 UBSan(UndefinedBehaviorSanitizer)检测 C++ 中的未定义行为,核心就两点:编译时加 -fsanitize=undefined,运行时观察报错信息。它比静态分析更准,比 Valgrind 更轻量,专治数组越界、整数溢出、空指针解引用、未初始化读取等常见“玄学崩溃”。

编译时启用 UBSan

在 g++ 或 clang++ 编译命令中加入 sanitizer 选项即可:

  • 基础启用:g++ -fsanitize=undefined -O1 -g main.cpp -o main
  • 推荐组合(兼顾性能与检测精度):clang++ -fsanitize=undefined -O2 -g -fno-omit-frame-pointer main.cpp -o main
  • 若想捕获更多类型(如返回未初始化值、不完整类型转换),可扩展:-fsanitize=undefined,return,unreachable,vptr
  • 注意:UBSan 不兼容 -O3 下的部分优化(如循环展开可能掩盖问题),-O2 是较稳妥选择

运行时快速定位问题

程序触发未定义行为时,UBSan 会直接打印带源码位置的错误,例如:

main.cpp:12:15: runtime error: signed integer overflow: 2147483647 + 1 cannot be represented in type 'int'

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

零一万物开放平台
零一万物开放平台

零一万物大模型开放平台

下载
  • 错误信息含文件名、行号、列号和具体原因,无需额外调试器就能快速跳转修复
  • 默认只报第一次错误;如需报告所有(适合批量排查),加环境变量UBSAN_OPTIONS=halt_on_error=0:print_stacktrace=1 ./main
  • print_stacktrace=1 可显示调用,对函数内联或模板代码尤其有用

常见误报与规避技巧

UBSan 对某些合法但“危险”的写法也会警告(尤其涉及底层操作),不必盲目修复,但要理解是否真有问题:

  • 有符号整数溢出:C++ 标准确实未定义,但若逻辑上允许(如哈希计算),可用 std::numeric_limits::max() 检查,或改用 unsigned int
  • 未初始化局部变量读取:UBSan 默认检测,但可能误报(如 union 成员切换使用)。可加 -fsanitize=undefined -fno-sanitize=undefined 局部禁用,或用 __attribute__((no_sanitize("undefined"))) 标记函数
  • 虚函数表(vptr)相关警告:多出现在对象构造/析构期间调用虚函数,属标准未定义行为,应重构避免

集成进日常开发流程

别只在出 bug 时才想起来用——把它变成 CI 或本地构建的一部分:

  • CMake 中开启:set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=undefined -g -O2"),配合 add_compile_options(-fno-omit-frame-pointer)
  • CI 脚本里跑 UBSan 版本的单元测试,失败即阻断合并
  • 本地开发时,用一个 alias 快速切换:alias ubbuild='clang++ -fsanitize=undefined -O2 -g -fno-omit-frame-pointer'
  • 注意:UBSan 生成的二进制不能发布(体积大、性能降约 2–3 倍),仅用于调试和测试

基本上就这些。UBSan 不复杂但容易忽略,一旦养成习惯,很多“偶发崩溃”、“结果不一致”类问题会在编码阶段就被拦住。

相关专题

更多
scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

188

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

291

2023.10.25

c语言union的用法
c语言union的用法

c语言union的用法是一种特殊的数据类型,它允许在相同的内存位置存储不同的数据类型,union的使用可以帮助我们节省内存空间,并且可以方便地在不同的数据类型之间进行转换。使用union时需要注意对应的成员是有效的,并且只能同时访问一个成员。本专题为大家提供union相关的文章、下载、课程内容,供大家免费下载体验。

125

2023.09.27

string转int
string转int

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

358

2023.08.02

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

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

542

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

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

394

2023.07.18

c++空格相关教程合集
c++空格相关教程合集

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

0

2026.01.23

热门下载

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

精品课程

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

共94课时 | 7.4万人学习

C 教程
C 教程

共75课时 | 4.2万人学习

C++教程
C++教程

共115课时 | 13.5万人学习

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

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