上次

我在这篇文章中专门讲了带符号数据存储规则。今天再结合原码、反码和补码再来验证一下。

先从二进制规则说起:

原码:

数值的原码如果是正数,原码是它本身,负数第一位为1

比如 4,它的原码是0000 0100 ,-4的原码是1000 0100

反码:

正数的反码是其本身

负数的反码是在其原码的基础上, 符号位不变,其余各个位取反.

4,它的反码是0000 0100 ,-4的反码是1111 1011

补码:

正数的补码就是其本身

负数的补码是在其原码的基础上, 符号位不变, 其余各位取反, 最后+1. (即在反码的基础上+1)

4,它的补码是0000 0100 ,-4的补码是1111 1100

结合上次文章中表格:

大家有没有惊奇的发现,-4的补码是1111 1100 ,这个数值表示252,对应的表格数值正好是252,证明我们前面的猜测完全正确。

结论:计算机内部对数值完全采用补码形式存储。

计算机巧妙地把符号位参与运算, 并且将减法变成了加法, 背后蕴含了化繁为简的巧妙的数学原理

有了补码知识,后面再使用golang的位运算就完全无问题了。

 &            与  AND    两个都为1才为1
  |            或 OR      两个有一个为1才为1
  ^           异或 XOR   两个要相异
  &^         位清空 (AND NOT)  a &^ b == a & (^b),把b按位取非之后和a做与
  

 func main() {
    fmt.Println(6 & 11)   
    fmt.Println(6 | 11)
    fmt.Println(6 ^ 11)
    fmt.Println(6 &^ 11)
}  

6 =0000 0110

11=0000 1011

6 & 11=0000 0010 =2

6 | 11= 0000 1111 =15

6 ^ 11=0000 1101 =13

6 &^ 11=0000 0110 & (^0000 1011)= 0000 0110 & 1111 0100 =0000 0100 =4

用代码验证,完全一致。下面再用负数算一次:

 package main

import (
"fmt"
)

func main() {
fmt.Println(-6 & 11)
fmt.Println(-6 | 11)
fmt.Println(-6 ^ 11)
fmt.Println(-6 &^ 11)
}  

-6 =1111 1010

11 =0000 1011

-6 & 11=0000 1010 =10

-6 | 11= 1111 1011 =-1后 1111 1010取反1000 0101 也就是补码-5

-6 ^ 11=1111 0001 =-1后 1111 0000取反1000 1111 也就是补码-15

-6 &^ 11=1111 1010 & (^0000 1011)= 1111 1010 & 1111 0100 =1111 0000

-1后 1110 1111取反1001 0000 也就是补码-16

运行验证:

结果完全正确。golang作为新语言,基本原理和c语言都是一致的。