题目一:找出数组中重复的数字。
在一个长度为n的数组里的所有数字都在0~n-1的范围内。数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了几次。请找出数组中任意一个重复的数字。例如,如果输入长度为7的数组{2,3,1,0,2,5,3},那么对应的输出是重复的数字2或者3。
答案:
bool duplicate(int numbers[], int length, int *duplication)
{
if(numbers == NULL || length <= 0)
return false;
int temp,i;
for(i=0; i<length; i++)
{
if(numbers[i]<0 || numbers[i]>length-1)
return false;
}
for(i=0; i<length; i++)
{
while(i != numbers[i])
{
if(numbers[i] == numbers[numbers[i]])
{
*duplication = numbers[i];
return true;
}
else
{
temp = numbers[i];
numbers[i] = numbers[temp];
numbers[temp] = temp;
}
}
}
return false;
}
我写的错误代码:
/*
ERR:
1.结尾的return没有写
2.没有限制数组元素在0~n-1范围
3.交换代码出错
*/
bool duplicate_bak(int numbers[], int length, int *duplication)
{
if(numbers == NULL || length <= 0)
return false;
int temp;
for(int i=0; i<length; )
{
if(i != Numbers[i])
{
if(Numbers[i] != Numbers[Numbers[i]])
{
temp = Numbers[i];
Numbers[i] = Numbers[Numbers[i]];
Numbers[Numbers[i]] = temp;
}
else
{
*duplication = Numbers[i];
return true;
}
}
else
{
i++;
}
}
}
题目二:不修改数组找出重复的数字。
在一个长度为n+1的数组里的所有数字都在1~n的范围内,所以数组中至少有一个数字是重复的。请找出数组中任意一个重复的数字,但不能修改输入的数组。例如,如果输入长度为8的数组{2,3,5,4,3,2,6,7},那么对应的输出是重复的数字2或者3。
答案:
int countRange(const int* numbers, int start, int end, int length)
{
if(NULL == numbers)
return -1;
int count=0;
for(int i = 0; i < length; i++)
if(numbers[i] >= start && numbers[i] <= end)
count++;
return count;
}
int duplicate3(const int* numbers, int length)
{
if(NULL == numbers || length <= 0)
return -1;
int start = 1;
int end = length - 1;
int middle;
int count;
while(end >= start)
{
middle = ((end - start)>>1) + start;
count = countRange(numbers, start, middle, length);
if(start == end)
{
if(count > 1)
return start;
else
break;
}
if(count > middle - start + 1)
end = middle;
else
start = middle + 1;
}
return -1;
}
我写的错误代码:
/* {2,3,5,4,3,2,6,7} */
/* 二:2,2,3,3,4 的情况,现有的算法1~2中不能查找出重复,只能找出后面3~4中的重复。具体的需求,时间、空间复杂度要求,都要跟出题的人确认清楚,多交流思考 */
/* 数字在start~end范围内 */
/*
ERR:
1.middle的计算
2.输入数组没有const
3.审题不够清楚,题目假定数组中存在重复数字
4.返回值类型有误
5.+ - 的优先级为4,比<< >> >>>(优先级5)高
*/
int duplicate2(int numbers[], int length, int *duplication)
{
if(NULL == numbers || length <= 0)
return -1;
int start = 1;
int end = length - 1;
int middle;
while(end > start)
{
middle = ((end - start)>>1) + 1;
if (countRange(numbers, start, middle, length) > middle - start + 1)
{
end = middle;
}
else if (countRange(numbers, middle, end, length) > end - middle + 1)
{
start = middle+1;
}
else
{
return false;
}
}
*duplication = start;
return true;
}
测试用例:
1.长度为n的数组里包含一个或多个重复的数字。
2.数组中不包含重复的数字。
3.无效输入测试用例(输入空指针)。