11

一些背景知识:例如,如果我想使用 forscanf()将字符串转换为标准整数类型,例如uint16_t,我会使用SCNu16from <inttypes.h>,如下所示:

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

但是更不常见的整数类型pid_t没有这样的东西。仅支持普通整数类型<inttypes.h>。要以另一种方式转换为可移植printf()的 a 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()ing intmax_t,然后在强制转换为 之前检查该值是否在pid_t's 范围内pid_t,但似乎没有办法获得 的最大值或最小值pid_t

4

3 回答 3

8

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

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

#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在本例中为 ),否则它也不会使用其他任何东西。

于 2009-02-12T19:18:10.550 回答
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 位。我认为可能发生的最糟糕的事情是“具有适当特权”的进程读取了一个太大的值,并由于类型不匹配而向错误的进程发送了信号。我不相信我会为此担心。

...哦哦!...p400下<sys/types.h>

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

于 2009-01-13T02:54:33.073 回答
0

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

正如这个答案中所解释的,规范说signed int。如果 'int' 改变了,你的 '%u' 根据定义也会随之改变。

于 2009-01-13T03:10:08.813 回答