如何查看C++中<优化>变量的值?
How do I view the value of an <optimized out> variable in C++?
我正在使用gdb调试C++程序。
我有这个代码:
int x = floor(sqrt(3));
我想查看x的值。然而,gdb声称x是"<optimized_out>"。如何查看x的值?我应该更改编译器标志吗?
在高优化级别上,编译器可以消除中间值,正如您在这里看到的那样。有许多选项:
- 您可以降低优化级别,使调试器更容易跟踪事情。
-O0
肯定会工作(但会慢得多),-O1
也可能工作得很好 - 您可以添加一些显式打印语句来记录输出值
- 您通常也可以通过使其变为易失性来强制编译器保留此特定值(但请记住,完成后要取消使其变易失性!)。然而,请注意,由于优化代码中的控制流也会发生更改,即使您可以看到变量的值,当您查看有问题的变量时,也可能不完全清楚您在代码中的哪个点
如果不能或不想禁用优化,则可以尝试将变量声明为volatile。这通常足以使编译器在最终代码中保留变量。
或者,在最近的GCC版本中,您可以仅禁用一个函数的优化,例如:
void my_function() __attribute__((optimize(0)))
{
int x = floor(sqrt(3));
}
使用反向调试时,请尝试后退一步,接近变量的定义点
如所示:What dos<优化出的值>gdb中的平均值?通常情况下,在功能内部:
- 在函数开始时,可以观察变量的值
- 然而,在函数的末尾,变量越来越有可能变成
<optimized out>
,因为由于优化,它只存储在寄存器中,而不存储在堆栈上的内存中。因此,当不再需要它时,寄存器可能会被另一个变量重用和覆盖,然后调试元数据会通知GDB这一点
因此,如果你正在使用某种反向调试,如Mozilla rr
,一旦你尝试过一次,你就会一直这样做,那么一个很好的选择是尝试后退一步,接近reverse-finish
+reverse-next
变量的定义/上次使用点,看看你是否能在那里观察到它。
这可以通过What dos<优化出的值>gdb中的平均值?并为我节省了几次时间,尤其是当运行未优化的程序使它花费太长时间才能到达感兴趣的点时(考虑到-O0
生成的程序集效率极低,这不足为奇)。
创建自己的"全局变量",并将优化后的输出变量打印到此全局变量中。确保在调试完成后删除这些由您创建的全局变量!
在Visual Studio中使用带有VisualGDB扩展的C++,我看到类范围的变量在语法上是正确的,但运行时变量检查和悬停文本声称这些值已优化,尽管它们实际上没有。
为了查看值,请在为我解析的快速监视或监视窗口中,在变量名前面加上类名
例如:在myclass
中似乎被优化的myvariable
值可以用myclass::myvariable
查看。
- 如何创建一个CMake变量,除非显式重写,否则使用默认值
- 将成员变量添加到共享库中的类中,不会破坏二进制兼容性吗
- 将数组的地址分配给变量并删除
- 为"adjacent"变量赋值时出现问题
- enum是C++中的宏变量还是整数变量
- 在全局变量中保存类的实例以重新创建类(创建"backup")
- 用C++中的一个变量定义一个常量
- 具有奇怪重复模板模式的派生类中的成员变量已损坏
- 你能重载对象变量名本身返回的内容吗
- 内置函数可查看CPP中的成员变量
- 是否可以初始化不可复制类型的成员变量(或基类)
- 尝试通过多个向量访问变量时,向量下标超出范围
- 试图让变量检查数组中的某些内容
- Cpp-Tuple使用带有变量的get
- 将包含C样式数组的对象初始化为成员变量(C++)
- 当vector是tje全局变量时,c++中vector的内存管理
- 通过多个头文件使用常量变量
- std::threads可以从Windows DLL中的全局变量创建/销毁吗?
- 执行函数时导致崩溃的变量
- ``这个''不能用this-&gt;指针变量