Python示例——负数的位运算
平时在coding的时候虽然会遇到位运算但一般也都是正数的位运算,今天突然见到了使用负数的位运算,对此十分好奇和困惑,为此做了下了解,于是有了此文。
给出一些位运算的例子:

其中,正数的位运算是最为常见的,如:
1<<0
1<<1
1<<2
(1<<0)&2
(1<<1)&2
(1<<2)&2
但是对于负数的位运算还是没有见过的,如:
(1<<0)&-2
(1<<1)&-2
(1<<2)&-2
搜索了下网上的资料,还真有这方面的解释:
答案就是:
其实,负数在进行位运算时是以补码形式参与计算的。
由 https://baike.baidu.com/item/%E8%A1%A5%E7%A0%81/6854613?fr=aladdin 可知:
在计算机系统中,数值一律用补码来表示和存储。原因在于,使用补码,可以将符号位和数值域统一处理;同时,加法和减法也可以统一处理
由 http://c.biancheng.net/view/290.html 可知:
原码:用最高位表示符号位,其余位表示数值位的编码称为原码。其中,正数的符号位为 0,负数的符号位为 1。
正数的原码、反码、补码均相同。
负数的反码:把原码的符号位保持不变,数值位逐位取反,即可得原码的反码。
负数的补码:在反码的基础上加 1 即得该原码的补码。
===============================
扩展的知识:
既然在编程语言中负数是以反码的形式参与计算的,那么负数是以几个字节的形式来表示的呢,不然他的反码形式怎么写呢?
从上面的问题我们又引出了一个新问题,那就是python中int类型数据是几个字节的?
计算机编程基础好的人应该会知道其实在编程语言中int类型字节数不仅与编程语言有关,也与硬件环境有关,比如:C语言的int类型字节数就受计算机硬件平台所影响。但是,在JAVA语言中不论在什么硬件平台上int类型的字节数都是不变的。有关int字节数这个问题,这些年很少有人讨论了,主要原因就是现在大家都是使用X86平台,这样就像C语言这样的编程语言在相同计算硬件下int类型字节数也是一样的。那么,在python语言中这个问题又是如何的呢?
参考:https://blog.csdn.net/weixin_39997173/article/details/109947726
import sys print(type(0), end=' ')
print(sys.getsizeof(0))# 24 print(type(1), end=' ')
print(sys.getsizeof(1))# 28 print(type(2), end=' ')
print(sys.getsizeof(2))# 28 print(type(2**15), end=' ')
print(sys.getsizeof(2**15))# 28 print(type(2**30), end=' ')
print(sys.getsizeof(2**30))# 32 print(type(2**128), end=' ')
print(sys.getsizeof(2**128))# 44
从上面的代码中,我们可以知道,python语言中int类型的字节数是不固定的,具体使用多少字节数是与这个int类型的表示大小有关的,具体解释见:https://blog.csdn.net/weixin_39997173/article/details/109947726
既然我们知道了python中int类型的特性,那么在python中使用int类型进行位运算时我们不需要考虑int类型的表示范围。在参与运算的数为正数情况下,如果二元运算则需要考虑int类型的数值部分最大的那个数的二进制表示,并对数值部分小的那个数进行补0;如果是一元运算也只考虑数值部分的二进制表示,不需要考虑补0操作。如果参与运算的数为负数,先将其当做正数写出原码形式并补0,然后再对补0后的数的符号位置为负并求补码参与运算。
具体操作:(0b代表符号位为正,-0b代表符号位为负)
1. 与、或操作
与操作:
-22 & -7 = -24
-22的补码为-0b01010,7的补码为0b111,由于7的补码位数小于22,补0后得到7的补码为0b00111,因此参与计算的-7的补码为-0b11001,
因此,参与与操作的两个数分别为-0b01010和-0b11001,因此得到运算结果的补码:-0b01000,(符号位参与运算)
对-0b01000求补码得:-0b11000,即-24。
或操作:
-22 | -7 = -24
-22的补码为-0b01010,7的补码为0b111,由于7的补码位数小于22,补0后得到7的补码为0b00111,因此参与计算的-7的补码为-0b11001,
因此,参与或操作的两个数分别为-0b01010和-0b11001,因此得到运算结果的补码:-0b11011,(符号位参与运算)
对-0b11011求补码得:-0b00101,即-5。
2. 异或操作
22 ^ 7 = 17
22的补码为0b10110,7的补码为0b111,由于7为正数并且补码的位数小于22,于是对7的补码进行补0操作,得到:0b00111
因此,参与异或操作的两个数分别为0b10110和0b00111,因此得到运算结果的补码:0b10001,
对0b10001求补码得:0b10001,即17。
-22 ^ -7 = 19
-22的补码为-0b01010,7的补码为0b111,由于7的补码位数小于22,补0后得到7的补码为0b00111,因此参与计算的-7的补码为-0b11001,
因此,参与异或操作的两个数分别为-0b01010和-0b11001,因此得到运算结果的补码:0b10011,(符号位参与运算)
对0b10011求补码得:0b10011,即19。
3. 非操作
~10 = -11
10的补码为0b 1010,对其取反操作得-0b 0101(此过程中符号位参与运算),得到我们想要的运算结果的补码形式;
我们再对-0b 0101求补码即可得到想要的运算结果的原码形式:-0b 1011,即-11 。
~(-2)= 1
-2的补码为-0b 10,对其取反操作得0b 01(此过程中符号位参与运算),得到我们想要的运算结果的补码形式;
我们再对0b 01求补码即可得到想要的运算结果的原码形式:0b 01,即 1 。
~(-3)= 2
-3的补码为-0b 01,对其取反操作得0b 10(此过程中符号位参与运算),得到我们想要的运算结果的补码形式;
我们再对0b 10求补码即可得到想要的运算结果的原码形式:0b 01,即 2 。
4. 移位操作
-7<<2
7的补码为0b 111,因此,-7的补码为-0b 001,对其左移2位得-0b 00100(此过程中符号位不参与运算),得到我们想要的运算结果的补码形式;
我们再对-0b 00100求补码即可得到想要的运算结果的原码形式:-0b 11100,即 -28 。
也可以这样计算:
因为 7<<2 = 28,因此 -7<<2 = -28 。
===============================
在python中由于int的表示范围是不固定的,因此如果参与位运算的int数为负,那么我需要给与其足够大的表示范围才可以,也就是说要对其正数表示下的数补足够的零;不过更为直接的方法就是按照C语言中的标准,对python中参与位运算的数直接使用一个较大的统一的表示范围,如:2**8以内的数用8个数位来表示,2**16以内的数用16个数位来表示,2**32以内的数用32个数位来表示。
----------------------------------------------
Python示例——负数的位运算的更多相关文章
- Java负数的位运算
/** * 求负数的位运算 *///1. -10 >> 2 = ?//2. -10的原码: 1000 0000 0000 0000 0000 0000 0000 1010 最高位代表符号位 ...
- 4.Python 进制和位运算
.button, #logout { color: #333; background-color: #fff; border-color: #ccc; } span#login_widget > ...
- 简简单单学会C#位运算
一.理解位运算 要学会位运算,首先要清楚什么是位运算?程序中的所有内容在计算机内存中都是以二进制的形式储存的(即:0或1),位运算就是直接对在内存中的二进制数的每位进行运算操作 二.理解数字进制 上面 ...
- C 进制 类型说明符 位运算 char类型
一 进制 1. 什么是进制 是一种计数的方式 数值的表示形式 2. 二进制 1> 特点: 只有0和1 逢2进1 2> 书写格式: 0b或者0B开头 3> %d 以带符号的十进制形式输 ...
- 剑指offer用位运算实现两个数相加,及python相关的位操作
题目:写一个函数,求两个整数之和,要求在函数体内不得使用+.-.*./四则运算符号. 代码: # -*- coding:utf-8 -*-class Solution: def Add(self ...
- python位运算
什么是位运算 位运算就是把数字当成二进制来进行计算,位运算有六种:&(与), |(或), ^(异或), ~(非), <<(左移), >>(右移) &(与) &a ...
- Python语言中的按位运算
(转)位操作是程序设计中对位模式或二进制数的一元和二元操作. 在许多古老的微处理器上, 位运算比加减运算略快, 通常位运算比乘除法运算要快很多. 在现代架构中, 情况并非如此:位运算的运算速度通常与加 ...
- Python 进制转换、位运算
一.进制转换 编程用十进制,十进制转换为二进制.八进制.十六进制 In [135]: bin(23) Out[135]: '0b10111' In [136]: oct(23) Out[136]: ' ...
- Python笔记_第一篇_面向过程_第一部分_3.进制、位运算、编码
通过对内存这一个部分的讲解,对编程会有一个相对深入的认识.数据结构是整个内存的一个重要内容,那么关于数据结构这方面的问题还需要对进制.位运算.编码这三个方面再进行阐述一下.前面说将的数据结构是从逻辑上 ...
- 利用位运算进行a+b的计算(Java&&Python)
题目链接 需要用到的位运算操作:异或(^).与(&).右移(<<) 异或运算:又称不进位加法,a^b得到的结果为a与b相加,但是需要进位的地方不进位得到的结果 与运算:找出来a和b ...
随机推荐
- vue安装及升级
先装好note.js 安装过程很简单,一直点下一步就ok了.1.1我们通过打开命令行工具(win+R),输入node -v查看node的版本,若出现相应的版本号说明你安装成功了 1.2.npm包管理器 ...
- IDEA环境编译Spring源码
一.下载源码 1.官网下载 官网地址 如下图 2.github git下载 github地址 如下图 3.gitee git下载(国内推荐使用) gitee地址 如图 查看对应的gradle版本 在下 ...
- nordic——long range测试
简介:本案例测试了long range,注意nrf52系列芯片中,部分硬件是不支持CADE的,也就是不支持long range,如nrf52832就不支持long range.同时协议栈也是部分支持, ...
- mysql数据库慢SQL优化
mysql数据库慢SQL优化优化来源: 阿里云 云数据库RDS 慢sql 或者CAT监控系统中的Transaction SQL or URL根据平均时间反馈来排查,决定是否增加索引,或者调整业务逻辑代 ...
- 关于java-Class类的简单分析
java有个Class的基类,高级编程的时候,务必要理解这个. 粗略看下,可以明白了解Class这个类对于了解和掌握java非常重要. Class这个类的包路径是:java.lang.Class< ...
- ETL服务器连接GaussDB(DWS)集群客户端配置
问题描述:给ETL的服务器上安装gsql的工具,用来连接GaussDB(DWS)集群,做数据抽取用 DWS:GaussDB(DWS) 8.2.1-ESL 1.获取软件包 登录FusionInsight ...
- mermaid语法画图
mermaid 脚本语言 graph TB 从上到下 graph BT 从下到上 graph RL 从右到左 graph LR 从左到右 graph LR; A001-->B001; graph ...
- ubuntu22 flask项目 pyinstaller打包后运行报错: jinja2.exceptions.TemplateNotFound: index.html 的一种解决方案
前言 有一个flask项目a.py, 目录结构如下: |- a.py |- templates | - index.html |- static |- images 运行 python3 a.py可以 ...
- wireshark常用过滤指令
前言 wireshark是一款高效且免费的网络封包分析软件,现就自己使用过的过滤表达式进行记录,随时更新. 正文 与.或.非指令 与:and && 示例:tcp and ip.src ...
- js - 面向对象--手稿