移位的位数是负数,结果会怎样?

开发 前端
有过编程经验的同学,对于移位操作应该很熟悉了,日常工作中或多或少都有用到,当 移位位数是负数 或者 移位位数超过了 类型的最大二进制位时,和正常移位处理是不一样的,下面将详细说明这两种情况,在此之前,先了解下正常的移位操作。

[[432672]]

有过编程经验的同学,对于移位操作应该很熟悉了,日常工作中或多或少都有用到,当 移位位数是负数 或者 移位位数超过了 类型的最大二进制位时,和正常移位处理是不一样的,下面将详细说明这两种情况,在此之前,先了解下正常的移位操作

正数的左移和右移

正数的左移是二进制位向左移动,右边留空的位置补 0,右移是二进制位向右移动,左边留空的位置补 0 ( 符号位为 0 )

  • 左移

左移操作,最高位的符号位会出现 0 或 1 , 因此结果会出现正数和负数的情况

新建测试文件 base.cpp,代码如下

  1. #include <stdint.h> 
  2. #include <iostream> 
  3. using namespace std; 
  4. int32_t main(int32_t argc , char *argv[]) 
  5.     int32_t a = 7; 
  6.     cout << "(a << 2) = " << (a << 2) << endl; 
  7.     cout << "(a << 30) = " << (a << 30) << endl; 
  8.  
  9.  
  10.     return 0; 

执行 g++ -g -Wall -std=c++11 -o base base.cpp 命令编译代码,再执行 ./base 运行程序,结果如下

  1. (a << 2) = 28 
  2. (a << 30) = -1073741824 

上述代码中,变量 a 的值为 7,对应的二进制是 0000 0000 0000 0000 0000 0000 0000 0111

左移 2 位:二进制向左移动 2 位,右边补充 2 位 0 ,左边丢弃超出的 2 位二进制, 结果是 0000 0000 0000 0000 0000 0000 0001 1100, 对应的十进制数是 28

左移 30 位 的流程如下图

由上图可知,二进制向左移动 30 位后, 左边 30 位二进制 0000 0000 0000 0000 0000 0000 0000 01 因超出被丢弃,同时最右边剩下的 2 位二进制 11 左移 30 位,右边空的位置补充 30 位 0,最终的结果是 1100 0000 0000 0000 0000 0000 0000 0000, 对应的十进制数是 -1073741824

可以看出,正数 7 左移 30 位后,二进制的符号位变成了 1 ,也即正数变成了负数了

  • 右移

正数右移,最小为 0 , 不会出现负数,下面是右移的测试代码

修改 base.cpp 文件,代码如下

  1. #include <stdint.h> 
  2. #include <iostream> 
  3. using namespace std; 
  4. int32_t main(int32_t argc , char *argv[]) 
  5.    int32_t a = 7; 
  6.    cout << "(a >> 1) = " << (a >> 1) << endl; 
  7.    cout << "(a >> 31) = " << (a >> 31) << endl; 
  8.  
  9.     return 0; 

编译并运行上面程序,结果如下:

  1. (a >> 1) = 3 
  2. (a >> 31) = 0 

上述实例代码中,变量 a 的值为 7,对应的二进制是 0000 0000 0000 0000 0000 0000 0000 0111

右移 1 位:二进制向右移动 1 位,左边补充一位 0 ,右边丢弃超出的一位二进制, 结果是 0000 0000 0000 0000 0000 0000 0000 0011, 对应的十进制数是 3

右移 31 位:二进制向右移动 31 位,左边补充 31 位 0 ,右边丢弃超出的 31 位二进制,结果是 0000 0000 0000 0000 0000 0000 0000 0000, 对应的十进制是 0

负数的左移和右移

负数的左移是二进制位向左移动,右边留空的位置补 0,右移是二进制位向右移动,左边留空的位置补 1 ( 符号位为 1 ),这一点跟正数是不一样的

计算机中是用补码的形式进行各种运算的,正数的补码是其自身,负数的补码是将其正数按位取反加 1

  • 左移

负数左移,符号位可能会变成0,因此结果会出现正数和负数的情况,一直左移的话,最终会变成 0

