T3:数组中重复的数字

题目一:找出数组中重复的数字。

在一个长度为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.无效输入测试用例(输入空指针)。

你可能感兴趣的:(Code,算法)