- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章移位的位数是负数,结果会怎样?由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
有过编程经验的同学,对于移位操作应该很熟悉了,日常工作中或多或少都有用到,当 移位位数是负数 或者 移位位数超过了 类型的最大二进制位时,和正常移位处理是不一样的,下面将详细说明这两种情况,在此之前,先了解下正常的移位操作 。
正数的左移是二进制位向左移动,右边留空的位置补 0,右移是二进制位向右移动,左边留空的位置补 0 ( 符号位为 0 ) 。
左移操作,最高位的符号位会出现 0 或 1 , 因此结果会出现正数和负数的情况 。
新建测试文件 base.cpp,代码如下 。
执行 g++ -g -Wall -std=c++11 -o base base.cpp 命令编译代码,再执行 ./base 运行程序,结果如下 。
上述代码中,变量 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 文件,代码如下 。
编译并运行上面程序,结果如下
上述实例代码中,变量 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文件,代码如下 。
编译并运行上面程序,结果如下
上面代码中,变量 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,代码如下 。
编译并运行上面程序,结果如下
上面代码中,变量 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 文件,内容如下:
编译并执行,结果如下 。
变量 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 文件,内容如下:
编译代码并执行,结果如下 。
变量 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 表示移位的位数,需要注意的是,当移位位数是负数或者大于等于类型最大位数时,编译器对他们的处理和正常的移位是不一样的 。
原文链接:https://mp.weixin.qq.com/s/-bBf6b0PcHuy8mo1Zl5mvg 。
最后此篇关于移位的位数是负数,结果会怎样?的文章就讲到这里了,如果你想了解更多关于移位的位数是负数,结果会怎样?的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
赏金:对于提供代码以使此子程序与负数一起工作的任何人,+50 信誉点。 我编写了一个 MIPS 程序来将华氏温度转换为摄氏温度。它打开自己的输出窗口(即 UART)并正确显示摄氏值。它在从 C 调用到
我得到了以下代码 # On va convertir en somme monétaire # We convert our integer into a way to read money
我得到了以下代码 # On va convertir en somme monétaire # We convert our integer into a way to read money
我使用以下 RegEx 基本上过滤掉任何文本,并接受数字 + 运算符。 ([-+]?[0-9]*\.?[0-9]+[\/\+\-\*])+([-+]?[0-9]*\.?[0-9]+) 所以它抓取 1+
我有一个查询,它计算我在查询中使用 union all 的平均值,以便获取我最终使用 max 函数的数据。 当联合返回结果时,如下所示:- col 1 col2 1 0
我有这样一个类: public class SpiralGenerator implements Iterator> { private void generate(int pos, E...
A = numpy.matrix([[36, 34, 26], [18, 44, 1], [11, 31, 41]]) X1 = numpy.matrix([[462
我有一个应用程序,其中有一个显示硬币 00 的 TextView ,一个按钮显示奖励视频广告,为用户提供 10 个硬币,还有一个购买按钮,将硬币减少 30 个。现在,当用户有 30 个硬币时,单击购买
话不多少,直接附上代码实例,仅供参考 ? 1
我有一系列正数和负数,我想将每个数字的绝对值增加一个,同时仍保持正数/负数。0.2 -> 1.2-0.3 -> -1.3我怎样才能做到这一点? 最佳答案 让我们尝试使用numpysign s=pd.S
我有这段代码,只允许在 keypress() 的输入字段中输入数字 if (e.which != 8 && e.which != 0 && (e.which 57)) { return fa
我试图用“-1”作为所有值填充二维数组。我正在使用的代码是: int c [] []=new int[4][4]; Arrays.fill(c,-1) 这会引发以下错误: Exception in t
在学校作业中,我们应该编写一个程序,该程序接受一个数字并将其分为三个部分:1. 检查数字是正数还是负数2. 整数(大小)3.小数部分 要求是应该有一个自己的函数,名为separate,具有输入和输出参
有没有什么方法可以在 C# 中执行整数除法(没有 float 或小数,我需要保持这个非常快)来向下舍入数字? 默认除法只是丢弃分数参数。考虑: 1 / 2 = 0 // That is correc
我正在使用 matplotlib 为报告生成图表,并指定我自己的样式表来指定文本格式以符合报告的指定文档样式。在我的 .mplstyle 样式表中,我按如下方式指定字体系列: font.family
在 C++11 中,如果我们尝试使用全局运算符 new 分配负大小的数组,它会抛出 std::bad_array_new_length,但是 C++98/C++03 呢?是 UB 还是会抛出 std:
我试过 scanf("%u",&number) 并且我输入了负数问题是当我 printf("%d",number) 我得到负数。我认为这会阻止我读取负数。scanf("%d",&number) 和 s
我的任务是解释一些看似奇怪的C代码行为(在x86上运行)。我可以轻松完成所有其他工作,但是这确实让我感到困惑。 代码段1输出-2147483648 int a = 0x80000000; int
js有问题吗? if("hello".indexOf("world")) { // I forgot to add > -1 here console.log("hello world");
我正在尝试使用 Antlr 4 设置一个简单的计算器。 语法: grammar calcGrammar; input : expression EOF; expression : MINUS
我是一名优秀的程序员,十分优秀!