我是looking的一个类似sprintf
的格式字符串,该格式字符串将以变量的小数位数设置浮点数的格式,但以固定的字符总数(例如3个),以便提供最大的信息。例如:
0 -> `.00` or `0.0` or ` 0`
0.124 -> `.12`
0.357 -> `.36`
1.788 -> `1.8`
9.442 -> `9.4`
10.25 -> `10.` or ` 10`
75.86 -> `76.` or ` 76`
99.44 -> `99.` or ` 99`
100.0 -> `100`
(是的,我的数字都是0到100个浮点数)
该如何实现?
这种固定宽度格式是否以sprintf格式字符串语言实现?
答案 0 :(得分:3)
这种固定宽度的格式是否以sprintf格式字符串语言实现?
否。
如何实现?
为达到OP的目标,代码可以分析float f
的范围,以尝试使用最大的信息进行打印。
此操作以"%.*f"
开头,以控制.
之后的位数。
一个常见的陷阱是值小于10的幂并四舍五入并导致输出中的另一个数字。
例如,代码可以尝试针对f < 0.995f
而不是f < 1.0
进行测试,但是鉴于浮点的二进制性质,这会导致失败的转折情况。
更好地根据确切的常数(例如1.0、10.0,...)测试范围
以下内容最多尝试两次到sprintf()
。第二次尝试使用的讨厌值接近10的幂。
#include <assert.h>
#include <stdio.h>
#include <stdbool.h>
#include <string.h>
void iago_sprint1(char *dest, int len, float f) {
if (f < 1.0) {
char buf[len + 2];
//printf("p1\n");
snprintf(buf, sizeof buf, "%.*f", len - 1, f);
if (buf[0] == '0') {
strcpy(dest, buf + 1);
return;
}
}
float limit = 10.0;
int prec = len - 2;
while (prec >= -1) {
if (f < limit) {
char buf[len + 2];
//printf("p2\n");
int cnt = snprintf(buf, sizeof buf, "%.*f", prec < 0 ? 0: prec, f);
if (cnt <= len) {
strcpy(dest, buf);
return;
}
}
prec--;
limit *= 10;
}
assert(0); // `f` was out of range
}
#define N 3
int main(void) {
float f[] = {0, 0.124f, 0.357f, 1.788f, 9.442f, 10.25f, 75.86f, 99.44f,
100.0f, 99.999f, 9.9999f, .99999f, .099999f, 1.04f, 1.06f};
char line[N + 1];
for (unsigned i = 0; i < sizeof f / sizeof *f; i++) {
//snprintf(line, sizeof line, "%#*g", N, f[i]);
//puts(line);
iago_sprint1(line, N, f[i]);
puts(line);
}
}
输出
.00
.12
.36
1.8
9.4
10
76
99
100
100
10
1.0
.10
1.0
1.1
预先计算精度需要陷阱
如果代码尝试预先计算仅 1 sprintf()
个调用所需的精度,则计算需要像sprintf()
一样推导精度-实际上,代码正在执行sprint()`重新工作。
考虑len==3
和float x = 9.95f;
。由于二进制float
不能完全代表它,因此它的值刚好在9.95 1 之上或之下。如果在下面,则字符串应为"9.9"
;如果在上面,则字符串应为"10."
。如果代码具有double x = 9.95;
(同样无法精确表示),则输出可能会有所不同。如果代码使用float
,但使用FLT_EVAL_MODE > 1
,则传递的实际值可能不是预期的float 9.95
。
Precision--> .1 .0
9.94001... "9.9" "10."
9.94999... "9.9" "10." // Is this 9.95f
9.95001... "10.0" "10." // or this?
9.95999... "10.0" "10."
在接受之后,我重新设计并简化了代码。
技巧将根据"%*.f"
的10的幂以f
打印计算出的精度到比目标宽一倍的缓冲区 / em>-假设由于舍入没有进位到另一个数字。
10的幂计算可以通过一个小循环精确完成。
当开头字符为0
时,不需要像"0.xx"
-> ".xx"
中那样。
否则,由于舍入而没有进位到另一个数字,因此该字符串适合并且已经完成。
否则为进位,则最后一个字符是小数点后的'.'
或'0'
,因此不需要。当f
的幂数刚好低于10时,会发生这种情况,而印刷版本的整数倍数是10的幂。因此,只能复制长度为1的数字。
// `len` number of characters to print
// `len+1` is the size of `dest`
void iago_sprint3(char *dest, int len, float f) {
assert(len >= 1);
int prec = len - 1;
float power10 = 1.0;
while (f >= power10 && prec > 0) {
power10 *= 10;
prec--;
}
char buf[len + 2];
int cnt = snprintf(buf, sizeof buf, "%.*f", prec, f);
assert (cnt >= 0 && cnt <= len + 1);
if (buf[0] == '0') {
strcpy(dest, buf + 1);
return;
}
strncpy(dest, buf, (unsigned) len);
dest[len] = 0;
}
1 典型的float 9.95f
就是
9.94999980926513671875
典型的double 9.95
就是
9.949999999999999289457264239899814128875732421875
答案 1 :(得分:0)
您可以尝试:
void mysprintf(float a) {
if (a < 10 && a >= 0) {
printf("%2f", a);
} else if (a >= 10 && a < 100) {
printf("%1f", a);
} else {
printf("%d", (int)a);
}
}