signed、unsigned 关键字

本文解析了计算机中负数的存储原理及补码表示方法,通过具体实例阐述了有符号char类型数据的范围限制及其在内存中的表现形式。

目录

前言

实例


前言

我们知道计算机底层只认识 0、1.任何数据到了底层都会变计算转换成 0、1.那负数怎么存储呢?肯定这个“-”号是无法存入内存的,怎么办?很好办,做个标记。

把基本数据类型的最高位腾出来,用来存符号,同时约定如下:最高位如果是 1,表明这个数是负数,其值为除最高位以外的剩余位的值添上这个“-”号;如果最高位是 0,表明这个数是正数,其值为除最高位以外的剩余位的值。

这样的话,一个 32 位的 signed int 类型整数其值表示法范围为:- 231~ 231 -1;8 位的char 类型数其值表示的范围为- 27 ~ 27 -1。一个 32 位的 unsigned int 类型整数其值表示法范围为:0~ 232 -1;8 位的 char 类型数其值表示的范围为 0~ 28-1。同样我们的 signed 关键字也很宽恒大量,你也可以完全当它不存在,编译器缺省默认情况下数据为 signed 类型的。

实例

上面的解释很容易理解,下面就考虑一下这个问题:

int main()

{

    char a[1000];

    int i;

    for(i=0; i<1000; i++) {

        a[i] = -1-i;

    }

    printf("%d",strlen(a));

    return 0;

}

此题看上去真的很简单,但是却鲜有人答对。答案是 255。别惊讶,我们先分析分析。for 循环内,当 i 的值为 0 时,a[0]的值为-1。

关键就是-1 在内存里面如何存储。我们知道在计算机系统中,数值一律用补码来表示(存储)。主要原因是使用补码,可以将符号位和其它位统一处理;同时,减法也可按加法来处理。另外,两个用补码表示的数相加时,如果最高位(符号位)有进位,则进位被舍弃。正数的补码与其原码一致;负数的补码:符号位为 1,其余位为该数绝对值的原码按位取反,然后整个数加 1。

按照负数补码的规则,可以知道-1 的补码为 0xff,-2 的补码为 0xfe……当 i 的值为 127时,a[127]的值为-128,而-128 是 char 类型数据能表示的最小的负数。当 i 继续增加,a[128]的值肯定不能是-129。因为这时候发生了溢出,-129 需要 9 位才能存储下来,而 char 类型数据只有 8 位,所以最高位被丢弃。剩下的 8 位是原来 9 位补码的低 8 位的值,即 0x7f。

当 i 继续增加到 255 的时候,-256 的补码的低 8 位为 0。然后当 i 增加到 256 时,-257 的补码的低 8 位全为 1,即低八位的补码为 0xff,如此又开始一轮新的循环……

按照上面的分析,a[0]到 a[254]里面的值都不为 0,而 a[255]的值为 0。strlen 函数是计算字符串长度的,并不包含字符串最后的‘\0’。而判断一个字符串是否结束的标志就是看是否遇到‘\0’。如果遇到‘\0’,则认为本字符串结束。

分析到这里,strlen(a)的值为 255 应该完全能理解了。这个问题的关键就是要明白 char类型默认情况下是有符号的,其表示的值的范围为[-128,127],超出这个范围的值会产生溢出。另外还要清楚的就是负数的补码怎么表示。弄明白了这两点,这个问题其实就很简单了。

### C++ 中 unsigned 关键字的含义和用法 #### 1. 定义与基本特性 在 C++ 中,`unsigned` 是一个关键字,用于声明无符号类型变量。无符号类型意味着该变量只能存储非负整数值(即零或正数)。通过使用 `unsigned`,可以扩展变量的正数范围,因为其所有位都用于表示正值[^1]。 例如,一个标准的 `int` 类型通常占用 4 字节(32 位),其取值范围为 -2,147,483,648 到 2,147,483,647。而 `unsigned int` 的取值范围则为 0 到 4,294,967,295。这种扩展使得 `unsigned` 类型非常适合需要处理大数值但不需要负数的场景。 ```cpp unsigned int max_value = 4294967295; // 最大值示例 ``` #### 2. 数据类型结合 `unsigned` 可以与多种整数类型结合使用,包括但不限于: - `unsigned char` - `unsigned short` - `unsigned int` - `unsigned long` - `unsigned long long` 每种类型的大小和范围取决于目标平台的编译器实现。例如,在大多数现代系统中: - `unsigned short` 的范围是 0 到 65,535。 - `unsigned long long` 的范围是 0 到 18,446,744,073,709,551,615。 #### 3. 使用场景 以下是 `unsigned` 的典型使用场景: - **数组索引**:由于数组索引始终是非负数,因此 `unsigned` 是一种自然选择。 ```cpp unsigned int index = 10; std::cout << "Element at index: " << array[index] << std::endl; ``` - **计数器**:当循环计数器仅需非负值时,`unsigned` 提供了更大的上限。 ```cpp for (unsigned int i = 0; i < large_number; ++i) { // 循环体 } ``` - **位操作**:在涉及位掩码或位字段的操作中,`unsigned` 类型能够避免意外的符号扩展问题。 ```cpp unsigned int flags = 0x0F; if (flags & 0x01) { std::cout << "Flag 0 is set." << std::endl; } ``` #### 4. 注意事项 尽管 `unsigned` 提供了更大的正数范围,但在实际编程中需要注意以下几点: - **溢出行为**:`unsigned` 类型在发生溢出时会进行模运算,这可能导致意外结果。例如,`unsigned int x = -1;` 实际上会将 `x` 设置为其最大值[^1]。 ```cpp unsigned int x = -1; // 等价于 x = 4294967295 std::cout << "x: " << x << std::endl; ``` - **混合类型运算**:当 `unsigned` 和 `signed` 类型混合运算时,可能会导致隐式类型转换,从而引发错误或不符合预期的结果[^1]。 ```cpp int a = -1; unsigned int b = 1; if (a < b) { // 此条件可能不按预期工作 std::cout << "a is less than b" << std::endl; } ``` #### 5. 示例代码 以下是一个综合示例,展示 `unsigned` 的使用及其注意事项: ```cpp #include <iostream> int main() { unsigned int count = 0; while (count < 5) { std::cout << "Count: " << count << std::endl; ++count; } // 溢出示例 unsigned int overflow = 0; --overflow; // 溢出,结果为最大值 std::cout << "After decrement: " << overflow << std::endl; return 0; } ``` --- ###
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值