鱼C论坛

 找回密码
 立即注册
查看: 1400|回复: 8

[已解决]有大佬帮忙看看吗?

[复制链接]
发表于 2021-10-14 22:25:26 | 显示全部楼层 |阅读模式

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

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

x
本帖最后由 戌砚 于 2021-10-14 22:54 编辑

为什么结果为0啊?
  1. #include<stdio.h>
  2. #include<math.h>
  3. int main(){
  4.    unsigned long long a=0;
  5.    int i;
  6.     for(i=0;i<64;i++){
  7.         a+=pow(2,i);
  8.     }
  9.     unsigned long long b=a/25000;
  10.     printf("舍罕王应该给予达依尔%llu粒麦子!\n",a);
  11.     printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!",b);
  12.     return 0;
  13. }
复制代码


为什么a在之前就已经定义为unsigned类型,而还是需要强制转换一下,而参考答案中加入了一个sum变量就没有问题
  1. #include<stdio.h>
  2. #include<math.h>
  3. int main(){
  4.    unsigned long long a=0;
  5.    int i;
  6.     for(i=0;i<64;i++){
  7.         a+=(unsigned long long)pow(2,i);
  8.     }
  9.     unsigned long long b=a/25000;
  10.     printf("舍罕王应该给予达依尔%llu粒麦子!\n",a);
  11.     printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!",b);
  12.     return 0;
  13. }
复制代码


参考答案
  1. #include <stdio.h>
  2. #include <math.h>

  3. int main()
  4. {
  5.         unsigned long long sum = 0;
  6.         unsigned long long temp;
  7.         unsigned long long weight;
  8.         int i;

  9.         for (i=0; i < 64; i++)
  10.         {
  11.                 temp = pow(2, i);
  12.                 sum = sum + temp;
  13.         }

  14.         weight = sum / 25000;

  15.         printf("舍罕王应该给予达依尔%llu粒麦子!\n", sum);
  16.         printf("如果每25000粒麦子为1kg,那么应该给%llu公斤麦子!\n", weight);

  17.         return 0;
  18. }
复制代码
最佳答案
2021-10-15 12:12:21
弄清楚了
如果你手动转换到 unsigned long long,那就是 unsigned long long + unsigned long long
是整数运算,这不会有问题
但是如果你没有手动转换到 unsigned long long,那编译器就自动转换到 double,是把 unsigned long long 自动转换到 double
进行浮点运算,浮点运算是有精度损失的
看下面的程序
如果是整数运算,在最后一次加法之前
temp: 9223372036854775808
sum: 9223372036854775807
temp + sum 的结果是
s: 18446744073709551615

但是,如果是浮点运算
temp: 9223372036854775808.000000
a: 9223372036854775808
s: 0

9223372036854775808 + 9223372036854775808 = 18446744073709551616
18446744073709551616 已经超过了 unsigned long long 能表示的范围,加法溢出了

对于当前这个程序来说,浮点运算的精度是从 i == 53 开始 出现问题的


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

  3. int main()
  4. {
  5.         unsigned long long sum = 0;
  6.         unsigned long long temp;
  7.         unsigned long long weight;
  8.         int i;

  9.         for (i=0; i < 64; i++)
  10.         {
  11.                 temp = pow(2, i);
  12.                 if(i == 63) {
  13.                         printf("temp: %llu\n", temp);
  14.                         printf("sum: %llu\n", sum);
  15.                         unsigned long long s = sum + temp;
  16.                         printf("s: %llu\n", s);
  17.                 }
  18.                 sum = sum + temp;
  19.         }

  20.         weight = sum / 25000;

  21.         printf("舍罕王应该给予达依尔%llu粒麦子!\n", sum);
  22.         printf("如果每25000粒麦子为1kg,那么应该给%llu公斤麦子!\n", weight);

  23.         return 0;
  24. }

  25. /*
  26. temp: 9223372036854775808
  27. sum: 9223372036854775807
  28. s: 18446744073709551615
  29. 舍罕王应该给予达依尔18446744073709551615粒麦子!
  30. 如果每25000粒麦子为1kg,那么应该给737869762948382公斤麦子!
  31. */
复制代码

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

  3. int main() {
  4.     unsigned long long a = 0;
  5.     int i;
  6.     for(i = 0; i < 64; i++) {
  7.         double temp = pow(2, i);
  8.         if(i == 63) {
  9.             printf("temp: %lf\n", temp);
  10.             printf("a: %llu\n", a);
  11.             unsigned long long s = a + temp;
  12.             printf("s: %llu\n", s);
  13.         }
  14.         a += temp;
  15.     }
  16.     unsigned long long b = a / 25000;
  17.     printf("舍罕王应该给予达依尔%llu粒麦子!\n", a);
  18.     printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!\n", b);
  19.     return 0;
  20. }

  21. /*
  22. temp: 9223372036854775808.000000
  23. a: 9223372036854775808
  24. s: 0
  25. 舍罕王应该给予达依尔0粒麦子!
  26. 如果每25000粒麦子为1kg,那么应该给0kg麦子!
  27. */
