我想到的方法是:先对数组排序,如果有重复元素排序后将会相邻。然后相邻元素两两比较,有相等的情况就找到了重复数字。排序一个长度为n的数组时间复杂度为O(nlg n)。
代码如下
public boolean duplicate2(int numbers[],int length,int [] duplication) { if (numbers == null || length == 0){ return false; } Arrays.sort(numbers); for (int i = 0;i < length - 1;i++) { if (numbers[i] == numbers[i + 1]) { duplication[0] = numbers[i]; return true; } } return false; }上面的实现对于任意数组都是通用的。我们要善于抓住题干的已知信息,比如题目中明确说明了长度为n的数组里面的数字全是[0, n-1]之间的。从这句话能获得什么信息呢?如果将数组排好序,那么数组每一个位置都有numbers[i] = i,如果有重复元素,说明[0, n-1]中某些数字空缺了,某些数字有多个。那么必然有某个j !=i的numbers[j] == numbers[i],如果找到满足上述条件的数字,就找到了重复数字。
如果值i没有在正确的位置(满足numbers[i] == i),通过交换两个元素,将值i放到正确的位置。这个过程可以看成是排序。当numbers[i] != i,若令numbers[i] = j,显然j != i;如果此时numbers[j] == numbers[i]说明在与i不同的位置j找到重复元素。否则重复上一步。 package Chap2; import java.util.Arrays; public class FindDuplicate { /** * 推荐的做法,通过交换元素,将值i保存到numbers[i] * 在numbers[i]不和i相等时,如果numbers[i]和numbers[numbers[i]]相等就说明重复元素; * 否则就交换这两个元素,这个过程相当于排序。举个例子,通过交换将2放入numbers[2]。 * @param numbers 待查找的数组 * @param length 数组的长度,其实就是numbers.length * @param duplication 用于保存重复数字,第一个被找到的重复数字存放在duplication[0]中 * @return 如果在数组中有重复元素 */ public boolean duplicate(int numbers[],int length,int [] duplication) { if (numbers == null || length <= 0) { return false; } for (int i = 0;i < length;i++){ if (numbers[i] < 0 || numbers[i] > length -1) { return false; } } for (int i = 0; i< length; i++) { while (numbers[i] != i) { // 现在numbers[i] != i ,设numbers[i] = j,所以如果下面的if成立,就是numbers[i] == numbers[j],说明找到 重复 if (numbers[i] == numbers[numbers[i]]) { duplication[0] = numbers[i]; return true; } swap(numbers, i, numbers[i]); } } return false; } // 交换numbers[i]和numbers[numbers[i]] private void swap(int[] numbers, int p, int q) { int temp = numbers[p]; numbers[p] = numbers[q]; numbers[q] = temp; } }代码中尽管有一个两重循环,但是每个数字只需交换一两次就能被放在正确的位置上,因此总的时间复杂度为O(n)。而且所有操作都在原数组上进行,没有引入额外的空间,空间复杂度为O(1)。
本文参考文献: [1]github.com/haiyusun/data-structures
快乐李同学(李俊德-大连理工大学) 认证博客专家 数据结构 Java Android B站/微博/微信公众号:快乐李同学。大连理工大学软件工程2020毕业学生。大连理工大学2018-2019学年科技创新奖学金。2个国家级项目,2个国家级奖项,5个省级奖项,8个校级奖项(总项目经费和竞赛奖金达2万2千元)。2018-2019年在中国核心期刊《现代计算机》发表2篇项目相关论文,分别署名第一、第二作者(知网可查)。2018-2019年申请2份项目软件著作权,并发布软件(编程乐园、编程学院)到Google,腾讯,百度,华为,小米等应用商店。大学英语六级568分。