数据结构与算法:异或的妙用(找出一堆数中唯一/唯二出现奇数次的项、swap的优化)

本文主要是介绍数据结构与算法:异或的妙用(找出一堆数中唯一/唯二出现奇数次的项、swap的优化),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

异或的本质是相同为0,不同为1,实际上也就是无进位加法
那么对于任何数x和y,以下公式永远成立

x^0=x
x^x=0
(x^y)^x=x^(y^x)=(x^x)^y=0^y=y//交换律和结合律

也就是说一堆数里面,出现偶数次的项先自己异或,那么就会得0,然后再和出现奇数次的项进行异或,那么最终结果就是自己本身

找出一堆数中唯一/唯二出现奇数次的项

找出一堆数中唯一出现奇数次的项

如果一堆数中,只有一个数出现了奇数次,其他数都出现了偶数次,如何快速的找出这个数。
我们只需要将这堆数逐一异或,那么最后出来的结果就是这个数。
时间复杂度为O(N),空间复杂度为O(1)

public static void printOddTimesNum1(int[] arr){int eor=0;for(int cur:arr){eor^=cur;}System.out.println(eor);
}

找出一堆数中唯二出现奇数次的项

如果一堆数中,只有a和b出现了奇数次,其他数都出现了偶数次,如何快速的找出a和b。
(不需要找出a和b分别是多少,但是要找出a和b这一对)

我们只需要将这堆数逐一异或,那么最后出来的结果就是a^b,记作eor
在eor的二进制表示中,一定至少存在一位为1,因为a和b不相同,那么异或出来的二进制至少有一位为1,我们记这位为第x位。
那么要么是a在这第x位上为1,要么是b在这第x位上为1.在这我们就假设是a为1。
且一堆数中出现偶数次的项也分为两大部分,要么在这第x位上为1,要么在这第x位上为0。
无论怎样,这一堆数整体会分为两大部分,在这第x位上为1的分类A(包含a),和在这第x为上为0的分类B
我们取这最右侧一位1和分类A的数逐一异或,得到的就是第x位上为1的a(因为假设a在第x位上为1)然后a再与eor异或,就得到b了。
时间复杂度为O(N),空间复杂度为O(1)

public static void printOddTimesNum2(int[] arr){int eor=0;for(int i=0;i<arr.length;i++){eor^=arr[i];}int rightOne=eor&(~eor+1);int onlyOne=0;for(int cur:arr){if((cur&rightOne)==1){//也可以改为==0,那么最终得到的就是分类BonlyOne^=cur;}}System.out.println(onlyOne+" "+(eor^onlyOne));}

具体如何提取该数二进制最右侧的1

int a
int rightone = a & (~a+1)

swap的优化

往常swap的代码为:

int temp=a;
a=b;
b=temp;

但是通过异或操作,我们可以省去这额外的一个常数空间。代码如下:

a=a^b;
b=a^b;
a=a^b;

但是注意,a和b一定要是不同的地址,否则就相当于自己和自己异或,会直接变为0

所以,假设a原本存储着信息x,b原本存储着信息y
a=a^b;即,a变为x异或y,b仍为y
b=a^b;即,b变为(x异或y)异或y,即b变为x,a仍为x异或y
a=a^b;即,a变为(x异或y)异或x,即a变为y,b仍为x

本篇文章还是不懂的话可以去看https://www.bilibili.com/video/BV13g41157hK

这篇关于数据结构与算法:异或的妙用(找出一堆数中唯一/唯二出现奇数次的项、swap的优化)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/542837

相关文章

golang字符串匹配算法解读

《golang字符串匹配算法解读》文章介绍了字符串匹配算法的原理,特别是Knuth-Morris-Pratt(KMP)算法,该算法通过构建模式串的前缀表来减少匹配时的不必要的字符比较,从而提高效率,在... 目录简介KMP实现代码总结简介字符串匹配算法主要用于在一个较长的文本串中查找一个较短的字符串(称为

通俗易懂的Java常见限流算法具体实现

《通俗易懂的Java常见限流算法具体实现》:本文主要介绍Java常见限流算法具体实现的相关资料,包括漏桶算法、令牌桶算法、Nginx限流和Redis+Lua限流的实现原理和具体步骤,并比较了它们的... 目录一、漏桶算法1.漏桶算法的思想和原理2.具体实现二、令牌桶算法1.令牌桶算法流程:2.具体实现2.1

Go语言中三种容器类型的数据结构详解

《Go语言中三种容器类型的数据结构详解》在Go语言中,有三种主要的容器类型用于存储和操作集合数据:本文主要介绍三者的使用与区别,感兴趣的小伙伴可以跟随小编一起学习一下... 目录基本概念1. 数组(Array)2. 切片(Slice)3. 映射(Map)对比总结注意事项基本概念在 Go 语言中,有三种主要

Deepseek使用指南与提问优化策略方式

《Deepseek使用指南与提问优化策略方式》本文介绍了DeepSeek语义搜索引擎的核心功能、集成方法及优化提问策略,通过自然语言处理和机器学习提供精准搜索结果,适用于智能客服、知识库检索等领域... 目录序言1. DeepSeek 概述2. DeepSeek 的集成与使用2.1 DeepSeek API

Tomcat高效部署与性能优化方式

《Tomcat高效部署与性能优化方式》本文介绍了如何高效部署Tomcat并进行性能优化,以确保Web应用的稳定运行和高效响应,高效部署包括环境准备、安装Tomcat、配置Tomcat、部署应用和启动T... 目录Tomcat高效部署与性能优化一、引言二、Tomcat高效部署三、Tomcat性能优化总结Tom

解读Redis秒杀优化方案(阻塞队列+基于Stream流的消息队列)

《解读Redis秒杀优化方案(阻塞队列+基于Stream流的消息队列)》该文章介绍了使用Redis的阻塞队列和Stream流的消息队列来优化秒杀系统的方案,通过将秒杀流程拆分为两条流水线,使用Redi... 目录Redis秒杀优化方案(阻塞队列+Stream流的消息队列)什么是消息队列?消费者组的工作方式每

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

一文带你理解Python中import机制与importlib的妙用

《一文带你理解Python中import机制与importlib的妙用》在Python编程的世界里,import语句是开发者最常用的工具之一,它就像一把钥匙,打开了通往各种功能和库的大门,下面就跟随小... 目录一、python import机制概述1.1 import语句的基本用法1.2 模块缓存机制1.