一些背景知识:例如,如果我想使用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
的最大值或最小值的方法。
答案 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>
实现应支持一个或多个宽度的编程环境 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'按定义会随之改变。