一. 什么是异或?

1. Wikipedia的解释:

在逻辑学中,逻辑算符异或(exclusive or)是对两个运算元的一种逻辑析取类型,符号为 XOR 或 EOR 或 ⊕(编程语言中常用^)。但与一般的逻辑或不同,异或算符的值为真仅当两个运算元中恰有一个的值为真,而另外一个的值为非真。转化为命题,就是:“两者的值不同。”或“有且仅有一个为真。”

2. 定义

1 ⊕ 1 = 0

0 ⊕ 0 = 0

1 ⊕ 0 = 1

0 ⊕ 1 = 1

3. 真值表

Y B = 0 B = 1
A = 0 0 1
A = 1 1 0

4, 表达式:

Y = A ’ ⋅ B + A ⋅ B ’ Y = A’ · B + A · B’ Y=A’⋅B+A⋅B’

解释:我使用·作为与,我使用+作为或,我使用’作为否(本来应该使用头上一横,但是太难编辑了,就使用了’);


二. 异或有什么特性?

根据定义我们很容易获得异或两个特性:

恒 等 律 : X ⊕ 0 = X 恒等律:X ⊕ 0 = X 恒等律:X⊕0=X

归 零 律 : X ⊕ X = 0 归零律:X ⊕ X = 0 归零律:X⊕X=0

然后我们使用真值表可以证明:

(1)交换律

A ⊕ B = A' · B + A · B'
B ⊕ A = B' · A + B · A'

因为·与和+或两个操作满足交换律,所以:

A ⊕ B = B ⊕ A A ⊕ B = B ⊕ A A⊕B=B⊕A

(2)结合律

