操作运算符
位运算
位操作 | 含义 | 备注 | |
---|---|---|---|
& | 按位与 | 两个操作数对应的二进制位,都为 1 则为 1,否则为 0 | |
| | 按位或 | 两个操作数对应的二进制位,有 1 则为 1,否则为 0 | |
~ | 按位非 | 两个操作数对应的二进制位,取反操作,1 则为 0,0 则为 1 | |
^ | 异或 | 两个操作数对应的二进制位,相同为 0,相异为 1 | |
<< | 左移 | 将二进制位全部左移 n 位,相当于乘以 2 的 n 次方,如 1<<6 相当于 1×64=64 | |
>> | 右移 | 将二进制位全部右移 n 位,相当于除以 2 的 n 次方,如 64>>3 相当于 64÷8=8 | |
常见操作
-
快速交换 2 个数字值
var a = 5 var b = 99 a = a ^ b b = a ^ b a = a ^ b fmt.Println(a == 99) fmt.Println(b == 5)
任何数与自己异或为 0,与 0 异或为自己。
a=a^b;
b=a^b; (把 a=a ^ b 代入 b=a ^b ^b , b=a ^b ^b ,b=a ^0,b=a)
a=a^b; (把 a=a ^ b 代入 a=a ^b ^b ,a=a ^b ^a ,a=b ^0,a=b) -
判断奇偶数
var a = 10 var b = 11 a & 1 = 0 b & 1 = 1
原理:
任何偶数的二进制的第一位都是 0 如 2 ==> 0010
任何奇数的二进制的第一位都是 1 如 1 ==> 0001
所以只需要 按位与 1 判断最后一位 是 0 还是 1 即可
-
乘除法运算
乘数和除数必须是 2 的 n 次方才可以,利用左右移的原理
var a = 4 var b = 16 a = a * 4 b = b / 4 // 可以改为: a = a << 2 b = b >> 2
说明:
除 2 = 右移 1 位 乘 2 = 左移 1 位
除 4 = 右移 2 位 乘 4 = 左移 2 位
除 8 = 右移 3 位 乘 8 = 左移 3 位 -
交换符号
var a = 1 // 0001 a = ~a + 1 // 1110 fmt.Println(a == -1) a = ~a + 1 fmt.Println(a == 1)
整数取反加 1,正好变成其对应的负数(补码表示);负数取反加一,则变为其原码,即正数
-
判断一个数是正负数
整数右移 31 位得到 0,负数右移 31 位得到 -1
var a = 10 var b = -10 if a>>31 == 0 { fmt.Println("a是正数") } if b>>31 == -1 { fmt.Println("b是负数") }
-
求绝对值
整数的绝对值是其本身,负数的绝对值正好可以对其进行取反加一求得,即我们首先判断其符号位,然后根据符号进行相应的操作
var a = 10 if a>>31 == 0 { // 整数直接返回 return a } // 负数 a = ^a + 1
-
高低位交换
// 34520的二进制表示: // 10000110 11011000 // 将其高8位与低8位进行交换,得到一个新的二进制数: // 11011000 10000110 // 其十进制为55430 var a = 34520 a = (a >> 8) | (a << 8)
从上面移位操作我们可以知道,只要将无符号数 a>>8 即可得到其高 8 位移到低 8 位,高位补 0;将 a<<8 即可将 低 8 位移到高 8 位,低 8 位补 0,然后将 a>>8 和 a<<8 进行或操作既可求得交换后的结果。
-
二进制逆序