鱼C论坛

 找回密码
 立即注册
查看: 1037|回复: 3

[已解决]s1e6舍罕王算法用double定义变量时所出现的问题

[复制链接]
发表于 2020-8-22 11:27:59 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
#include <stdio.h> //自写用double定义变量  结果与小甲鱼答案给出的粒数多1
#include <math.h>

int main()
{
        int i;
        double result1 = 0;
        double result2 = 0;

        for (i = 0; i < 64; i++)
        {
                result1 += pow(2, i);
        }
        printf("舍罕王应该给予达依尔%.0f粒麦子!\n", result1);
        result2 = result1 / 25000;
        printf("如果每25000粒麦子为1kg,那么应该给%.0fkg麦子!\n", result2);

        return 0;
}
结果如下:
1.png 为什么会仅仅多出来这1粒?我实在是不明白,请广大鱼友赐教
还有同样的代码放入codeblocks中运行的时候结果会是这样:
2.png
请问这是为什么呢?
还有就是小甲鱼说他所的出的麦子数是不对的,可是我用计算机算了2^64-1结果与他给出的一致,这又是为什么呢?
希望广大鱼友能够解答我的问题,谢谢大家!

[img]file:///C:\Users\zzh\AppData\Roaming\Tencent\Users\710583435\TIM\WinTemp\RichOle\`EEK3[IGY2R_WZ{VJK9HMHN.png[/img]
最佳答案
2020-8-22 12:05:07
本帖最后由 sunrise085 于 2020-8-22 12:06 编辑

这是double运算精度问题。
你可以看看下面的结果,当i为53的时候,出现精度错误。
#include <stdio.h> //自写用double定义变量  结果与小甲鱼答案给出的粒数多1
#include <math.h>

int main()
{
    int i;
    double result1 = 0;
    double result2 = 0;
    double temp=1.0;
    double a=9007199254740991,b=9007199254740992;//这里是精度问题导致出错的那个数值。
    printf("%.0f\n",a+b);

    for (i = 0; i < 64; i++)
    {
        temp=pow(2, i);
        result1 += temp;
        printf("%d,%.0f,%.0f\n",i,temp,result1);
    }
    printf("舍罕王应该给予达依尔%.0f粒麦子!\n", result1);
    result2 = result1 / 25000;
    printf("如果每25000粒麦子为1kg,那么应该给%.0fkg麦子!\n", result2);

    return 0;
}

所以不应该用double,应该用unsigned long long类型数据
#include <stdio.h> 

int main()
{
    int i;
    unsigned long long result1 = 0;
    unsigned long long result2 = 0;
    unsigned long long temp=1;
    unsigned long long a=9007199254740991,b=9007199254740992;
    printf("%lld\n",a+b);

    for (i = 0; i < 64; i++)
    {
        result1 += temp;
        printf("%d,%llu,%llu\n",i,temp,result1);
        temp*=2;
    }
    printf("舍罕王应该给予达依尔%llu粒麦子!\n", result1);
    result2 = result1 / 25000;
    printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!\n", result2);

    return 0;
}
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2020-8-22 12:05:07 | 显示全部楼层    本楼为最佳答案   
本帖最后由 sunrise085 于 2020-8-22 12:06 编辑

这是double运算精度问题。
你可以看看下面的结果,当i为53的时候,出现精度错误。
#include <stdio.h> //自写用double定义变量  结果与小甲鱼答案给出的粒数多1
#include <math.h>

int main()
{
    int i;
    double result1 = 0;
    double result2 = 0;
    double temp=1.0;
    double a=9007199254740991,b=9007199254740992;//这里是精度问题导致出错的那个数值。
    printf("%.0f\n",a+b);

    for (i = 0; i < 64; i++)
    {
        temp=pow(2, i);
        result1 += temp;
        printf("%d,%.0f,%.0f\n",i,temp,result1);
    }
    printf("舍罕王应该给予达依尔%.0f粒麦子!\n", result1);
    result2 = result1 / 25000;
    printf("如果每25000粒麦子为1kg,那么应该给%.0fkg麦子!\n", result2);

    return 0;
}

所以不应该用double,应该用unsigned long long类型数据
#include <stdio.h> 

int main()
{
    int i;
    unsigned long long result1 = 0;
    unsigned long long result2 = 0;
    unsigned long long temp=1;
    unsigned long long a=9007199254740991,b=9007199254740992;
    printf("%lld\n",a+b);

    for (i = 0; i < 64; i++)
    {
        result1 += temp;
        printf("%d,%llu,%llu\n",i,temp,result1);
        temp*=2;
    }
    printf("舍罕王应该给予达依尔%llu粒麦子!\n", result1);
    result2 = result1 / 25000;
    printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!\n", result2);

    return 0;
}
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-8-22 12:27:43 | 显示全部楼层
sunrise085 发表于 2020-8-22 12:05
这是double运算精度问题。
你可以看看下面的结果,当i为53的时候,出现精度错误。

明白了!感谢指教!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-8-23 13:15:33 | 显示全部楼层
当然,你也可以这样:

result1 += (unsigned long long)pow(2, i); //把出问题的变量给强制转换为无符号的“长长型”变量。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-1-13 07:34

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表