修改 base.cpp文件,代码如下

  1. #include <stdint.h> 
  2. #include <iostream> 
  3. using namespace std; 
  4. int32_t main(int32_t argc , char *argv[]) 
  5.    int32_t b = -3; 
  6.    cout << "(b << 1) = " << (b << 1) << endl; 
  7.    cout << "(b << 30) = " << (b << 30) << endl; 
  8.  
  9.     return 0; 

编译并运行上面程序,结果如下:

  1. (b << 1) = -6 
  2. (b << 30) = 1073741824 

上面代码中,变量 b 的值为 -3,对应的二进制是 1111 1111 1111 1111 1111 1111 1111 1101

左移 1 位:整个二进制串向左移动 1 位,右边补充 0 ,左边丢弃超出的一位二进制,结果是 1111 1111 1111 1111 1111 1111 1111 1010,对应十进制数 -6

左移 30 位 的流程如下

由上图可知,左移 30 位,左边的 30 位二进制 1111 1111 1111 1111 1111 1111 1111 11 因超出数值最大位数而被丢弃,原来最右边的 01 移到了最左边,紧接着后面的 30 个空位全部补 0 ,最终的结果是 0100 0000 0000 0000 0000 0000 0000 0000 ,对应十进制是 1073741824

可以看出,负数 -3 左移 30 位后,二进制的符号位变成了 0 ,由开始的负数变成了正数了

  • 右移

负数右移是在左边补 1, 所以结果不会出现正数的情况,如果一直右移,最终二进制位会全部变成 1,即十进制的 -1 ( 二进制全 1 在补码中表示 -1 )

修改 base.cpp,代码如下

  1. #include <stdint.h> 
  2. #include <iostream> 
  3. using namespace std; 
  4. int32_t main(int32_t argc , char *argv[]) 
  5.    int32_t b = -3; 
  6.    cout << "(b >> 1) = " << (b >> 1) << endl; 
  7.    cout << "(b >> 31) = " << (b >> 31) << endl; 
  8.  
  9.     return 0; 

编译并运行上面程序,结果如下:

  1. (b >> 1) = -2 
  2. (b >> 31) = -1 

上面代码中,变量 b 的值为 -3,对应的二进制是 1111 1111 1111 1111 1111 1111 1111 1101

右移 1 位:二进制位向右移动 1 位,左边补充 1 ,右边丢弃一位超出的二进制,结果为 1111 1111 1111 1111 1111 1111 1111 1110,对应的十进制是 -2

右移 31 位 的流程如下

根据上图可知,右移 31 位,最右边的 31 位二进制 1111 1111 1111 1111 1111 1111 1111 110 因超出数值最大位数而被丢弃, 原来左边的 1 移到了最右边,左边补 31 位 1,最后结果为:1111 1111 1111 1111 1111 1111 1111 1111 , 对应的十进制数 -1

移位数超过类型最大位数

移位的位数大于等于数值类型的最大位数时,实际的移的位数是:移位的位数和该类型的最大位数做取模运算,余数就是要移的位数,不管左移还是右移,这种方法都适用

比如:类型为 int32_t,移位的位数是 34,实际移位的位数为:34 % 32 = 1

修改 base.cpp 文件,内容如下:

  1. #include <stdint.h> 
  2. #include <iostream> 
  3. using namespace std; 
  4. int32_t main(int32_t argc , char *argv[]) 
  5.     int32_t a = 7; 
  6.     cout << "(a >> 32) = " << (a >> 32) << endl; 
  7.     cout << "(a >> 33) = " << (a >> 33) << endl; 
  8.     cout << "(a << 34) = " << (a << 34) << endl; 
  9.     return 0; 

编译并执行,结果如下

  1. (a >> 32) = 7 
  2. (a >> 33) = 3 
  3. (a << 34) = 28 

变量 a 的值为 7,对应的二进制是 0000 0000 0000 0000 0000 0000 0000 0111

右移 32 位:因移位位数等于 int32_t 最大位数,所以实际移位数为 32 % 32 = 0,右移 0 位 表示没有移位,所以结果还是 7