复制代码

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

  3. int main()
  4. {
  5.         unsigned long long sum = 0;
  6.         unsigned long long temp;
  7.         unsigned long long weight;
  8.         int i;

  9.         for (i=0; i < 64; i++)
  10.         {
  11.                 temp = pow(2, i);
  12.                 if(i == 52) {
  13.                         printf("temp: %llu\n", temp);
  14.                         printf("sum: %llu\n", sum);
  15.                         unsigned long long s = sum + temp;
  16.                         printf("s: %llu\n", s);
  17.                 }
  18.                 sum = sum + temp;
  19.         }

  20.         weight = sum / 25000;

  21.         printf("舍罕王应该给予达依尔%llu粒麦子!\n", sum);
  22.         printf("如果每25000粒麦子为1kg,那么应该给%llu公斤麦子!\n", weight);

  23.         return 0;
  24. }

  25. /*
  26. temp: 4503599627370496
  27. sum: 4503599627370495
  28. s: 9007199254740991
  29. 舍罕王应该给予达依尔18446744073709551615粒麦子!
  30. 如果每25000粒麦子为1kg,那么应该给737869762948382公斤麦子!
  31. */
复制代码

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

  3. int main() {
  4.     unsigned long long a = 0;
  5.     int i;
  6.     for(i = 0; i < 64; i++) {
  7.         double temp = pow(2, i);
  8.         if(i == 52) {
  9.             printf("temp: %lf\n", temp);
  10.             printf("a: %llu\n", a);
  11.             unsigned long long s = a + temp;
  12.             printf("s: %llu\n", s);
  13.         }
  14.         a += temp;
  15.     }
  16.     unsigned long long b = a / 25000;
  17.     printf("舍罕王应该给予达依尔%llu粒麦子!\n", a);
  18.     printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!\n", b);
  19.     return 0;
  20. }

  21. /*
  22. temp: 4503599627370496.000000
  23. a: 4503599627370495
  24. s: 9007199254740991
  25. 舍罕王应该给予达依尔0粒麦子!
  26. 如果每25000粒麦子为1kg,那么应该给0kg麦子!

  27. */
复制代码

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

  3. int main()
  4. {
  5.         unsigned long long sum = 0;
  6.         unsigned long long temp;
  7.         unsigned long long weight;
  8.         int i;

  9.         for (i=0; i < 64; i++)
  10.         {
  11.                 temp = pow(2, i);
  12.                 if(i == 53) {
  13.                         printf("temp: %llu\n", temp);
  14.                         printf("sum: %llu\n", sum);
  15.                         unsigned long long s = sum + temp;
  16.                         printf("s: %llu\n", s);
  17.                 }
  18.                 sum = sum + temp;
  19.         }

  20.         weight = sum / 25000;

  21.         printf("舍罕王应该给予达依尔%llu粒麦子!\n", sum);
  22.         printf("如果每25000粒麦子为1kg,那么应该给%llu公斤麦子!\n", weight);

  23.         return 0;
  24. }

  25. /*
  26. temp: 9007199254740992
  27. sum: 9007199254740991
  28. s: 18014398509481983
  29. 舍罕王应该给予达依尔18446744073709551615粒麦子!
  30. 如果每25000粒麦子为1kg,那么应该给737869762948382公斤麦子!
  31. */
复制代码

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

  3. int main() {
  4.     unsigned long long a = 0;
  5.     int i;
  6.     for(i = 0; i < 64; i++) {
  7.         double temp = pow(2, i);
  8.         if(i == 53) {
  9.             printf("temp: %lf\n", temp);
  10.             printf("a: %llu\n", a);
  11.             unsigned long long s = a + temp;
  12.             printf("s: %llu\n", s);
  13.         }
  14.         a += temp;
  15.     }
  16.     unsigned long long b = a / 25000;
  17.     printf("舍罕王应该给予达依尔%llu粒麦子!\n", a);
  18.     printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!\n", b);
  19.     return 0;
  20. }

  21. /*
  22. temp: 9007199254740992.000000
  23. a: 9007199254740991
  24. s: 18014398509481984
  25. 舍罕王应该给予达依尔0粒麦子!
  26. 如果每25000粒麦子为1kg,那么应该给0kg麦子!
  27. */
复制代码

