朋友们,我想请教下C语言中1.0/3*3用%f输出结果居然是1.000000,不该是0.999999吗?
#include<stdio.h>int main()
{
printf("%f\n",1.0/3*3);//输出结果为1.000000
printf("%f",0.333333*3);//输出结果为0.999999
}
广大朋友们,我想知道这是为什么? 我推测下,不一定是对的。
printf里是从右往左的顺序读取,按优先级计算。/和*是同一个优先级,1.0/3*3 = 1.0*(1/3)*3,后面的先算,就是1,1.0*1=1{:10_245:} 丶忘却的年少o 发表于 2017-9-27 11:52
我推测下,不一定是对的。
printf里是从右往左的顺序读取,按优先级计算。/和*是同一个优先级,1.0/3*3 =...
其实我也不知道,感觉好奇怪啊!
还会自动四舍五入的 primer123 发表于 2017-9-27 18:21
其实我也不知道,感觉好奇怪啊!
还会自动四舍五入的
肯定不是四舍五入,只会舍弃。我记得小甲鱼老师好像提到过,从右往左,可能是记错了不是在printf里。 它输出的是第一个数字。。。 我试了一下,
printf("%f\n",(1.33333/3)*3);
printf("%f",0.3333*3);
输出的是1.33333,0.9999
它不支持运算{:10_243:} 怎么说呢,printf是有读取顺序不过是要执行两个以上的语句就有从右往左的读取顺序。
我实验了一下
printf("%f, %f\n", (1.0/3)*3, 0.3333333*3);//从右到左
printf("%.3f, %.3f\n", 3.1415, 3.1414);//从右到左
printf("%f\n", 1.1111115);
printf("%.9f\n", 1.1111115);
}
第一个printf分别是1.000000和1.000000
第二个:3.142和3.141
第三个:1.111112
第四个:1.11111500
我发现printf中
1.%f不加说明(如%.3f)默认保留小数点后6位
2.如果超出位数,会按超出的第一位的数进行四舍五入
3.你那个1.0/3*3我在加了括号 (1.0/3)*3值还是不变,不是因为读取顺序,是因为1.0/3是无限循环0.3333........乘3后是0.9999999............第七位是9四舍五入就变成1.000000(六位)了
鱼c开挂帐号 发表于 2017-9-27 20:45
我试了一下,
printf("%f\n",(1.33333/3)*3);
printf("%f",0.3333*3);
你说有没可能是浮点数的精确的问题啊?{:10_266:} boot 发表于 2017-9-27 22:14
怎么说呢,printf是有读取顺序不过是要执行两个以上的语句就有从右往左的读取顺序。
我实验了一下
看了你的例子我现在懂了,不过现在又出现了一个新的问题!
%m.nf,m是输出总的列数(包含小数点),n是表示占n位小数;
printf("%7.2f\n",12.35)结果是12.35(12前面有两个空格),这个没问题;但是你的第二个代码:
printf("%.3f, %.3f\n", 3.1415, 3.1414);为什么m是零还可以输出3.142和3.141??
我想请教下,谢谢!{:10_280:}
primer123 发表于 2017-9-28 11:12
看了你的例子我现在懂了,不过现在又出现了一个新的问题!
%m.nf,m是输出总的列数(包含小数点),n是表 ...
你想说的是浮点型有效位数的问题,对吧?
printf("%f\n", 12345678912345671.);
printf("%f\n", 12345678912345672.);
printf("%f\n", 123456789123456789.);
printf("%.17f\n",1.23456789123456789);
vc++中%f的有效位数应该为16位(第一个和第二个printf可以看出)。
你那个算出来的0.3333333.........乘3变为0.9999999......(小数点不加说明默认是6位,溢位了就四舍五入,这个问题和精度无关)。
第三个和第四个printf你就可以看出来这个的确是和精度有关了,都是18位(由于默认小数点后6位,我给加长了)。
那个%m.nf这里的“m”表示指定数据的最小宽度。若实际位数多于定义的宽度,则按实际位数输出。
这也就是为什么我m是”0“但还是可以正确输出。 其实从数学的角度而言。0.99999999999999就等于1
页:
[1]