右移 33 位:移位位数大于 int32_t 最大位数,故实际移位数为 33 % 32 = 1,右移 1 位,左边补 0 ,右边丢弃超出的位,结果是: 0000 0000 0000 0000 0000 0000 0000 0011,对应的十进制是 3

左移 34 位:移位位数大于 32 ,实际移位数为 34 % 32 = 2,左移 2 位,右边补 0 ,左边丢弃超出的位,结果是:0000 0000 0000 0000 0000 0000 0001 1100,对应的十进制是 28

正常情况下,在移位数相同时,分几次移位操作和单次移位操作的结果是一样的, 比如:一个 int32_t 变量,值为 7, 7 << 1 << 2 和 7 << 3 的结果相同的

当移位数大于等于数值类型最大位数时,上述的结果是不一样的,比如:一个 int32_t 变量,值为 7, 虽然 7 << 33 和 7 << 20 << 13 两者移位数都是 33,但结果却是不同的,前者是 14,而后者是 0

移位的位数是负值

当移位的位数是负值时,实际移位的位数是:用被移位数值类型的最大位数和移位位数相加,如果结果还是负数,结果继续 加上被移位数值类型的最大位数,直到结果不为负数为止,此时的结果即为最终移位的位数

比如:被移位的数据类型是 int32_t,移位位数是 -31,最终移位的位数是:32 + ( -31 ) = 1

当移位位数是 -60,计算最终移位位数,32 + ( -60 ) = -28,由于结果还是负数,所以继续相加,32 + ( -28 ) = 4,此次结果不为负数了,所以最终移位的位数是 4

修改 base.cpp 文件,内容如下:

  1. #include <stdint.h> 
  2. #include <iostream> 
  3. using namespace std; 
  4. int32_t main(int32_t argc , char *argv[]) 
  5.     int32_t a = 7; 
  6.     cout << "(a >> -31) = " << (a >> -31) << endl; 
  7.     cout << "(a << -60) = " << (a >> -60) << endl; 
  8.     return 0; 

编译代码并执行,结果如下

  1. (a >> -31) = 3 
  2. (a >> -60) = 0 

变量 a 的值为 7,对应的二进制是 0000 0000 0000 0000 0000 0000 0000 0111

右移 -31 位:移位数是负数,实际右移 32 + ( -31 ) = 1 位,结果为:0000 0000 0000 0000 0000 0000 0000 0011,对应的十进制是 3

右移 -60 位:移位数是负数,实际右移 32 + 32 + ( -60 ) = 4 位,结果为:0000 0000 0000 0000 0000 0000 0000 0000,对应的十进制是 0

小结

 

本文主要介绍了左移和右移操作,左移相当于乘以 2 的 N 次方,而右移相当于除以 2 的 N 次方,这里的 N 表示移位的位数,需要注意的是,当移位位数是负数或者大于等于类型最大位数时,编译器对他们的处理和正常的移位是不一样的

 

责任编辑:武晓燕 来源: Linux开发那些事儿
相关推荐

2013-07-29 17:04:18

2021-08-27 15:03:51

PythonC语言数组

2014-02-19 16:26:26

VDI部署

2013-08-20 09:48:59

2022-02-18 15:07:29

goroutinepanic协程

2009-12-03 13:32:04

Virtuozzo捆绑

2009-04-22 08:54:25

2010-03-24 15:40:39

网管运维管理摩卡软件

2020-10-27 07:34:41

基站手机蜂窝网络

2011-04-07 09:33:22

2015-11-10 09:09:23

代码程序员成长

2011-11-25 09:48:04

天线无线

2013-08-19 16:17:48

CIO

2024-03-28 08:13:51

GPTsOpenAI人工智能

2009-09-02 20:18:17

域名劫持域名安全

2022-11-24 11:09:03

自然语言处理(智能语音

2015-06-30 11:52:30

2023-11-07 07:21:04

2014-04-30 15:37:05

2014-12-31 10:02:14

Android可穿戴设备世界
点赞
收藏

51CTO技术栈公众号