按位取反怎么运算_按位取反运算

按位取反怎么运算_按位取反运算读本文前请首先搞懂“反码”,“取反”,“按位取反(~)”,这3个概念是不一样的。取反:0变1,1变0反码:正数的反码是其本身,对于负数其符号位不变其它各位取反(0变1,1变0)按位取反(~):这将是下面要讨论的。“~”运算符在c、c++、java、c#中都有,之前一直没有遇到这个运算符。要弄懂这个运算符的计算方法,首先必须明白二进制数在内存中的存放形式,二

大家好,又见面了,我是你们的朋友全栈君。

首先搞懂  “反码”,“取反”,“按位取反(~)”,这3个概念是不一样的。

取反:0变1,1变0

反码:正数的反码是其本身,对于负数其符号位不变其它各位取反(0变1,1变0)

按位取反(~): 这将是下面要讨论的。

————————————————————————————————-

“~”运算符在c、c++、java、c#中都有,之前一直没有遇到这个运算符。

要弄懂这个运算符的计算方法,首先必须明白二进制数在内存中的存放形式,二进制数在内存中是以补码的形式存放的。

另外正数和负数的补码不一样,正数的补码、反码都是其本身,既:

正数9:

  • 取原码: 0000 1001
  • 取补码: 0000 1001
  • 取反码: 0000 1001
  • 内存中存放格式 0000 1001

 

再例如: -2

  • 取原码: 1000 0010 (最高位1表示符号位)
  • 取反码: 1111 1101 (符号位不变,其余各位求反)
  • 取补码: 1111 1110 (反码末位+1)
  • 内存中存放格式: 1111 1110

————————————————————————————————-

弄懂了上述情况后,按位取反如何计算就好办了

假设要对正数9按位取反——> (~9),计算步骤如下,

  • 取原码 0000 1001,
  • 取反码 0000 1001,
  • 取补码 0000 1001,
  • 对其取反 1111 0110(符号位一起进行取反,这不是最终结果,只是补码的取反仅此而已)
  • 我们还需要把他转换成原码,由于最高位是1代表负数,下面进行负数补码到原码的逆运算
  • 先减1得反码: 1111 0101
  • 取反得原码:1000 1010,(取反过程符号位不变)
  • 前面最高位1是符号位,既得十进制:-10
  • 不知道说的明不明白,这里步骤就是:
  • 1. 先对正数求补码
  • 2. 然后对补码取反,包括符号位
  • 3. 最后进行补码求原码的逆过程。

经评论区朋友指正:原推算过程存在错误,很巧合的是当时用来举例的9按照错误的推算过程也可以获得正确的结果(用5可以推翻这个结论是错误的)。

现更新计算过程:接上面对 0000 1001 取反后得到 1111 0110,由于计算机需要以补码表示,需要对该值获取补码才能获得最终结果

  • 取原码:1111 0110
  • 取反码:1000 1001 (符号位不变,其余各位求反
  • 取补码:1000 1010 (反码+1

最终结果是 1000 1010,也就是-10

————————————————————————————————-

再对正数5按位取反——> (~5),计算步骤如下

原码,反码,补码皆为 0000 0101

对其取反 1111 1010(符号位一起进行取反) 

取反码:1000 0101(符号位不变,其余各位求反)

取补码:1000 0110 (反码+1)

最终结果 1000 0110 ,也就是-6

————————————————————————————————-

下面我们再计算 (~ -10)

-10的原码:1000 1010  

-10的反码:1111  0101 (符号位不变)

-10的补码:1111 0110 (反码+1)

补码取反:0000 1001 (符号位一起取反)

正好得到一个正数,那么对其求原码就可得到最终结果

再因正数的补码,反码,原码都一样,最终结果是 0000 1001 ,正好是9的二进制。

————————————————————————————————-

最后一个有趣的事实是:

1. 所有正整数的按位取反是其本身+1的负数

2. 所有负整数的按位取反是其本身+1的绝对值

3. 零的按位取反是 -1(0在数学界既不是正数也不是负数) 

 

	// 测试-1亿 到 1亿的所有整数  :)
	for (int i = 0; i <= 100000000;++i)
	{
		if (~i != -(i+1) ) {
			__asm { cli	}			// 汇编中断指令
		}

		if (i && ~(-i) != abs((-i)+1) ) {
			__asm { cli	}
		}
	}

 

 

 

 

 

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/125099.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • MyBatis中的JdbcType映射数据类型对照表

    MyBatis中的JdbcType映射数据类型对照表在Mybatisd的映射文件mapper.xml中,会用jdbcType把java类映射数据库表字段,有关映射关系做个笔记,方便查阅。JdbcType Oracle MySql JdbcType ARRAY JdbcType BIGINT BIGINT JdbcType BINARY JdbcType BIT BIT JdbcType BLOB BLOB BLOB J

    2022年10月20日
    1
  • LSTM详解 反向传播公式推导[通俗易懂]

    LSTM详解 反向传播公式推导[通俗易懂]部分参考:http://colah.github.io/posts/2015-08-Understanding-LSTMs/1.结构1.1比较RNN:LSTM:其中的notation:1.2核心思想:LSTM在原来的隐藏层上增加了一个ThekeytoLSTMsisthecellstate,thehorizont…

    2022年5月10日
    45
  • dm268_实时的意思

    dm268_实时的意思最近正好又用到DM368开发板,就将之前做的编解码的项目总结一下。话说一年多没碰,之前做的笔记全忘记是个什么鬼了。还好整理了一下出图像了。不过再看看做的这个东西,真是够渣的,只能作为参考了。项目效果就是,编码encode然后通过rtsp传输在VLC上实时播放。用的是sensor是 MT9P031。一、内核配置让其支持MT9P031二、硬件设计错误排查启动信息错误,无法检测到MT9

    2022年8月13日
    3
  • 一步设置Intellij IDEA 热部署处理方法

    一步设置Intellij IDEA 热部署处理方法  在IntellijIDEA中默认是关闭了自动编译的,可以按照如下1步设置开启自动编译:1.IDEA开启项目自动编译,进入设置,Build,Execut,Deployment-&gt;Compiler勾选中左侧的BuildProjectautomatically   …

    2022年6月1日
    80
  • js获取当前时间(特定的时间格式)[通俗易懂]

    js获取当前时间(特定的时间格式)[通俗易懂]在一个程序中需要对用户的操作进行记录,记录其操作信息,需要对操作进行归类,有时候用户的操作是重复性的操作,那对于重复的操作,也是要区分的,方便查找,可以通过设置类似GUID的唯一值,也可以获取当前的操作时间来区分,因为时间也是唯一的,在任何时候时间都不会出现重复,当然可以获取就可以设置,所以您也可以人为的去设置/修改操作时间。但是这样不好。使用javascript获取时间:在javas…

    2022年9月15日
    2
  • 策略篇&访问策略 ❀ (5.4) 01. Explicit Web Proxy 显式web代理 ❀ 飞塔 (Fortinet) 防火墙

    策略篇&访问策略 ❀ (5.4) 01. Explicit Web Proxy 显式web代理 ❀ 飞塔 (Fortinet) 防火墙【简介】提供代理服务的计算机或其它类型的网络节点称为代理服务器,其具体过程为:客户端首先与代理服务器创建连接,接着发出一个对另外的目标服务器的文件或其它资源的连接请求,代理服务器通过与目标服务器连接或从缓存中取得请求的资源,并返回给客户端。通常在这个过程中,代理服务器可能改变客户端请求或服务器端响应的一些内容以满足各种代理需要。显式代理飞塔防火墙支持支持…

    2022年6月21日
    31

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号