UBSan和TSan是C++中用于检测未定义行为和数据竞争的实用工具,通过编译时添加-fsanitize=undefined或-thread启用,配合-g和-O1可精准定位问题。
在C++开发中,内存错误、未定义行为和数据竞争是常见但难以排查的问题。Sanitizers(检测工具)是一组编译时和运行时工具,能帮助开发者快速定位这些问题。其中,Undefined Behavior Sanitizer(UBSan)和Thread Sanitizer(TSan)是两个非常实用的工具。下面详细介绍如何使用它们进行代码诊断。
Sanitizers由编译器支持,主要在Clang和GCC中可用。使用时需在编译和链接阶段添加特定标志。
以Clang或g++为例:
示例编译命令:
clang++ -g -O1 -fsanitize=undefined -fsanitize=thread main.cpp -o mainUBSan用于捕获C++中常见的未定义行为(Undefined Behavior),这类行为在标准中没有规定结果,可能导致程序崩溃或不可预测的行为。
常见可检测的未定义行为包括:
示例代码:
int main() {
}
启用 -fsanitize=undefined 后,运行时会提示类似:
runtime error: left shift of 1 by 31 places cannot be represented in type 'int'可根据提示快速定位并修复问题。
TSan专门用于检测多线程程序中的数据竞争(Data Race)。当多个线程并发访问同一内存地址,且至少有一个是写操作,又无同步机制时,就会触发数据竞争。
TSan通过插桩代码记录内存访问和线程同步事件,运行时开销较大(可能慢2-10倍),但能精准报告竞争点。
示例代码:
#include这段代码中,两个线程同时修改 global 变量,没有加锁。使用 -fsanitize=thread 编译后运行,TSan会输出详细报告,指出哪两处访问导致了竞争,并标注文件名和行号。
修复方法通常是引入互斥锁:
#include修复后,TSan不再报错。
Sanitizers是开发和测试阶段的强大辅助工具,但需注意以下几点:
基本上就这些。合理使用UBSan和TSan,能显著提升C++代码的健壮性和可靠性。