蛮荆

异或运算的妙用

2023-02-01

概述

异或运算 通过对两个相同长度的二进制数进行逐位比较,若对应位的值不同,结果为 1, 否则结果为 0, Go 语言中使用的运算符号为 ^

下面举几个简单的小例子:

0 ^ 0 = 0

0 ^ 1 = 1

1 ^ 0 = 1

1 ^ 1 = 0

图片来源: https://www.build-electronic-circuits.com/xor-gate/

运算定律

交换律

x ^ y = y ^ x

结合律

x ^ (y ^ z) = (x ^ y) ^ z

任何数与自身进行异或运算,结果都为 0

x ^ x = 0

任何数与 0 进行异或运算, 结果为其自身

x ^ 0 = x

有了上面的理论基础之后,下面来看下异或运算的几个应用。


交换两个变量的值

有个经典的笔试题:交换两个变量的值,不能使用第三个变量。

解法一

利用 Go 语言的原生 “骚操作” :

package main

import "fmt"

func main() {
	x, y := 1, 2
	
	x, y = y, x

	fmt.Printf("x = %d, y = %d\n", x, y) 
	// 输出: x = 2, y = 1
}

当然了,如果在真实面试中这样写,只能回去等消息了。

解法二

利用加法分配率

package main

import "fmt"

func main() {
	x, y := 1, 2

	x += y    // x = 3
	y = x - y // y = 1
	x -= y    // x = 2

	fmt.Printf("x = %d, y = %d\n", x, y)
	// 输出: x = 2, y = 1
}

解法三

异或运算性质: 三个值中的任意两个值进行异或运算,都可以得出第三个值。

package main

import "fmt"

func main() {
	x, y := 1, 2

	x ^= y    // x = 3
	y ^= x    // y = 1
	x ^= y    // x = 2

	fmt.Printf("x = %d, y = %d\n", x, y)
	// 输出: x = 2, y = 1
}

加密解密

异或运算可以实现简单高效的加密解密,例如:

  • 明文数据为 text
  • 密钥为 secret
  • 密文数据为 token
# 服务端响应时,返回加密数据

token = text ^ secret

# 服务端接收到请求时,解密数据

text = token ^ secret

这个经典应用在 CSRF 攻击防范 一文中已经讲过,这里不再赘述,感兴趣的读者可以跳转阅读。

数据备份

将数据 x 和数据 y 进行异或运算得到备份数据 z, 然后将三个数据同时保存 (备份数据 z 保存到可用性更高的数据中心),之后不论数据 x 或数据 y 损坏了,都可以根据备份数据 z 进行恢复。

简化计算

在多个值进行计算的场景中,可以根据运算定律简化计算过程:

x ^ y ^ z ^ x ^ y

上述计算式子根据交换律,可以得到:

x ^ x ^ y ^ y ^ z

根据 “任何数与自身进行异或运算,结果都为 0” 运算定律,可以得到:

0 ^ 0 ^ z = z

算法设计

异或运算经常被用于设计高效的数据结构和算法,例如之前的文章提到的 布谷鸟过滤器

LeetCode 原题

给你一个 非空 整数数组 nums ,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。

要求: 必须设计并实现线性时间复杂度的算法来解决此问题,且该算法只使用常量额外空间。

# 示例 1
输入:nums = [2,2,1]
输出:1

# 示例 2
输入:nums = [4,1,2,1,2]
输出:4

# 示例 3
输入:nums = [1]
输出:1

解题代码如下:

// 利用异或运算的性质:
// 1. 任何数和 0 做异或运算,结果仍然是原来的数
// 2. 任何数和其自身做异或运算,结果是 0
// 3. 异或运算满足交换律和结合律,a^b^a = b^a^a = b^(a^a) = b^0 = b
func singleNumber(nums []int) int {
	res := 0
	for _, v := range nums {
		res ^= v
	}
	return res
}

转载申请

本作品采用 知识共享署名 4.0 国际许可协议 进行许可,转载时请注明原文链接,图片在使用时请保留全部内容,商业转载请联系作者获得授权。