(A ⊕ B) ⊕ C
= (A' · B + A · B') ⊕ C
= (A' · B + A · B')' · C + (A' · B + A · B') · C '
= ((A' · B)' · (A · B')')· C + A' · B · C ' + A · B' · C '
= ((A + B') · (A' + B))· C + A' · B · C ' + A · B' · C '
= (A · B + A' · B') · C + A' · B · C ' + A · B' · C '
= A · B · C + A' · B' · C + A' · B · C ' + A · B' · C '

你可以使用同样推导方法得出(请允许我偷懒一下,数学公式敲起来不容易 +_+):

A ⊕ (B ⊕ C)
= A · B · C + A' · B' · C + A' · B · C ' + A · B' · C '

证明过程中使用了如下几个方法(·与 +或 '否):

·与 +或交换律:

A · B = B · A
A + B = B + A

·与 +或结合律:

(A · B) · C = A · (B · C)
(A + B) + C = A + (B + C) 

·与 +或分配律:

A · (B + C)= A · B + A · C
A + B · C = (A + B) · (A + C)

摩尔定理

(A · B)' = A' + B'
(A + B)' = A' · B'

结论:

交换律:A ⊕ B = B ⊕ A 结合律:A ⊕ (B ⊕ C) = (A ⊕ B) ⊕ C

有了归零率和结合律,我们就可以轻松证明:

自反:A ⊕ B ⊕ B = A ⊕ 0 = A

可能这些特性会很顺其自然的理解,但是如果你在解决问题的时候,你可能会忘记异或的这些特性,所以适当的应用可以让我们加深对异或的理解;

A ⊕ 1 = A';
A ⊕ 0 = A;
A ⊕ A = 0;
A ⊕ A' = 1;

异或有什么神奇之处(应用)?

说明:以下的的异或全部使用符号^

可能你已经被乱七八糟的公式和演算搞的有点烦了,不就是很简单的异或运算吗?还解释的那么复杂,嘿嘿,不要着急,打好了基础,你就站在了巨人的肩膀,让我们开始异或的神奇之旅吧;

(1)快速比较两个值

先让我们来一个简单的问题;判断两个int数字a,b是否相等,你肯定会想到判断a - b == 0,但是如果判断a ^ b == 0效率将会更高,但是为什么效率高呢?就把这个给你当家庭作业吧。


(2)我们可以使用异或来使某些特定的位翻转,因为不管是0或者是1与1做异或将得到原值的相反值;

0 ^ 1 = 1

1 ^ 1 = 0

例如:翻转10100001的第6位, 答案:可以将该数与00100000进行按位异或运算;10100001 ^ 00100000 = 10000001


(3)我们使用异或来判断一个二进制数中1的数量是奇数还是偶数

例如:求10100001中1的数量是奇数还是偶数; 答案:1 ^ 0 ^ 1 ^ 0 ^ 0 ^ 0 ^ 0 ^ 1 = 1,结果为1就是奇数个1,结果为0就是偶数个1; 应用:这条性质可用于奇偶校验(Parity Check),比如在串口通信过程中,每个字节的数据都计算一个校验位,数据和校验位一起发送出去,这样接收方可以根据校验位粗略地判断接收到的数据是否有误


(4)校验和恢复

校验和恢复主要利用的了异或的特性:IF a ^ b = c THEN a ^ c = b 应用:一个很好的应用实例是RAID5,使用3块磁盘(A、B、C)组成RAID5阵列,当用户写数据时,将数据分成两部分,分别写到磁盘A和磁盘B,A ^ B的结果写到磁盘C;当读取A的数据时,通过B ^ C可以对A的数据做校验,当A盘出错时,通过B ^ C也可以恢复A盘的数据。


(5)经典题目:不使用其他空间,交换两个值

a = a ^ b;
b = a ^ b; //a ^ b ^ b = a ^ 0 = a;
a = a ^ b;

原理:
第一步没啥好说a = a^b
第二步:b=b^a ,也就是 b=b^a^b ,也就是b=a^0,此处换值
第三步:a=a^b 也就是a=a^b^a,也就是b

(6)最最常出现的面试题:一个整型数组里除了N个数字之外,其他的数字都出现了两次,找出这N个数字;

比如,从{1, 2, 3, 4, 5, 3, 2, 4, 5}中找出单个的数字: 1

让我们从最简单的,找一个数字开始;

题目:(LeetCode 中通过率最高的一道题) Single Number: Given an array of integers, every element appears twice except for one. Find that single one. Note:Your algorithm should have a linear runtime complexity. Could you implement it without using extra memory? 思路: 拿到这个题目,本能的你会使用排序(数字文字我们常常需要排序),排序后可以来判断是否数字成对出现,思路很明显,但是排序的算法上限是 O(nlogn),不符合题目要求;

学习了强大的异或,我们可以轻松的使用它的特性来完成这道题目:
(1)A ^ A = 0;
(2)异或满足交换律、结合律; 所有假设有数组:A B C B C D A使用异或:

A ^ B ^ C ^ B ^ C ^ D ^ A
= A ^ A ^ B ^ B ^ C ^ C ^ D
= 0 ^ 0 ^ 0 ^ D
= 0 ^ D
= D

是不是很神奇?时间复杂度为O(n),当然是线性的,空间复杂度O(1);
代码:

class Solution {public:int singleNumber(int A[], int n) {//特殊情况1,2if(n<=0) return -1;if(n==1) return A[0];int result = 0;for (int i = 0; i < n; i ++) {result = result ^ A[i];}return result;}
};

接下来让我们增加一些难度:

题目:一个整型数组里除了两个数字之外,其他的数字都出现了两次。请写程序找出这两个只出现一次的数字?

思路: 第一步:肯定还是像我们上面的解法一样,所有数进行异或,不过最终得到的结果是 a 和 b(假设 a 和 b 是落单的数字)两个值的异或结果 a^b,没有直接得到 a 和 b 的值;

第二步:想办法得到 a 或者 b,假设 结果 为 00001001(F肯定不为0),根据结果 的值我们发现,如果某一位的值为1,则在两个出现一次的数字中,在这一位上,一定一个是1,一个是0

进而可以将该整型数组分为两组, 第一组的数在这一位上的值为1,第二组的数在这一位上的值为0

进而可以推出:第一组的数中包含一个只出现一次的数, 第二组的数中包含一个只出现一次的数。

最后将第一组的数全部异或,得到的值就是第一组中只出现一次的数,第二组中的数同理。 最后得到了两个只出现一次的数。

这样我们的时间复杂度是 O(n),空间复杂度是 O(1) 代码:

#include <iostream>
#include <assert.h>
using namespace std;//输出 num 的低位中的第一个 1 的位置
int getFirstOneBit(int num) { return num & ~(num - 1); // num 与 -num 相与找到
}void findTwo(int *array, int length){int res = 0;int firstOneBit = 0;int a = 0;int b = 0;for (int i = 0; i < length; i++) {res ^= array[i];}assert(res != 0); //保证题目要求,有两个single的数字firstOneBit = getFirstOneBit(res);for (int i = 0; i < length; ++i) {if(array[i] & firstOneBit) {a ^= array[i];}}b = res ^ a;cout << "a: " << a << endl;cout << "b: " << b << endl;
}int main() {int array1[] = {2, 5, 8, 2, 5, 8, 6, 7};findTwo(array1, 8);return 0;
}

力扣题目链接:https://leetcode-cn.com/problems/single-number-iii/

史上最通俗易懂的异或运算详解【含例题及应用】相关推荐

  1. 史上最小白之CNN 以及 TextCNN详解

    本小白是一名立志从事NLP的菜鸟,本来只准备写一篇TextCNN来加深自己理解地,但想要了解TextCNN那必然需要了解CNN的原理,写的过程中突然想起了自己在学习时各种看博客的心路历程,看一篇博客要 ...

  2. 史上最经典垃圾回收器(CMS,G1)详解、适用场景及特点、使用命令

    文章目录 垃圾收集器介绍总结 各个垃圾收集器之间的关系 垃圾收集器使用命令及默认值 详解各个垃圾收集器 Serial ParNew Parallel Scavenge Serial Old Paral ...

  3. 史上最全互联网八大技术岗位详解

    "互联网技术岗位详解,涉及到前段开发.后端开发.移动端开发.大数据.项目管理.测试.运维.技术管理等八大领域. 架构师 每个产品线都有架构师,在技术平台部门也需要技术平台的架构师. 架构师负 ...

  4. 项目管理学习总结(9)——史上最全互联网八大技术岗位详解

    互联网技术岗位详解,涉及到前段开发.后端开发.移动端开发.大数据.项目管理.测试.运维.技术管理等八大领域. 架构师 每个产品线都有架构师,在技术平台部门也需要技术平台的架构师. 架构师负责设计系统整 ...

  5. 史上最全Android build.gradle配置详解

    Android Studio是采用gradle来构建项目的,gradle是基于groovy语言的,如果只是用它构建普通Android项目的话,是可以不去学groovy的.当我们创建一个Android项 ...

  6. 史上最强Java学习路线(详解)

    Java是一个通用的编程语言,其实可以干很多事,怎么学Java就看怎么用了,但有一些一般的步骤: 熟悉一种文本编辑器,比如:Vim, Emacs, Notepad++, TextMate等.知道哪些是 ...

  7. 史上最强C语言教程----操作符详解

    目录 1.操作符分类 2.算术操作符 3.移位操作符 3.1 左移操作符 3.2 右移操作符 4.位操作符 5.赋值操作符 赋值操作符 复合赋值符 6.单目操作符 6.1 单目操作符介绍 6.2 si ...

  8. 网络协议分析 | 传输层 :史上最全UDP、TCP协议详解,一篇通~

    文章目录 UDP 概念 格式 UDP如何实现可靠传输 基于UDP的应用层知名协议 TCP 概念 格式 保证TCP可靠性的八种机制 确认应答.延时应答与捎带应答 超时重传 滑动窗口 滑动窗口协议 后退n ...

  9. rem适配的浏览器_[史上最全]UI相关尺寸单位详解 | px、pt、dp、sp、rem、vwvh、rpx、ppi、dpi、dppx...

    先给进来看文章的你点个赞 尺寸适配应该由开发同事负责处理,处理不好是他的问题,你有兴趣了解这些让人头疼的事,证明你是共产主义好社畜,还有工作量不饱和,Good for you~适配问题是影响设计复现的 ...

最新文章

  1. 成功解决Exception “unhandled ImportError“cannot import name ‘imread‘ from ‘scipy.misc‘
  2. adf 自动输稿器_在ADF实体PK属性中使用MySQL自动增量PK列
  3. ospf工作原理_OSPF动态路由配置经典案例
  4. leetcode 的shell部分4道题整理
  5. 数组名与指向数组的指针之间的联系与区别【数据结构】
  6. Linux内核--基于Netfilter的内核级包过滤防火墙实现
  7. 风变Python编程13类的学习2
  8. appnium连接夜神模拟器
  9. MySQL的LIKE模糊查询优化
  10. 作词家下岗系列:教你用 AI 做一个写歌词的软件!
  11. python预测控制_【模型工具】耦合python和 SWMM的城市排水系统模型预测算法
  12. vs2010 正式版官方下载地址
  13. ROI Pooling和ROI Align、ROI Warp解析
  14. 决策表是什么?怎么使用决策表?
  15. 数据处理案例一之求同存异
  16. OLTP OLAP
  17. 如何分析linux下的几种目标文件
  18. armadillo matlab,科学网—C++下媲美MATLAB矩阵运算的Armadillo 库 - 吴泓润的博文
  19. XCode使用一:Xcode基本操作
  20. 2030年销售额突破200亿美元!瑞萨电子揭秘智能汽车版图

热门文章

  1. 时隔两年,LitePal终于又更新了!
  2. DRAM知识整理系列(二):DRAM状态转换图与命令(真值表)相关整理
  3. 助力学生自主学习,实现因材施教,是人工智能赋能教育的价值所在
  4. Good Morning Beijing,Good Night Shanghai
  5. 惰性求值 php,使用 JavaScript 进行函数式编程 (一) 翻译
  6. 15个Vue自定义指令,让你的项目开发爽到爆
  7. 关于计算机游戏的英语读法,电脑游戏,computer game,音标,读音,翻译,英文例句,英语词典...
  8. auc计算公式_图解机器学习的准确率、精准率、召回率、F1、ROC曲线、AUC曲线
  9. [Qt5布局] 控件自动填满所在布局框架
  10. 做跨境电商,人工客服成本太大怎么办?教你这一招!