如何将字符串可移植转换为不常见的整数类型?

时间:2009-01-13 02:41:20

标签: c string integer c99 scanf

一些背景知识:例如,如果我想使用scanf()将字符串转换为标准整数类型,例如uint16_t,我会使用SCNu16来自<inttypes.h> 1}},像这样:

#include <stdio.h>
#include <inttypes.h>
uint16_t x;
char *xs = "17";
sscanf(xs, "%" SCNu16, &x);

但是像pid_t这样的更不常见的整数类型没有任何这样的东西; <inttypes.h>仅支持普通整数类型。要将另一种方式转换为便携式printf() pid_t,我可以将其转换为intmax_t并使用PRIdMAX,如下所示:

#include <stdio.h>
#include <inttypes.h>
#include <sys/types.h>
pid_t x = 17;
printf("%" PRIdMAX, (intmax_t)x);

但是,似乎没有办法将scanf()移植到pid_t中。所以这是我的问题:如何便携地做到这一点?

#include <stdio.h>
#include <sys/types.h>
pid_t x;
char *xs = 17;
sscanf(xs, "%u", &x);  /* Not portable! pid_t might not be int! /*

我想到了scanf() intmax_t,然后在转换为pid_t之前检查该值是否在pid_t的限制范围内,但似乎没有是获取pid_t的最大值或最小值的方法。

3 个答案:

答案 0 :(得分:8)

有一个强大且可移植的解决方案,即使用strtoimax()并检查溢出。

也就是说,我解析intmax_t,检查来自strtoimax()的错误,然后通过投射它并将其与之比较来查看它是否“适合”pid_t原始intmax_t值。

#include <inttypes.h>
#include <stdio.h>
#include <iso646.h>
#include <sys/types.h>
char *xs = "17";            /* The string to convert */
intmax_t xmax;
char *tmp;
pid_t x;                    /* Target variable */

errno = 0;
xmax = strtoimax(xs, &tmp, 10);
if(errno != 0 or tmp == xs or *tmp != '\0'
   or xmax != (pid_t)xmax){
  fprintf(stderr, "Bad PID!\n");
} else {
  x = (pid_t)xmax;
  ...
}

无法使用scanf(),因为(正如我在评论中所述) scanf()无法检测到溢出。但我说错误strtoll()相关函数没有intmax_t; strtoimax()呢!

除非您知道整数类型的大小(在这种情况下为strtoimax()),否则除pid_t之外的其他任何内容都无法使用。

答案 1 :(得分:3)

这取决于你想要的便携性。 POSIX表示pid_t是一个有符号整数类型,用于存储进程ID和进程组ID。在实践中,您可以安全地假设long足够大。如果做不到这一点,您的intmax_t必须足够大(因此它会接受任何有效的pid_t);麻烦的是,该类型可以接受pid_t中不合法的值。你被困在岩石和坚硬的地方之间。

我会使用long而不用担心它,除了100年后的软件考古学家会发现并观察到的一个模糊的评论,这就说明了为什么256位CPU嘎然而止当将512位值作为pid_t传递时。

POSIX 1003.1-2008现已在网上提供(所有3872页,PDF和HTML)。你必须注册(免费)。我从Open Group Bookstore获得了它。

我看到的一切都是有符号整数类型。显然,所有有效的有符号整数值都适合intmax_t。我在<inttypes.h><unistd.h>中找不到表示PID_T_MAX或PID_T_MIN或其他此类值的任何信息(但我只是在今晚才能访问它,所以它可能隐藏在我没有的地方寻找它)。 OTOH,我支持我的原始评论 - 我相信32位值是实用的,我会使用long无论如何,这将是8位机器上的64位。我认为,可能发生的最糟糕的事情是“适当特权”进程读取的值过大,并且由于类型不匹配而向错误的进程发送信号。我不相信我会为此担心。

... oooh!... <sys/types.h>

下的p400
  

实现应支持一个或多个宽度的编程环境   of blksize_t,pid_t,size_t,ssize_t和suseconds_t不大于long类型的宽度。

答案 2 :(得分:0)

如果您真的担心,可以_assert(sizeof(pid_t) <= long)或您为'%'选择的任何类型。

正如this answer中所述,规范称为signed int。如果'int'发生变化,你的'%u'按定义会随之改变。