我曾经写过一个小程序,用于显示int类型数据的二进制表示形式(补码). 代码如下:
#include #include
int main(void){int i = 0; int值= -128; // 0xffffff80 int shift = INT_MIN; // 0x80000000
printf(“十进制值为: %d \ n \ n”,值);
printf(“ **** \ n”); //打印标记
for(i = 0; i <32; ++ i){if(shift&value){printf(“ 1”);} else {printf(“ 0”);}
shift = shift》 1; //右移1位} //打印位
返回0;}预期输出如下:
十进制值为: -128
**** 11111111111111111111111110000000但是实际输出如下:
十进制值为: -128
**** 11111111111111111111111111111111跟踪变量移位的值,发现最高位(MSB)始终为1,我突然意识到这是算术右移,这意味着将保留符号位. 因此,将shift更改为(unsigned int)类型,并真正解决了该错误. 请注意,对于无符号数字,将执行逻辑右移,并在删除的位中填充0.
现在我们自然地想到它了,无符号数的左移是逻辑左移,而有符号数的左移是算术左移(即符号位被保留)吗?答案是对还是错. 实际上,逻辑左移和算术左移是相同的,并且都没有保留符号位.
右移操作的结果将不会超出指示范围. 对于带符号的数字,如果在向右移一位后将符号位腾出,则必须保留原始符号位的值;否则,将保留原来的符号位. 对于无符号数字,没有符号位c语言移位运算,并且自然采用逻辑右移.
左移操作不同,有可能超出表示范围,即溢出. 对于MSB位0和MSB-1位1的带正号的签名,以及MSB位1和MSB-1位0的带负号的负数字,左移肯定会受益(以int类型为例,这意味着范围是-2 ^ 31〜2 ^ 31-1. 对于大于或等于2 ^ 30且小于或等于-2 ^ 30-1的值,将其向左移1位,即相乘乘以2(必须超出范围),并且只有在这种情况下,符号位才会更改. 现在已经溢出了,保留原始符号位的意义何在?
总的来说,可以执行C语言中位操作的char / short / int / long(int)/ long long(int)类型是带符号的,而无符号前缀是无符号的c语言移位运算,这两个在右边换档操作有所不同.
本文摘自: 《中国IT实验室软件月度参考》:
本文来自电脑杂谈,转载请注明本文网址:
http://www.pc-fly.com/a/jisuanjixue/article-270923-1.html