- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
将整数转换为文本时,通常我会创建一个大 缓冲区以与sprintf()
一起使用保留任何潜在的结果。
char BigBuffer[50];
sprintf(BugBuffer, "%d", SomeInt);
我想要更节省空间并且当然更便携,所以而不是 50
,找到替代方案:
(sizeof(integer_type)*CHAR_BIT*0.302) + 3
// 0.0302 about log10(2)
#define USHORT_DECIMAL_BUFN ((size_t) (sizeof(unsigned short)*CHAR_BIT*0.302) + 3)
#define INT_DECIMAL_BUFN ((size_t) (sizeof(int) *CHAR_BIT*0.302) + 3)
#define INTMAX_DECIMAL_BUFN ((size_t) (sizeof(intmax_t) *CHAR_BIT*0.302) + 3)
int main() {
char usbuffer[USHORT_DECIMAL_BUFN];
sprintf(usbuffer, "%hu", USHRT_MAX);
printf("Size:%zu Len:%zu %s\n", sizeof(usbuffer), strlen(usbuffer), usbuffer);
char ibuffer[INT_DECIMAL_BUFN];
sprintf(ibuffer, "%d", INT_MIN);
printf("Size:%zu Len:%zu %s\n", sizeof(ibuffer), strlen(ibuffer), ibuffer);
char imbuffer[INTMAX_DECIMAL_BUFN];
sprintf(imbuffer, "%" PRIdMAX, INTMAX_MIN);
printf("Size:%zu Len:%zu %s\n", sizeof(imbuffer), strlen(imbuffer), imbuffer);
return 0;
}
Size:7 Len:5 65535
Size:12 Len:11 -2147483648
Size:22 Len:20 -9223372036854775808
所以问题是:
1 替代方程有问题吗?
2 有什么更好的解决办法? - 因为这个替代方案有点浪费,而且看起来过于复杂。
[编辑答案]
答案提供了 3 种深思熟虑的方法:
1 使用缓冲区[类型的最大大小](已选择答案)
2 asprintf()
3 snprintf()
1 使用等式 (sizeof(integer_type)*CHAR_BIT*0.302) + 3
的编译时最大缓冲区大小没有损坏也没有改进。 <locale.h>
的影响|按照@paddy 的建议进行了研究,没有区域设置影响整数转换%d %x %u %i
.我们发现,如果已知类型是有符号或无符号的(如下),则可以对等式进行轻微改进。 @paddy 关于“更保守”的警告是个好建议。
2 asprintf()
确实是一个很好的通用解决方案,但不可移植。也许在 C11 之后?
3 snprintf()
,虽然是标准的,但在提供的缓冲区过小时存在已知的一致实现问题。这意味着使用超大缓冲区调用它,然后生成一个大小合适的缓冲区。 @jxh 建议使用线程安全的全局暂存缓冲区,以使用本地大小合适的缓冲区来形成答案。这种新颖的方法值得考虑,我可能会使用它,但最初的问题更侧重于在 s(n)printf()
之前确定调用保守的缓冲区大小。
signed ((sizeof(integer_type)*CHAR_BIT-1)*0.302) + 3
unsigned (sizeof(integer_type)*CHAR_BIT*0.302) + 2
*28/93
可以用来代替 *0.302
.
最佳答案
我觉得不错。您已将小数点四舍五入,为负号和空值添加了一个额外的字符,并为良好的度量添加了一个额外的字符。如果您不使用 <locale.h>
中的功能,我认为您不必担心数字会变长。 .
我的问题是你打算用这些做什么。您是简单地在堆栈上构建它们,还是将大量它们放入内存?
对于堆栈上的临时数组,您通常不会为几个字节而大惊小怪,因为它不太可能影响缓存性能。它肯定不会耗尽您的内存力。
如果您打算存储大量此类数据,则可能需要考虑池化。但是,您需要考虑池的内存开销。池的本质意味着您保留的内存多于您将要使用的内存。如果编译 64 位,你的指针是 8 个字节。如果您的大多数数字都是 4 个字符长,那么 8 字节指针加上每个数字的 5 字节存储空间将抵消任何可能的好处,64 位数字可能除外。
这些只是我的思考过程。在我看来,你已经很好地减少了脂肪。我倾向于更保守一点,但这可能主要是偏执狂。保持简单通常是要走的路,而过度思考可能是一个陷阱。如果您考虑过度,请考虑原因,并确定它是否是一个真正需要考虑太多的问题。
关于c - 如何改进 printf 各种整数类型的缓冲区大小确定?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18708679/
我想知道以下语句在 C 中会打印什么? printf("hello\n") || (printf("goodbye\n") || printf("world\n")); 我通常习惯于使用“cout”在
这是我目前正在学习的系统编程类(class)的幻灯片: catch_child 是 SIGCHLD 处理程序。输出与代码如何对应?为什么没有打印一些“Child #x started”消息? 最佳答案
这有点像拼图……我刚刚又回到了C,打算这次掌握它。所以我一直在阅读 The C Programming Language ,我得到了这个声明: Among others, printf also re
如何使用 printf 在字符串末尾附加空格? 我找不到任何在右侧附加空格的示例。 A similar question我发现使用 printf改为在字符串的左侧添加空格。 最佳答案 使用负数左对齐(
我想通过 usart 从 stm32f405 注销。 在我的 syscall.c 文件中,我实现了通过 usart 打印的功能: int _write(int file, char *ptr, int
我想定义一个记录器函数,比如 myPutStrLn = putStrLn . (++) "log: " main = do myPutStrLn "hello" 这很好。现在我想用 printf 格式
Printf module API详细介绍了类型转换标志,其中: %B: convert a boolean argument to the string true or false %b: conv
@H2CO3 这是我的主要代码: #pragma OPENCL EXTENSION cl_ amd_ printf : enable #define PROGRAM_FILE "matvec.cl"
Printf module API详细介绍了类型转换标志,其中: %B: convert a boolean argument to the string true or false %b: conv
您可以使用 printf 字段宽度说明符截断字符串: printf("%.5s", "abcdefgh"); > abcde 不幸的是,它不适用于数字(将 d 替换为 x 是相同的): printf(
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 4 年前。 Improve th
我遇到了我见过的最奇怪的错误之一。 我有一个简单的程序,可以打印多个整数数组。 对数组进行排序,然后打印...... place_in_buf(n100, 100); insertion(10
我的程序是每隔一段时间获取文件大小并显示它以记录任何更改。由于某种原因,执行下面的代码挂起,只为我提供了一个光标。没有打印或显示任何内容。 代码: #include #include #inclu
printf("It is currently %s's turn.\n", current->name); 我想知道为什么在 %s 之后打印出额外的换行符。我知道C 中的字符串总是以\0 结尾。没有
这个问题已经有答案了: printf anomaly after "fork()" (3 个回答) fork() in c using printf [duplicate] (2 个回答) 已关闭 9
我对编程很陌生。 我正在尝试编写一个程序,从数组中调用水果的价格。但我希望代码在写价格之前也写水果的名称。如果我键入 2,如何使输出为“Orange price : 10”而不仅仅是 price :
这个问题在这里已经有了答案: How do I print a non-null-terminated string using printf? (2 个答案) 关闭 7 年前。 例如,我有一个字符
我有一个 atmel UC3-L0 和罗盘传感器。现在我安装 AtmelStudio 并将一些演示代码下载到电路板中。但是我不知道演示代码中的函数 printf 会在哪里出现数据。我应该如何获取数据?
我有一个 atmel UC3-L0 和罗盘传感器。现在我安装 AtmelStudio 并将一些演示代码下载到电路板中。但是我不知道演示代码中的函数 printf 会在哪里出现数据。我应该如何获取数据?
嗨,我是 C 世界的新手,我的代码确实有些奇怪。目标是创建一个函数,可以在开头和结尾处用空格和/或制表符修剪字符串。我无法使用字符串库。 问题是我的代码中有一个 printf 只是用于测试,它的工作非
我是一名优秀的程序员,十分优秀!