6_3.2.png
6_3.3.png
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2021-10-14 22:28:23 | 显示全部楼层
unsigned long long a = 0;
改成这样试试

还有,提问题要贴代码,可以复制粘贴的那种
你想贴图片完全可以,但是可以复制粘贴的代码必须贴
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-10-14 22:29:22 | 显示全部楼层
人造人 发表于 2021-10-14 22:28
unsigned long long a = 0;
改成这样试试

好滴
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-10-14 22:32:35 | 显示全部楼层
人造人 发表于 2021-10-14 22:28
unsigned long long a = 0;
改成这样试试

结果还是0,没有变啊
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2021-10-14 22:39:01 | 显示全部楼层
戌砚 发表于 2021-10-14 22:32
结果还是0,没有变啊

强制转换一下类型

  1. #include<stdio.h>
  2. #include<math.h>
  3. int main(){
  4.     unsigned long long a=0;
  5.     int i;
  6.     for(i=0;i<64;i++){
  7.         a += (unsigned long long)pow(2,i);
  8.     }
  9.     unsigned long long b=a/25000;
  10.     printf("舍罕王应该给予达依尔%llu粒麦子!\n",a);
  11.     printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!",b);
  12.     return 0;
  13. }
复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-10-14 22:41:04 | 显示全部楼层
人造人 发表于 2021-10-14 22:39
强制转换一下类型

可以了,这是为什么呀?
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2021-10-14 22:45:34 | 显示全部楼层
戌砚 发表于 2021-10-14 22:41
可以了,这是为什么呀?

因为类型不一样
我认为编译器应该自动转换类型的,但是这里编译器确实没有转换,要自己手动转换
我也不清楚为什么会这样
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-10-14 22:48:09 | 显示全部楼层
人造人 发表于 2021-10-14 22:45
因为类型不一样
我认为编译器应该自动转换类型的,但是这里编译器确实没有转换,要自己手动转换
我也不 ...

好滴,谢谢
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2021-10-15 12:12:21 | 显示全部楼层    本楼为最佳答案   
弄清楚了
如果你手动转换到 unsigned long long,那就是 unsigned long long + unsigned long long
是整数运算,这不会有问题
但是如果你没有手动转换到 unsigned long long,那编译器就自动转换到 double,是把 unsigned long long 自动转换到 double
进行浮点运算,浮点运算是有精度损失的
看下面的程序
如果是整数运算,在最后一次加法之前
temp: 9223372036854775808
sum: 9223372036854775807
temp + sum 的结果是
s: 18446744073709551615

但是,如果是浮点运算
temp: 9223372036854775808.000000
a: 9223372036854775808
s: 0

9223372036854775808 + 9223372036854775808 = 18446744073709551616
18446744073709551616 已经超过了 unsigned long long 能表示的范围,加法溢出了

对于当前这个程序来说,浮点运算的精度是从 i == 53 开始 出现问题的


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

  3. int main()
  4. {
  5.         unsigned long long sum = 0;
  6.         unsigned long long temp;
  7.         unsigned long long weight;
  8.         int i;

  9.         for (i=0; i < 64; i++)
  10.         {
  11.                 temp = pow(2, i);
  12.                 if(i == 63) {
  13.                         printf("temp: %llu\n", temp);
  14.                         printf("sum: %llu\n", sum);
  15.                         unsigned long long s = sum + temp;
  16.                         printf("s: %llu\n", s);
  17.                 }
  18.                 sum = sum + temp;
  19.         }

  20.         weight = sum / 25000;

  21.         printf("舍罕王应该给予达依尔%llu粒麦子!\n", sum);
  22.         printf("如果每25000粒麦子为1kg,那么应该给%llu公斤麦子!\n", weight);

  23.         return 0;
  24. }

  25. /*
  26. temp: 9223372036854775808
  27. sum: 9223372036854775807
  28. s: 18446744073709551615
  29. 舍罕王应该给予达依尔18446744073709551615粒麦子!
  30. 如果每25000粒麦子为1kg,那么应该给737869762948382公斤麦子!
  31. */
复制代码

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

  3. int main() {
  4.     unsigned long long a = 0;
  5.     int i;
  6.     for(i = 0; i < 64; i++) {
  7.         double temp = pow(2, i);
  8.         if(i == 63) {
  9.             printf("temp: %lf\n", temp);
  10.             printf("a: %llu\n", a);
  11.             unsigned long long s = a + temp;
  12.             printf("s: %llu\n", s);
  13.         }
  14.         a += temp;
  15.     }
  16.     unsigned long long b = a / 25000;
  17.     printf("舍罕王应该给予达依尔%llu粒麦子!\n", a);
  18.     printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!\n", b);
  19.     return 0;
  20. }

  21. /*
  22. temp: 9223372036854775808.000000
  23. a: 9223372036854775808
  24. s: 0
  25. 舍罕王应该给予达依尔0粒麦子!
  26. 如果每25000粒麦子为1kg,那么应该给0kg麦子!
  27. */
