仅使用 ANSI C,确定 C 样式字符串是整数还是实数(即浮点数/双精度数)的最佳方法是什么?
9 回答
不要使用 atoi 和 atof,因为这些函数在失败时返回 0。上次我检查 0 是一个有效的整数和浮点数,因此没有用于确定类型。
使用 strto{l,ul,ull,ll,d} 函数,因为这些函数在失败时设置 errno,并且还报告转换数据的结束位置。
strtoul:http ://www.opengroup.org/onlinepubs/007908799/xsh/strtoul.html
此示例假定字符串包含要转换的单个值。
#include <errno.h>
char* to_convert = "some string";
char* p = to_convert;
errno = 0;
unsigned long val = strtoul(to_convert, &p, 10);
if (errno != 0)
// conversion failed (EINVAL, ERANGE)
if (to_convert == p)
// conversion failed (no characters consumed)
if (*p != 0)
// conversion failed (trailing data)
感谢 Jonathan Leffler 指出我忘了先将 errno 设置为 0。
使用sscanf,您可以确定字符串是 float 还是 int 或其他类型,而无需特殊情况 0,就像 atoi 和 atof 解决方案一样。
这是一些示例代码:
int i;
float f;
if(sscanf(str, "%d", &i) != 0) //It's an int.
...
if(sscanf(str "%f", &f) != 0) //It's a float.
...
如果不能, atoi 和 atof 将转换或返回 0。
我同意 Patrick_O 的观点,即 strto{l,ul,ull,ll,d} 函数是最好的方法。不过有几点需要注意。
- 在调用函数之前将 errno 设置为零;没有任何功能可以为您做到这一点。
- 链接到的 Open Group 页面(我在注意到 Patrick 也链接到它之前访问过)指出 errno 可能未设置。如果值超出范围,则设置为 ERANGE;如果参数无效,则可以将其设置(但同样也可以 不设置)为 EINVAL。
根据手头的工作,我有时会安排跳过返回的转换指针末尾的尾随空格,然后如果最后一个字符不是终止的 null '\0',则抱怨(拒绝)。或者我可以草率地让垃圾出现在最后,或者我可以接受可选的乘数,如 'K'、'M'、'G'、'T' 表示千字节、兆字节、千兆字节、太字节……或任何其他基于上下文的需求。
我想您可以单步调试字符串并检查其中是否有任何.
字符。不过,这只是我脑海中闪现的第一件事,所以我相信还有其他(更好的)方法可以更加确定。
使用 strtol/strtoll(不是 atoi)检查整数。使用 strtof/strtod(不是 atof)检查双打。
atoi 和 atof 转换字符串的初始部分,但不告诉您他们是否使用了所有字符串。strtol/strtod 告诉你字符转换后是否有多余的垃圾。
所以在这两种情况下,请记住传入一个非空的 TAIL 参数,并检查它是否指向字符串的末尾(即 **TAIL == 0)。还要检查下溢和上溢的返回值(有关详细信息,请参阅手册页或 ANSI 标准)。
另请注意,strod/strtol 会跳过初始空格,因此如果要将具有初始空格的字符串视为格式错误,还需要检查第一个字符。
这实际上取决于您首先要问的原因。
如果您只想解析一个数字并且不知道它是浮点数还是整数,那么只需解析一个浮点数,它也会正确解析一个整数。
如果你真的想知道类型,也许是为了分类,那么你真的应该考虑按照你认为最相关的顺序测试类型。就像尝试解析一个整数,如果你不能,然后尝试解析一个浮点数。(反过来只会产生更多的浮动......)
即使有尾随非数字字符, atoi 和 atof 也会转换数字。但是,如果您使用 strtol 和 strtod ,它不仅会跳过前导空格和可选符号,还会为您留下一个指向不在数字中的第一个字符的指针。然后您可以检查其余部分是否为空格。
好吧,如果你不想使用像 strtoul 这样的新函数,你可以添加另一个 strcmp 语句来查看字符串是否为 0。
IE
if(atof(token) != NULL || strcmp(token, "0") == 0)