一个无序数组里有99个不重复的正整数,范围从1-100,唯独缺少一个赠书,如何找出这个缺失的整数?
解法一:创建一个HashMap,以1-100为键,值都是0,然后遍历整个数组,每读到一个整数,就找到HashMap当中对应的键,让其值加一。
由于数组缺少一个整数,最终一定有99个键对应的值等于1,剩下一个键对应的值为0,遍历修改后的HashMap,找到这个值为0的键。
分析:假设数组长度是N,那么该解法的时间复杂的是O(N),空间复杂度是O(N)。
此解法应该是大多数人能想到的方法,但解法虽然在时间上是最优的,但额外开辟了空间。那么怎么样能降低空间复杂度呢?
解法二:先把数组元素进行排序,然后遍历数组,检查任意俩个相邻元素值是否连续,如果不连续,则中间缺少的整数就是所要寻找的;如果全部连续,则缺少的整数就是1或100.
分析:假设数组长度是N,如果用时间复杂度为O(N*LogN)的排序算法进行排序,那么该解法的时间复杂度是O(N*logN),空间复杂度是O(1).
OK,这种解法没有开辟额外空间但是时间复杂度又大了。那有没有办法让时间和空间都优化麽?
解法三:很简单也很高效的方法,先算出1+2+3+...+100的和,然后依次减去数组里的元素,最后得到的差,就是唯一缺失的整数
分析:假设数组长度是N,那么该解法的时间复杂度是O(N),空间复杂度是O(1).
题目扩展:一个无序数组里有若干个正整数,范围1-100,其中99个正整数都出现了偶数次,只有一个整数出现了奇数次(比如1,1,2,2,3,3,4,5,5),如何找到这个出现奇数次的整数?
分析:如果继续按刚才的求和方法肯定不行,因为根本不知道每个整数出现的次数,同时又要保证时间空间最优,所以此时我们想到用异或运算(在位运算时相同位结果为0,不同为1)
解法:遍历整个数组,依次做异或运算,由于异或在位运算时相同为0不同为1,因此所有出现偶数次的整数都会相互抵消变成0,只有唯一出现奇数次的整数会被留下。
分析:假设数组长度是N,那么该解法的时间复杂度是O(N),空间复杂度是O(1).
题目第二次扩展:一个无序数组里有若干个正整数,范围从1-100,其中98个整数都出现了偶数次,只有俩个整数出现了奇数次(比如1,1,2,2,3,4,5,5),如何找到这个出现奇数次的整数?
分析:这次要找俩个赠书,刚才的方法以及不够用了。把数组所有元素异或,最终只会得到俩个整数的异或结果...于是此时我们想到分治法,如果把数组分成俩部分,每一部分包含一个奇数次整数,这样就回归到上一题的情况。
解法:遍历整个数组,依次做异或运算。由于数组存在俩个出现奇数次的整数,所以最终异或的结果,等同于这俩个整数的异或结果,这个结果中,至少会有一个二进制位是1(如果都是0,说明俩个书相等,和题目不符)。
举个例子:如果最终异或结果是5,转换为二进制是00000101.此时我们可以选择任意一个是1的二进制位来分析,比如末尾是1,说明A和B转为二进制的末位不同,必定其中一个整数的末位是1,另一个整数末位是0.
根据这个结论,我们可以把原数组按照二进制的末位不同分为倆部分,一部分是末位是1,一部分末位是0.由于A和B的末位不用,所以A在其中一部分,B在另一部分,绝不会出现A和B在同一部分,另一部分没有的情况。
分析:假设数组长度是N,那么该解法的时间复杂度是O(N),把数组分成俩部分,并不需要借助额外存储空间,完全可以在按二进制位分组的同时来做异或运算,所以空间复杂度仍然是O(1)
以上分享自算法爱好者