复制代码

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

  3. int main()
  4. {
  5.         unsigned long long sum = 0;
  6.         unsigned long long temp;
  7.         unsigned long long weight;
  8.         int i;

  9.         for (i=0; i < 64; i++)
  10.         {
  11.                 temp = pow(2, i);
  12.                 if(i == 52) {
  13.                         printf("temp: %llu\n", temp);
  14.                         printf("sum: %llu\n", sum);
  15.                         unsigned long long s = sum + temp;
  16.                         printf("s: %llu\n", s);
  17.                 }
  18.                 sum = sum + temp;
  19.         }

  20.         weight = sum / 25000;

  21.         printf("舍罕王应该给予达依尔%llu粒麦子!\n", sum);
  22.         printf("如果每25000粒麦子为1kg,那么应该给%llu公斤麦子!\n", weight);

  23.         return 0;
  24. }

  25. /*
  26. temp: 4503599627370496
  27. sum: 4503599627370495
  28. s: 9007199254740991
  29. 舍罕王应该给予达依尔18446744073709551615粒麦子!
  30. 如果每25000粒麦子为1kg,那么应该给737869762948382公斤麦子!
  31. */
复制代码

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

  3. int main() {
  4.     unsigned long long a = 0;
  5.     int i;
  6.     for(i = 0; i < 64; i++) {
  7.         double temp = pow(2, i);
  8.         if(i == 52) {
  9.             printf("temp: %lf\n", temp);
  10.             printf("a: %llu\n", a);
  11.             unsigned long long s = a + temp;
  12.             printf("s: %llu\n", s);
  13.         }
  14.         a += temp;
  15.     }
  16.     unsigned long long b = a / 25000;
  17.     printf("舍罕王应该给予达依尔%llu粒麦子!\n", a);
  18.     printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!\n", b);
  19.     return 0;
  20. }

  21. /*
  22. temp: 4503599627370496.000000
  23. a: 4503599627370495
  24. s: 9007199254740991
  25. 舍罕王应该给予达依尔0粒麦子!
  26. 如果每25000粒麦子为1kg,那么应该给0kg麦子!

  27. */
复制代码

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

  3. int main()
  4. {
  5.         unsigned long long sum = 0;
  6.         unsigned long long temp;
  7.         unsigned long long weight;
  8.         int i;

  9.         for (i=0; i < 64; i++)
  10.         {
  11.                 temp = pow(2, i);
  12.                 if(i == 53) {
  13.                         printf("temp: %llu\n", temp);
  14.                         printf("sum: %llu\n", sum);
  15.                         unsigned long long s = sum + temp;
  16.                         printf("s: %llu\n", s);
  17.                 }
  18.                 sum = sum + temp;
  19.         }

  20.         weight = sum / 25000;

  21.         printf("舍罕王应该给予达依尔%llu粒麦子!\n", sum);
  22.         printf("如果每25000粒麦子为1kg,那么应该给%llu公斤麦子!\n", weight);

  23.         return 0;
  24. }

  25. /*
  26. temp: 9007199254740992
  27. sum: 9007199254740991
  28. s: 18014398509481983
  29. 舍罕王应该给予达依尔18446744073709551615粒麦子!
  30. 如果每25000粒麦子为1kg,那么应该给737869762948382公斤麦子!
  31. */
复制代码

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

  3. int main() {
  4.     unsigned long long a = 0;
  5.     int i;
  6.     for(i = 0; i < 64; i++) {
  7.         double temp = pow(2, i);
  8.         if(i == 53) {
  9.             printf("temp: %lf\n", temp);
  10.             printf("a: %llu\n", a);
  11.             unsigned long long s = a + temp;
  12.             printf("s: %llu\n", s);
  13.         }
  14.         a += temp;
  15.     }
  16.     unsigned long long b = a / 25000;
  17.     printf("舍罕王应该给予达依尔%llu粒麦子!\n", a);
  18.     printf("如果每25000粒麦子为1kg,那么应该给%llukg麦子!\n", b);
  19.     return 0;
  20. }

  21. /*
  22. temp: 9007199254740992.000000
  23. a: 9007199254740991
  24. s: 18014398509481984
  25. 舍罕王应该给予达依尔0粒麦子!
  26. 如果每25000粒麦子为1kg,那么应该给0kg麦子!
  27. */
复制代码

小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

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

本版积分规则

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

GMT+8, 2025-4-26 03:10

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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