进制和之间的转化
进制、位运算
1、什么是进制
1)理解个X进制的概念 :
每⼀位只允许出现 0~X-1 这⼏个数字,逢X进⼀,基是X, 每⼀位有⼀个权值⼤⼩是X的幂次。其表⽰的数值可以写成按位权展开的多项式之和。
⼗进制: 每⼀位只允许出现0~9这⼗个数字,逢⼗进1,基是⼗,每⼀位数字有⼀个权值⼤⼩是⼗的幂次。其表⽰的数值可以写成按位权展开的多项式之和。
⼆进制: 每⼀位只允许出现0~1这⼆个数字,逢⼆进1,基是⼆, 每⼀位数字有⼀个权值⼤⼩是⼆的幂次。其表⽰的数值可以写成按位权展开的多项式之和。
⼋进制:
⼗六进制
2)
假如⽤两个字节表⽰⼀个整数,如下:
⼗进制数字1 的⼆进制表现形式: 0000 0000 0000 0001
⼗进制数字2 的⼆进制表现形式: 0000 0000 0000 0010
如何表⽰⼆进制数的正负?
3)有符号数和⽆符号数的概念
规则:把⼆进制数中的最⾼位(最左边的那位)⽤作符号位
对于有符号数,最⾼位被计算机系统规定为符号位(0为正,1为负)
对于⽆符号数,最⾼位被计算机系统规定为数据位
按照这种说法,⽐如有符号数 +2 -2 的原码形式:
+2 = 0000 0000 0000 0010
-
2 = 1000 0000 0000 0010
真值机器数
+1 = 0000 0000 0000 0001
-1 = 1000 0000 0000 0001负数二进制补码运算法则
-----------------------------------------
1000 0000 0000 0010
-1+1 的结果?
-1+1 = 1000 0000 0000 0010 ----》 -2
不等于0,按理说-1+1等于0才对,为什么会是-2呢?
规则
数字在计算机中,是⽤⼆进制补码的形式来保存的,因此-1 +1需要按照补码进⾏相加才是正确的结果
2、原码、反码、补码
1)如何计算补码?
规则:
正数:原码 = 反码 = 补码
负数:反码 = 符号位不变,其他位取反
补码 = 反码+1
1 的原码:0000 0000 0000 0001
-1的原码:1000 0000 0000 0001
-1的反码:1111 1111 1111 1110
-1的补码:1111 1111 1111 1111
重新计算 -1+1 结果
1111 1111 1111 1111
0000 0000 0000 0001
---------------------------
0000 0000 0000 0000
2)从补码转回原码
负数补码转换原码的规则:
原码 = 补码的符号位不变 -->数据位取反--> 尾+1
-1的补码:1111 1111 1111 1111
取反:1000 0000 0000 0000
-1的原码:1000 0000 0000 0001
【了解】
可以把减法⽤加法来算,只需设计加法器就好了。运算的时候都是⽤补码去运算的。 2-1 = 2+(-1)=0000 0000 0000 0010 +1111 1111 1111 1111
【了解】
为何要使⽤原码, 反码和补码既然原码才是被⼈脑直接识别并⽤于计算表⽰⽅式, 为何还会有反码和补码呢? ⾸先, 因为⼈脑可以知道第⼀位是符号位, 在计算的时候我们会根据符号位, 选择对应加减,但是对于计算机,加减乘数已经是最基础的运算, 要设计的尽量简单。计算机辨别"符号位"显然会让计算机的基础电路设计变得⼗分复杂!于是⼈们想出了将符号位也参与运算的⽅法. 我们知道,根据运算法则减去⼀个正数等于加上⼀个负数, 即: 1-1 = 1 + (-1) = 0 , 所以机器可以只有加法⽽没有减法, 这样计算机运算的设计就更简单了.于是⼈们开始探索将符号位参与运算, 并且只保留加法的⽅法
3. 进制间转换
#10进制转为2进制
>>> bin(10)
'0b1010'
#2进制转为10进制
>>> int("1001",2)
9
#10进制转为16进制
>>> hex(10)
'0xa'
#16进制到10进制
>>> int('ff', 16)
255
>>> int('0xab', 16)
171
#16进制到2进制
>>> bin(0xa)
'0b1010'
>>>
#10进制到8进制
>>> oct(8)
'010'
#2进制到16进制
>>> hex(0b1001)
'0x9'
4. 位运算
看如下⽰例:
如果有⼀个⼗进制数 5,其⼆进制为:0000 0101
把所有的数向左移动⼀位其结果为: 0000 1010
想⼀想:⼆进制 0000 1010 ⼗进制是多少呢其答案为10,有没有发现是5的2倍呢!
再假设有⼀个⼗进制数 3,其⼆进制为:0000 0011
把所有的数向左移动⼀位其结果为: 0000 0110
⼆进制0000 0110 的⼗进制为6,正好也是3的2倍
通过以上2个例⼦,能够看出,把⼀个数的各位整体向左移动⼀个位,就变成原来的2倍
那么在Python中,怎样实现向左移动呢?还有其他的吗
<1>位运算的介绍
& 按位与
| 按位或
^ 按位异或
~ 按位取反
<< 按位左移
>> 按位右移
⽤途: 直接操作⼆进制,省内存,效率⾼
<2>位运算
1)<< 按位左移
各⼆进位全部左移n位,⾼位丢弃,低位补0
x << n 左移 x 的所有⼆进制位向左移动n位,移出位删掉,移进的位补零
【注意事项】
a. 左移1位相当于乘以2
⽤途:快速计算⼀个数乘以2的n次⽅ (8<<3 等同于8*2^3)
b.左移可能会改变⼀个数的正负性
2)>> 右移
各⼆进位全部右移n位,保持符号位不变
x >> n x的所有⼆进制位向右移动n位,移出的位删掉,移进的位补符号位右移不会改变⼀个数的符号
【注意事项】
右移1位相当于除以2
x 右移 n 位就相当于除以2的n次⽅⽤途:快速计算⼀个数除以2的n次⽅ (8>>3 等同于8/2^3)
3)& 按位与
全1才1否则0 :只有对应的两个⼆进位均为1时,结果位才为1,否则为0
⽤6和3这个例⼦。不要⽤9 和13的例⼦
4) | 按位或
有1就1 只要对应的⼆个⼆进位有⼀个为1时,结果位就为1,否则为0
5) ^ 按位异或
不同为1 当对应的⼆进位相异(不相同)时,结果为1,否则为0
6) ~ 取反
~9 = -10
【为什么9取反变成了-10的说明】:
9的原码 ==> 0000 1001 因为正数的原码=反码=补码,所以在真正存储的时候就是0000 1001
接下来进⾏对9的补码进⾏取反操作
进⾏取反==> 1111 0110 这就是对9 进⾏了取反之后的补码
既然已经知道了补码,那么接下来只要转换为咱们⼈能识别的码型就可以,因此按照规则,把这个1111 0110 这个补码转换为原码即可符号位不变,其它位取反==> 1000 1001
然后+1 ,得到原码 =======>1000 1010 这就是 -10
【扩展】
1)任何数和1进⾏&操作,得到这个数的最低位数字&1 = 数字的⼆进制形式的最低位
2)位运算优先级
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论