题目一 :找出数组中重复的数字
在一个长度为n的数组里的所有数字都在0~n-1的范围内。数组中某些数字是重复的,但是不知道有几个数字重复了,也不知道每个数字重复了几次。请找出数组中任意一个重复的数字。例如,如果输入长度为 7 的数组{2,3,1,0,2,5,3},那么对应的输出是重复的数组2或者3。
第一种解决方法:利用哈希表来解决这个问题 (时间复杂度O(n), 空间复杂度O(n))
分析理解:
* 类似于公司在一张纸(类似于hasNumber[]= {false})上签名领礼品,每个人只能领一份, * 当你领完之后在纸上签上自己(M)的名字(hasNumber[M] = true); * 当有一个与你同名(M)的人领取时, 会发现你(M)已经在纸上签名(hasNumber[M] = true), * 所以他因为出现重复名字就不能领取礼品,同样也验证这个人是与你同名的
/**
* 哈希表的方式
* 分析:
* 时间复杂度---最大循环次数由数组长度决定,执行了n次循环,时间复杂度O(n)
* 空间复杂度---由于hasNumber[]开辟了一个数组长度(n)的空间,空间复杂度O(n)
*
* ------该方法提高时间效率是以一个空间复杂度为O(n)的哈希表为代价的
*
* 理解:
* 类似于公司在一张纸(类似于hasNumber[]= {false})上签名领礼品,每个人只能领一份,
* 当你领完之后在纸上签上自己(M)的名字(hasNumber[M] = true);
* 当有一个与你同名(M)的人领取时, 会发现你(M)已经在纸上签名(hasNumber[M] = true),
* 所以他因为出现重复名字就不能领取礼品,同样也验证这个人是与你同名的
*
* @param array 目标数组
* @return 返回数组中第一个重复数字,返回-1 则表明没有重复数字
*/
public static int getDupliteNumber(int[] array) {
if (array == null || array.length == 0) {
// 如果目标数字不存在,则不会有重复数字
return -1;
}
/**
* 开辟一个空间长度为数组长度的哈希表, 此哈希表中初始化的默认值都为false
* 1. 因为数组中的值在[0, n-1]的范围内,数组长度为 n
* 所以 array[i]>= 0 && array[i] <= n-1
* 所以 hasNumber[array[i]] 不会出现数组越界
*
* 2. hasNumber 的作用就是记录哪些数字已经出现过
* 例如: 假设 m = array[i]
* hasNumber[m] = true, 代表该数字曾经出现过
* hasNumber[m] = false, 代表该数字没有出现过
*/
boolean hasNumber[] = new boolean[array.length];
for (int i = 0; i < array.length; i++) {
if (hasNumber[array[i]]) {
// 说明该数字曾经出现过,是重复数字,则返回该重复数字
return array[i];
} else {
// 如果该数字没有出现过,则将其设置为true(代表已出现)
hasNumber[array[i]] = true;
}
}
return -1;
}
每个人的理解方式可能不同,但是最基本的思想是一致的!死记代码,面试背锅。