51题
题目:在一个长度为n的数组里的全部数字都在0到n-1的范围内。
数组中某些数字是反复的,但不知道有几个数字反复了。也不知道每一个数字反复的次数。请找出数组中随意一个反复的数字。
比如假设输入长度为7的数组{2,3,1,0,2,5,3},那么相应的输出是反复的数字2或者3.
1.解决问题的一个简单的方法是先把输入的数组排序。从排序的数组中找出反复的数字是件easy的事情,仅仅须要从头到尾扫描排序后的数组就能够了。排序一个长度为n的数组须要时间为O(nlogn)时间。
2.还能够利用哈希表来解决问题。从头到尾按顺序扫描数组中的每一个数。每扫描到一个数字的时候,都能够用O(1)的时间来推断哈希表里是否已经包括了该数字。假设哈希表里没有这个数字。就把它增加到哈希表里。
假设哈希表里已经存在该数字了,那么就找到一个反复的数字。
这个算法的时间复杂度为O(n),但它提高了时间复杂度是一个大小为O(n)的哈希表为代价的。我们再看看有没有时间复杂度为O(1)的算法。
3.我们注意到数组中的数字都在0到n-1的范围内的。假设这个数组中没有反复的数字。那么当数组排序后数字i将出如今下标为i的位置。因为数组中有反复的数字。有些位置可能存在数字,同一时候有些位置可能没有数字。
如今让我们重排这个数组。
从头到尾稻苗这个数组中的每一个数字。当扫描到下标为i的数字的时候。首先比較这个数字(用m表示)是不是i。假设是,接着扫描下一个数字。
假设不是,再拿它和第m个数字进行比較。
假设它和第m个数字相等。就找到一个反复的数字(该数字在下标为i和m的位置都出现了)。假设它和第m个数字不想等,就把第i个数字和第m个数字交换,把m放到属于它的位置。接下来再反复这个比較。交换的过程。直到发现一个反复的数字。
以数组{2,3,1,0,2,5,3}为例来分析找到反复数字的步骤。
数组的第0个数字(从0開始计数,和数组的下标保持一致)是2,与它的下标不想等,于是把它和下标为2的数字1交换。交换后的数组是{1,3,2,0,2,5,3}。此时第0 个数字是1,仍然与它的下标不想等,继续把它和下标为1的数字3交换,得到数组{0。1。2。3,2,5,3}。此时第0 个数字为0。接着扫描下一个数字,在接下来的几个数字中。下标为1。2。3的三个数字分别为1。2,3,他们的下标和数值都分别相等,因此不须要做不论什么操作。接下来扫描下标为4的数字2.因为它的值与它的下标不登,再比較它和下标为2的数字。
注意到此时数组中下标为2的数字也是2,也就是数字2和下标为2和下标4的两个位置了,因此找到一个反复的数字。
Java代码实现例如以下:
/**
* 数组中反复的数字
*/
package swordForOffer;
/**
* @author JInShuangQi
*
* 2015年8月12日
*/
public class E51DuplicationInArray {
public boolean duplicate(int[] arr){
if(arr== null || arr.length <= 0){
return false;
}
for(int i = 0;i<arr.length;i++){
if(arr[i] < 0 || arr[i] > arr.length-1)
return false;
}
for(int i = 0;i<arr.length;i++){
while(arr[i] != i){
if(arr[i] == arr[arr[i]]){
System.out.println(arr[i]);
return true;
}
else{
int temp = arr[i];
arr[i] = arr[temp];
arr[temp] = temp;
}
}
}
return false;
}
public static void main(String[] args){
int[] arr = {2,3,1,0,2,5,3};
E51DuplicationInArray test = new E51DuplicationInArray();
System.out.println(test.duplicate(arr));
}
}
更深层次的:https://blog.csdn.net/ns_code/article/details/27568975
其它:https://blog.csdn.net/qq_21688757/article/details/53958400
《剑指offer》之数组专题
- 此文乃本人把《剑指offer》的算法题都做完之后的总结文章,用于本人复习与同仁们交流。
(一).二维数组中的查找 ok
【题目】在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
(二).二维数组中的查找 ok
【题目】把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。 输入一个非递减排序的数组的一个旋转,输出旋转数组的最小元素。
例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1。
NOTE:给出的所有元素都大于0,若数组大小为0,请返回0。
这道题其实就是有序数组的二分查找问题,以前做过并且整理为二分查找专题:二分查找专题(二)
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
(三).调整数组顺序使奇数位于偶数前面 ok
统计长度,新建数组,产生空间
【题目】输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变。
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
(四).数组中出现次数超过一半的数字 ok
hashmap 或 排序
【题目】数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}。由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2。如果不存在则输出0。
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
(五).数组中重复的数字 ok
【题目】在一个长度为n的数组里的所有数字都在0到n-1的范围内。
数组中某些数字是重复的,但不知道有几个数字是重复的。也不知道每个数字重复几次。请找出数组中任意一个重复的数字。
例如,如果输入长度为7的数组{2,3,1,0,2,5,3},那么对应的输出是重复的数字2或者3。
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
(六).构建乘积数组 ok
【题目】给定一个数组A[0,1,…,n-1],请构建一个数组B[0,1,…,n-1],其中B中的元素B[i]=A[0]A[1]…A[i-1]*A[i+1]…*A[n-1]。不能使用除法。
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
(七).把数组排成最小的数
【题目】输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个。例如输入数组{3,32,321},则打印出这三个数字能排成的最小数字为321323。
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
(八).统计一个数字在排序数组中出现的次数。ok
【题目】统计一个数字在排序数组中出现的次数。
这道题其实就是有序数组的二分查找问题,以前做过并且整理为二分查找专题:二分查找专题(二)
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
(九).数组中只出现一次的数字
【题目】一个整型数组里除了两个数字之外,其他的数字都出现了两次。请写程序找出这两个只出现一次的数字。
此题考查的是位运算!我曾经做过一些位运算题目,整理为位运算专题:位运算专题
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33