猿问

避免printf()中的尾随零

避免printf()中的尾随零

我一直不知道printf()函数家族的格式说明符。我想要的是能够在小数点之后打印出最大给定数字数的双(或浮点数)。如果我用:

printf("%1.3f", 359.01335);printf("%1.3f", 359.00999);

我得到

359.013359.010

而不是期望的

359.013359.01

有人能帮我吗?


交互式爱情
浏览 470回答 3
3回答

饮歌长啸

这不能用正常的printf格式说明符。你能得到的最接近的是:printf("%.6g",&nbsp;359.013);&nbsp;//&nbsp;359.013printf("%.6g",&nbsp;359.01);&nbsp;&nbsp;//&nbsp;359.01但“6”是共计数值宽度printf("%.6g",&nbsp;3.01357);&nbsp;//&nbsp;3.01357打破它。你能,会,可以做是为了sprintf("%.20g")字符串缓冲区中的数字,然后操作该字符串,使N个字符超过小数点。假设您的数字在变量num中,下面的函数将删除第一个之外的所有N小数,然后去掉尾随的零(如果它们都是零,则取小数点)。char&nbsp;str[50];sprintf&nbsp;(str,"%.20g",num);&nbsp;&nbsp;//&nbsp;Make&nbsp;the&nbsp;number.morphNumericString&nbsp;(str,&nbsp;3);:&nbsp;&nbsp;&nbsp;&nbsp;:void&nbsp;morphNumericString&nbsp;(char&nbsp;*s,&nbsp;int&nbsp;n)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;char&nbsp;*p; &nbsp;&nbsp;&nbsp;&nbsp;int&nbsp;count; &nbsp;&nbsp;&nbsp;&nbsp;p&nbsp;=&nbsp;strchr&nbsp;(s,'.');&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Find&nbsp;decimal&nbsp;point,&nbsp;if&nbsp;any. &nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(p&nbsp;!=&nbsp;NULL)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;count&nbsp;=&nbsp;n;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Adjust&nbsp;for&nbsp;more&nbsp;or&nbsp;less&nbsp;decimals. &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;while&nbsp;(count&nbsp;>=&nbsp;0)&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Maximum&nbsp;decimals&nbsp;allowed. &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;count--; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(*p&nbsp;==&nbsp;'\0')&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;If&nbsp;there's&nbsp;less&nbsp;than&nbsp;desired. &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;break; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;p++;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Next&nbsp;character. &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;} &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;*p--&nbsp;=&nbsp;'\0';&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Truncate&nbsp;string. &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;while&nbsp;(*p&nbsp;==&nbsp;'0')&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Remove&nbsp;trailing&nbsp;zeros. &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;*p--&nbsp;=&nbsp;'\0'; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(*p&nbsp;==&nbsp;'.')&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;If&nbsp;all&nbsp;decimals&nbsp;were&nbsp;zeros,&nbsp;remove&nbsp;".". &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;*p&nbsp;=&nbsp;'\0'; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;} &nbsp;&nbsp;&nbsp;&nbsp;}}如果您对截断方面不满意(这会导致0.12399进0.123而不是把它四舍五入0.124),您可以实际使用printf..您只需要分析手边的数字,就可以动态地创建宽度,然后使用这些数据将数字转换为字符串:#include&nbsp;<stdio.h>void&nbsp;nDecimals&nbsp;(char&nbsp;*s,&nbsp;double&nbsp;d,&nbsp;int&nbsp;n)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;int&nbsp;sz;&nbsp;double&nbsp;d2; &nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Allow&nbsp;for&nbsp;negative. &nbsp;&nbsp;&nbsp;&nbsp;d2&nbsp;=&nbsp;(d&nbsp;>=&nbsp;0)&nbsp;?&nbsp;d&nbsp;:&nbsp;-d; &nbsp;&nbsp;&nbsp;&nbsp;sz&nbsp;=&nbsp;(d&nbsp;>=&nbsp;0)&nbsp;?&nbsp;0&nbsp;:&nbsp;1; &nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Add&nbsp;one&nbsp;for&nbsp;each&nbsp;whole&nbsp;digit&nbsp;(0.xx&nbsp;special&nbsp;case). &nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(d2&nbsp;<&nbsp;1)&nbsp;sz++; &nbsp;&nbsp;&nbsp;&nbsp;while&nbsp;(d2&nbsp;>=&nbsp;1)&nbsp;{&nbsp;d2&nbsp;/=&nbsp;10.0;&nbsp;sz++;&nbsp;} &nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Adjust&nbsp;for&nbsp;decimal&nbsp;point&nbsp;and&nbsp;fractionals. &nbsp;&nbsp;&nbsp;&nbsp;sz&nbsp;+=&nbsp;1&nbsp;+&nbsp;n; &nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Create&nbsp;format&nbsp;string&nbsp;then&nbsp;use&nbsp;it. &nbsp;&nbsp;&nbsp;&nbsp;sprintf&nbsp;(s,&nbsp;"%*.*f",&nbsp;sz,&nbsp;n,&nbsp;d);}int&nbsp;main&nbsp;(void)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;char&nbsp;str[50]; &nbsp;&nbsp;&nbsp;&nbsp;double&nbsp;num[]&nbsp;=&nbsp;{&nbsp;40,&nbsp;359.01335,&nbsp;-359.00999, &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;359.01,&nbsp;3.01357,&nbsp;0.111111111,&nbsp;1.1223344&nbsp;}; &nbsp;&nbsp;&nbsp;&nbsp;for&nbsp;(int&nbsp;i&nbsp;=&nbsp;0;&nbsp;i&nbsp;<&nbsp;sizeof(num)/sizeof(*num);&nbsp;i++)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;nDecimals&nbsp;(str,&nbsp;num[i],&nbsp;3); &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf&nbsp;("%30.20f&nbsp;->&nbsp;%s\n",&nbsp;num[i],&nbsp;str); &nbsp;&nbsp;&nbsp;&nbsp;} &nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;0;}整点nDecimals()在本例中,要正确计算字段宽度,然后根据字段宽度使用格式字符串对数字进行格式化。试驾main()在行动中显示了这一点:&nbsp;&nbsp;40.00000000000000000000&nbsp;->&nbsp;40.000 &nbsp;359.01335000000000263753&nbsp;->&nbsp;359.013-359.00999000000001615263&nbsp;->&nbsp;-359.010 &nbsp;359.00999999999999090505&nbsp;->&nbsp;359.010 &nbsp;&nbsp;&nbsp;3.01357000000000008200&nbsp;->&nbsp;3.014 &nbsp;&nbsp;&nbsp;0.11111111099999999852&nbsp;->&nbsp;0.111 &nbsp;&nbsp;&nbsp;1.12233439999999995429&nbsp;->&nbsp;1.122一旦得到正确的舍入值,就可以再次将其传递给morphNumericString()通过简单的更改来删除尾随零点:nDecimals&nbsp;(str,&nbsp;num[i],&nbsp;3);转入:nDecimals&nbsp;(str,&nbsp;num[i],&nbsp;3);morphNumericString&nbsp;(str,&nbsp;3);(或呼叫)morphNumericString在.的末尾nDecimals但是,在这种情况下,我可能会将这两个函数合并为一个函数),最后您将得到:&nbsp;&nbsp;40.00000000000000000000&nbsp;->&nbsp;40 &nbsp;359.01335000000000263753&nbsp;->&nbsp;359.013-359.00999000000001615263&nbsp;->&nbsp;-359.01 &nbsp;359.00999999999999090505&nbsp;->&nbsp;359.01 &nbsp;&nbsp;&nbsp;3.01357000000000008200&nbsp;->&nbsp;3.014 &nbsp;&nbsp;&nbsp;0.11111111099999999852&nbsp;->&nbsp;0.111 &nbsp;&nbsp;&nbsp;1.12233439999999995429&nbsp;->&nbsp;1.122

