输入正整数a,b,c,输出a/b的小数形式,精确到小数点后c位。a,b≤10⁶,c≤100。
输入包含多组数据,结束标记为a=b=c=0。
样例输入:
1 6 4
0 0 0
样例输出:
Case 1: 0.1667
渣渣智本以为这道题很简单,然而。。。
如果很多很多很多小数点后的位数怎么办,也就是c,然后就找了很多大佬写的,真的不愧是大佬,太强了。
一开始连怎么输出多少位都不知道后来发现了一种东西叫“.*”。
有好多人说可以直接用C++的setprecision©解决,但是渣渣智不懂c++,只能望而却步,唉……但是还有bug就是一旦c大于16以后,输出的浮点数后面全为0,原因是double的有效精度只有16位。如果非要改成C语言的,可以简单的以下方法;(渣渣智简单方法也不会……)
#include <stdio.h>
int
main(void)
{
int a, b, c;
scanf("%d%d%d", &a, &b, &c);
printf("%.*f\n", c, (double)a/b);
return 0;
}
这样就会如上所说16位之后就会输出0。
因此必须手动模拟除法的运算,不停“乘10取余”法即可解决。先输出小数点前的数字,由于测试样例中最后一位需要四舍五入,因此先输出前c-1位,最后第c位根据c+1位判断是否舍入。
然而还会有有许多坑点:
1.正整数a,b,c
2.末位要进位
3.像0.9999保留三位小数这样
由于进位会影响之前的一连串数字,于是,就将边循环边输出的方式替换成用数组存储每一位的数字。
最终找到大佬的代码,真正的无bug:
来自 Capus_Burki的 [https://blog.csdn.net/Capus_Burki/article/details/79226817](https://blog.csdn.net/Capus_Burki/article/details/79226817)
#include<cstdio>
#include<cstdlib>
int main()
{
int a, b, c;
const int Size = 101;//c<=100
int x_arr[Size] = {};//存储商的数组
while(scanf("%d%d%d", &a, &b, &c)){
if(!a && !b && !c)//a,b,c均为0时结束
break;
int x = a / b;//x为商
int y = a % b;//y为余
x_arr[0] = x;//整数部分
for (int i = 1; i <= c; i++)//循环c次
{
if(y == 0)//如果能被整除
{
x_arr[i] = 0;//输出0
}
else//不能被整除
{
y *= 10;//余数乘以十
x = y / b;//余数除以商
y = y % b;//得到新余数
//最后一位小数,需要判断进位
if(c == i)
{
int temp = x;//储存商值
y *= 10;
x = y / b;//下一位的商
//如果进位之后出现类似0.9999的情况
if(x > 4 && temp == 9)
{
x_arr[i] = temp;//x_arr[c] = 9;
for (int j = c; j >= 0; j--) {//从后向前检索
if(x_arr[j] == 9 && j != 0)//到达整数部分时直接+1
x_arr[j] = 0;
else
{
x_arr[j]++;//进位
break;
}
}
}
else if(x > 4)
x_arr[i] = temp + 1;
else
x_arr[i] = temp;
}
//正常情况
else
x_arr[i] = x;
}
}
printf("%d.",x_arr[0]);//先输出整数部分
for (int k = 1; k <= c; k++)
printf("%d",x_arr[k]);//小数部分
printf("\n");
}
return 0;
}