左移时是不区分无符号数的,而进行右移时也不区分是否是无符号,移进的值在最高位补0。
例如:0xff进行右移一位0xff>>1,其结果为0x7f,请参见以下代码:
#include
int main()
{
int a = 0xff;
a = a >> 1;
printf("a = %x\n", a);
return 0;
}
运行结果为:a = 7f
因为左移定义是,每个移出的位进入进位标志,右边填充0,无论是有符号数还是无符号数都这么操作,这是规定。那么,这样的操作对有符号数和无符号数的结果都是一样的,所以左移不分有符号数还是无符号数。
计算机中存放的是二进制补码表示的,正数的补码是本身,而负数的补码是取反加1.
所以负数的补码在求其真值是也是对其作取补运算。
左移、右移其实就是乘或除2,这个不影响符号,所以负数的右移补符号位,保证是负数。
这只是大道理,您自己推一下就很容易理解了。
因为在C语言标准中,只规定了无符号数的移位操作是采用逻辑移位(即左移、右移都是使用的逻辑左移和逻辑右移)。
而对于有符号数,其左移操作还是逻辑左移,但右移操作是采用逻辑右移还是算术右移就取决于机器了!(算术右移和逻辑右移的区别是:算术右移不右移符号位,即最高位,右移后前补0还是1取决于符号位的值;逻辑右移执行时将移动符号位,右移后前补0)
因此,一个程序如果使用了有符号数的右移位操作,它就是不可移植的了。