萧十郎

要去掉尾随零,您应该使用“%g”格式:float&nbsp;num&nbsp;=&nbsp;1.33;printf("%g",&nbsp;num);&nbsp;//output:&nbsp;1.33在这个问题被澄清了一点之后,抑制零点并不是唯一被问到的问题,但是也需要将输出限制在小数点后的三位。我认为单靠sprintf格式字符串是做不到的。如暗黑指出,需要进行字符串操作。

蛊毒传说

我喜欢R.的回答,稍作改动:float&nbsp;f&nbsp;=&nbsp;1234.56789;printf("%d.%.0f",&nbsp;f,&nbsp;1000*(f-(int)f));“1000”决定了精确度。能量为0.5四舍五入。编辑好的,这个答案被编辑了几次,几年前我失去了我的想法(最初它并没有满足所有的标准)。下面是一个新版本(填充所有标准并正确处理负数):double&nbsp;f&nbsp;=&nbsp;1234.05678900;char&nbsp;s[100];&nbsp;int&nbsp;decimals&nbsp;=&nbsp;10;sprintf(s,"%.*g",&nbsp;decimals,&nbsp;((int)(pow(10,&nbsp;decimals)*(fabs(f)&nbsp;-&nbsp;abs((int)f))&nbsp;+0.5)) /pow(10,decimals));printf("10&nbsp;decimals:&nbsp;%d%s\n",&nbsp;(int)f,&nbsp;s+1);以及测试用例:#import&nbsp;<stdio.h>#import&nbsp;<stdlib.h>#import&nbsp;<math.h>int&nbsp;main(void){ &nbsp;&nbsp;&nbsp;&nbsp;double&nbsp;f&nbsp;=&nbsp;1234.05678900; &nbsp;&nbsp;&nbsp;&nbsp;char&nbsp;s[100]; &nbsp;&nbsp;&nbsp;&nbsp;int&nbsp;decimals; &nbsp;&nbsp;&nbsp;&nbsp;decimals&nbsp;=&nbsp;10; &nbsp;&nbsp;&nbsp;&nbsp;sprintf(s,"%.*g",&nbsp;decimals,&nbsp;((int)(pow(10,&nbsp;decimals)*(fabs(f)&nbsp;-&nbsp;abs((int)f))&nbsp;+0.5))/pow(10,decimals)); &nbsp;&nbsp;&nbsp;&nbsp;printf("10&nbsp;decimals:&nbsp;%d%s\n",&nbsp;(int)f,&nbsp;s+1); &nbsp;&nbsp;&nbsp;&nbsp;decimals&nbsp;=&nbsp;3; &nbsp;&nbsp;&nbsp;&nbsp;sprintf(s,"%.*g",&nbsp;decimals,&nbsp;((int)(pow(10,&nbsp;decimals)*(fabs(f)&nbsp;-&nbsp;abs((int)f))&nbsp;+0.5))/pow(10,decimals)); &nbsp;&nbsp;&nbsp;&nbsp;printf("&nbsp;3&nbsp;decimals:&nbsp;%d%s\n",&nbsp;(int)f,&nbsp;s+1); &nbsp;&nbsp;&nbsp;&nbsp;f&nbsp;=&nbsp;-f; &nbsp;&nbsp;&nbsp;&nbsp;decimals&nbsp;=&nbsp;10; &nbsp;&nbsp;&nbsp;&nbsp;sprintf(s,"%.*g",&nbsp;decimals,&nbsp;((int)(pow(10,&nbsp;decimals)*(fabs(f)&nbsp;-&nbsp;abs((int)f))&nbsp;+0.5))/pow(10,decimals)); &nbsp;&nbsp;&nbsp;&nbsp;printf("&nbsp;negative&nbsp;10:&nbsp;%d%s\n",&nbsp;(int)f,&nbsp;s+1); &nbsp;&nbsp;&nbsp;&nbsp;decimals&nbsp;=&nbsp;3; &nbsp;&nbsp;&nbsp;&nbsp;sprintf(s,"%.*g",&nbsp;decimals,&nbsp;((int)(pow(10,&nbsp;decimals)*(fabs(f)&nbsp;-&nbsp;abs((int)f))&nbsp;+0.5))/pow(10,decimals)); &nbsp;&nbsp;&nbsp;&nbsp;printf("&nbsp;negative&nbsp;&nbsp;3:&nbsp;%d%s\n",&nbsp;(int)f,&nbsp;s+1); &nbsp;&nbsp;&nbsp;&nbsp;decimals&nbsp;=&nbsp;2; &nbsp;&nbsp;&nbsp;&nbsp;f&nbsp;=&nbsp;1.012; &nbsp;&nbsp;&nbsp;&nbsp;sprintf(s,"%.*g",&nbsp;decimals,&nbsp;((int)(pow(10,&nbsp;decimals)*(fabs(f)&nbsp;-&nbsp;abs((int)f))&nbsp;+0.5))/pow(10,decimals)); &nbsp;&nbsp;&nbsp;&nbsp;printf("&nbsp;additional&nbsp;:&nbsp;%d%s\n",&nbsp;(int)f,&nbsp;s+1); &nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;0;}以及测试的输出:&nbsp;10&nbsp;decimals:&nbsp;1234.056789 &nbsp;&nbsp;3&nbsp;decimals:&nbsp;1234.057 &nbsp;negative&nbsp;10:&nbsp;-1234.056789 &nbsp;negative&nbsp;&nbsp;3:&nbsp;-1234.057 &nbsp;additional&nbsp;:&nbsp;1.01现在,所有标准都得到满足:零后面的最大小数是固定的。移除尾随零点它在数学上是正确的(对吗?)当第一个小数为零时(现在)也工作。不幸的是,这个答案是两条直线的sprintf不返回字符串。
随时随地看视频慕课网APP
我要回答