c - lldb 和 C 代码对 pow() 给出不同的结果

标签 c debugging int lldb

我有一个变量,Npart,它是一个 int 并初始化为 64。下面是我的代码(test.c):

#include <math.h>
#include <stdio.h>

int Npart, N;

int main(){

Npart = 64;

N = (int) (pow(Npart/1., (1.0/3.0)));
printf("%d %d\n",Npart, N);

return 0;
};

打印出64 3,可能是由于数字精度问题。我编译如下:

gcc -g3 test.c -o test.x

如果我尝试使用 lldb 进行调试,我尝试计算该值并在命令提示符中打印它,则会发生以下情况:

$ lldb ./test.x
(lldb) target create "./test.x"
Current executable set to './test.x' (x86_64).
(lldb) breakpoint set --file test.c --line 1
Breakpoint 1: where = test.x`main + 44 at test.c:8, address = 0x0000000100000f0c
(lldb) r
Process 20532 launched: './test.x' (x86_64)
Process 20532 stopped
* thread #1: tid = 0x5279e0, 0x0000000100000f0c test.x`main + 44 at test.c:8, queue = 'com.apple.main-thread', stop reason = breakpoint 1.1
frame #0: 0x0000000100000f0c test.x`main + 44 at test.c:8
   5    
   6    int main(){
   7    
-> 8    Npart = 64;
   9    
   10   N = (int) (pow(Npart/1., (1.0/3.0)));
   11   printf("%d %d\n",Npart, N);
(lldb) n
Process 20532 stopped
* thread #1: tid = 0x5279e0, 0x0000000100000f12 test.x`main + 50 at test.c:10, queue = 'com.apple.main-thread', stop reason = step over
frame #0: 0x0000000100000f12 test.x`main + 50 at test.c:10
   7    
   8    Npart = 64;
   9    
-> 10   N = (int) (pow(Npart/1., (1.0/3.0)));
   11   printf("%d %d\n",Npart, N);
   12   
   13   return 0;
(lldb) n
Process 20532 stopped
* thread #1: tid = 0x5279e0, 0x0000000100000f4a test.x`main + 106 at test.c:11, queue = 'com.apple.main-thread', stop reason = step over
    frame #0: 0x0000000100000f4a test.x`main + 106 at test.c:11
   8    Npart = 64;
   9    
   10   N = (int) (pow(Npart/1., (1.0/3.0)));
-> 11   printf("%d %d\n",Npart, N);
   12   
   13   return 0;
   14   };
(lldb) print Npart
(int) $0 = 64
(lldb) print (int)(pow(Npart/1.,(1.0/3.0)))
warning: could not load any Objective-C class information. This will significantly reduce the quality of type information available.
(int) $1 = 0
(lldb) print (int)(pow(64,1.0/3.0))
(int) $2 = 0

为什么 lldb 给出不同的结果?

编辑:澄清了问题并提供了一个最小的可验证示例。

最佳答案

您的代码计算 64 的立方根,应该是 4。

C 代码通过降低返回值将其转换为整数。 pow 通常以某种泰勒多项式或类似的形式实现 - 这在数值上往往不准确。您计算机上的结果似乎略小于 4.0,当转换为 int 时,该结果会被截断 - 解决方案是使用例如 lround 首先改为:

N = lround(pow(Npart/1., (1.0/3.0)));
<小时/>

对于lldb,关键是文本:

error: 'pow' has unknown return type; cast the call to its declared return type

即它知道函数的返回类型,即原型(prototype)。 pow 声明为

double pow(double x, double y);

但是由于 lldb 关于返回类型的唯一提示是您提供的转换,因此 lldb 认为原型(prototype)是

int pow(int x, double y);

这将导致未定义的行为 - 实际上,lldb 认为返回值应该是来自 EAX 寄存器的 int,因此 0 code> 已打印,但实际返回值位于某个浮点/SIMD 寄存器中。同样,由于参数的类型也不知道,因此您不能传入 int

因此我猜你会在调试器中获得正确的值

print (double)(pow(64.0, 1.0/3.0))

关于c - lldb 和 C 代码对 pow() 给出不同的结果,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45267980/

相关文章:

c - 如何在 C 中将随机大写字母数组填充为函数?

c - c程序中如何从任意目录加载文件

c - 旅行推销员的错误代码,即寻找具有最小图表成本的循环

ios - Xcode:禁用 cocoa 断点

mysql - MYSQL 中 int(11) 数据类型面临的问题

c - C语言读取用户输入

c - 全局变量数组指针C编程

在调用堆栈为空的 Visual Studio 中调试小型转储

go - Go 中的 int 数据类型

java - 如何接受和输出数组中的 double 和整数?