如何便携式检查SuSv3数据类型的极值
How to portably check extremal values for SuSv3 data types?
SuSv3要求ssize_t
为带符号整数类型。如果我想检查我计算的值是否大于这种数据类型所允许的最大值,我可以将其与INT_MAX
进行比较,这并不好。
有没有一种更便携的方法可以进行这种比较——一个与中一样工作的宏/函数f
f(<typedef'ed datatype>) = {maximum value allowed for <TDDT> on this system)?
,还是对同一类操作的短序列?
系统:
Ubuntu 12.04.
glibc 2.15
内核3.2.0
附言:当我在谷歌上搜索这个时,我首先认为gcc扩展"typeof"听起来很有前途;但这似乎对这里没有帮助(或者确实有帮助?)。这意味着我可以接受任何可能是gcc扩展/属性等的东西。
对于无符号算术类型,(type)-1
是最大值。由于您不知道类型的相对大小,请强制转换为uintmax_t
:
#define UNSIGNED_TYPE_MAX(t) ((uintmax_t)(t)-1)
if ((uintmax_t)x > UNSIGNED_TYPE_MAX(size_t)) puts("too large");
签名类型没有这样的快捷方式。事实上,我认为在严格可移植的C89或C99中,如果不使用相应的常数,比如ssize_t
的SSIZE_MAX
,就无法确定有符号类型的最大值。C99指定了为ISO C中定义的类型的stdint.h
中定义的算术而设计的每个类型的常数。对于在POSIX中定义但不在标准C中的类型,limits.h
中有许多值;请注意,它们是对类型的有效值的限制,而不是对类型中可以容纳的值的限制。例如,如果size_t
是32位类型,则SIZE_MAX
被保证为232-1,而如果实现不支持任何大于2的字节计数,则SSIZE_MAX
可以小于2-1。
添加了整数以二进制表示并且没有填充位的假设(如果您将自己限制为POSIX(其中CHAR_BIT
总是8),这是安全的),您可以通过计算类型的大小来推断最大值:有符号类型中有一个符号位,其他都是值位。
#define SIGNED_TYPE_MAX(t) (((uintmax_t)1 << (sizeof(t) * CHAR_BIT - 1)) - 1)
请注意,像"加倍直到它停止生长"或"推入位模式0111…111"这样的事情是不可靠的。C标准规定,签名类型的行为是未定义的,GCC利用这一点对签名类型的操作进行优化,如果发生溢出,这些操作可能会导致错误的值。例如,它可能在较大大小的寄存器中执行计算,这样就不会发生溢出。
- 防止主数据类型C++的隐式转换
- 处理小于cpu数据总线的数据类型.(c++转换为机器代码)
- 在C++中打印指向不同基元数据类型的指针的内存地址
- C++浮点数据类型和字符串数据类型无法子到模板函数中
- 如何计算数据类型的范围,例如int
- C++中数据类型修饰符的顺序
- C++LinkedList问题.数据类型之间存在冲突?没有匹配的构造函数
- 特定数据类型的模板类
- 具有多个模板的模板函数,用于特定数据类型(如字符串)?
- 有没有办法提示用户使用哪种数据类型作为模板 c++
- int数据类型的指针指向的是什么,如果是一个类的私有数据成员,我们创建了该类的两个对象?
- 时间复杂度 当具有复合数据类型(如元组或对)时?
- 如何获取C++字符数据类型的地址
- 将复杂的非基元C++数据类型转换为 Erlang/Elixir 格式,以使用 NIF 导出方法
- 构造智能点数据类型以及普通数据类型的通用方法
- 如何使映射键具有两种不同的数据类型?
- 数据类型"struct seq<0, 1, 2>{}"含义是什么?
- 如何在不破坏现有应用程序的情况下更改 API 中 stl 容器的数据类型?
- Static_cast转换为错误的数据类型,但结果仍然正确?
- 如何便携式检查SuSv3数据类型的极值