在c语言中,unsigned是用来修饰整型数据类型的保留关键字,它的核心作用是声明该类型变量为无符号类型,即变量只能存储非负整数,不存在负数表示的可能。默认情况下,c语言中的整型如int、short、long都默认是有符号类型,也就是signed类型,既可以存储正数也可以存储负数,而加上unsigned修饰后,类型的存储规则会发生改变。
unsigned的存储原理
有符号整型的最高位会被用作符号位,0表示正数,1表示负数,剩下的位表示数值大小。而unsigned类型没有符号位,所有的二进制位都用来表示数值大小,因此相同位数下,unsigned类型的取值范围比对应的有符号类型更大,但只能表示非负数。
以最常见的unsigned int为例,在32位系统中,int类型占4个字节共32位,其中1位是符号位,31位表示数值,取值范围是-2147483648到2147483647。而unsigned int的32位全部用来表示数值,取值范围是0到4294967295。
unsigned的常见使用场景
- 当确定变量不会出现负数时,使用unsigned类型可以避免符号位带来的数值范围浪费,比如表示年龄、数量、数组下标等场景。
- 进行位运算时,使用unsigned类型可以避免有符号类型右移时的符号扩展问题,保证运算结果符合预期。
- 处理底层硬件数据、网络传输数据或者文件二进制数据时,很多时候数据的表示是无符号的,使用unsigned类型更贴合实际数据格式。
unsigned的使用示例
下面是unsigned类型的基本定义和使用代码示例:
#include <stdio.h>
int main() {
// 定义有符号整型变量
int signed_num = -10;
// 定义无符号整型变量
unsigned int unsigned_num = 10;
printf("有符号整型变量值:%dn", signed_num);
printf("无符号整型变量值:%un", unsigned_num);
// 无符号类型不能存储负数,赋值负数会出现数值转换
unsigned int error_num = -1;
// 输出结果会是4294967295(32位系统下),因为-1的补码被当作无符号数解析
printf("赋值负数的无符号变量值:%un", error_num);
// 无符号数之间的运算
unsigned int a = 100;
unsigned int b = 200;
unsigned int sum = a + b;
printf("无符号数相加结果:%un", sum);
return 0;
}
使用unsigned的注意事项
- 无符号类型不能存储负数,如果给无符号变量赋值负数,编译器不会报错,但数值会按照二进制补码规则转换为对应的无符号值,导致结果不符合预期。
- 有符号类型和无符号类型混合运算时,有符号类型会被隐式转换为无符号类型,可能会出现逻辑错误。比如下面的代码:
#include <stdio.h>
int main() {
int a = -1;
unsigned int b = 1;
// 比较时a会被转换为无符号数,值变为很大的正数,因此条件成立
if (a < b) {
printf("a小于bn");
} else {
printf("a大于等于bn");
}
return 0;
}
上述代码中,a是有符号的-1,b是无符号的1,比较时a会被转换为无符号数,值变成4294967295,因此会输出"a大于等于b",和直观预期不符。
- 打印无符号类型变量时,要使用%u格式符,不能使用%d,否则可能会输出错误的结果。
- 循环中使用无符号变量作为计数器时,要注意终止条件,比如下面的循环会陷入死循环:
#include <stdio.h>
int main() {
unsigned int i = 10;
// 当i减到0之后,再减1会变成最大的无符号数,永远大于0,循环无法终止
for (; i >= 0; i--) {
printf("%un", i);
}
return 0;
}
常见unsigned类型对照
下面是c语言中常见的unsigned类型与对应有符号类型的对比:
| 有符号类型 | 无符号类型 | 32位系统下无符号类型取值范围 |
|---|---|---|
| int | unsigned int | 0 ~ 4294967295 |
| short | unsigned short | 0 ~ 65535 |
| long | unsigned long | 0 ~ 4294967295(32位系统)/ 0 ~ 18446744073709551615(64位系统) |
| char | unsigned char | 0 ~ 255 |
在实际编程中,需要根据变量的实际取值范围和是否可能出现负数,合理选择是否使用unsigned修饰符,避免因为类型选择不当导致的程序错误。