剑指offer——数组中重复的数字

题目描述

在一个长度为n的数组里的所有数字都在0到n-1的范围内。 数组中某些数字是重复的,但不知道有几个数字是重复的。也不知道每个数字重复几次。请找出数组中任意一个重复的数字。 例如,如果输入长度为7的数组{2,3,1,0,2,5,3},那么对应的输出是第一个重复的数字2。

(类似题目不修改数组找出重复的数字)

解题思路:

这题常规思路很多,打表(需要O(n)的空间)、排序(O(nlogn)的时间复杂度)都能做。

现在学习这种新的解法

因为题目中是长度为n,而且数组内的数字都在0到n-1范围,所以,如果他们是有序且不重复的,那么数组中的值和下标是一样的。如果有重复,则可能某一下标会对应几个相同的值,而有的下标则没有对应的值。

 

算法流程:

从无序数组开始,每次比较当前下标与该下标的值是否相等,

如果不相等 , 比较该值 V 与 numbers[V] 是否相等

          若相等,便找到第一个重复的值。

          若不相等, 则将这个下标对应的值 V 与下标为 v 的那个值交换,这样numbers[v] = v,交换过后再次比较当前下标与该下            标的值是否相等,如果不等重复上述操作知道这个位置的下标与值相等;

如果相等,则再比较下一位置的情况。

 

这个解法不需要额外的空间复杂度,也比排序要快。

class Solution {
public:
    // Parameters:
    //        numbers:     an array of integers
    //        length:      the length of array numbers
    //        duplication: (Output) the duplicated number in the array number
    // Return value:       true if the input is valid, and there are some duplications in the array number
    //                     otherwise false
    bool duplicate(int numbers[], int length, int* duplication) {
        for(int i = 0 ; i < length ; i++)
        {
            if(numbers[i] == i) continue;
        	else
			{
	            int temp = numbers[i];
	            if(temp == numbers[temp])
	            {
	                *duplication = numbers[i];
	                return true;
	            }
	            else
	            {
	                if(temp < length)
	                {
	                    numbers[i] = numbers[temp];
	                    numbers[temp] = temp;
	                } 
	                i--;                  
	            }         
	        } 
        }
        return false;
    }
};

标准代码: 

bool duplicate(int numbers[], int length, int* duplication)
{

    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)
        {
            if(numbers[i] == numbers[numbers[i]])
            {
                *duplication = numbers[i];
                return true;
            }

            // 交换numbers[i]和numbers[numbers[i]]             
            int temp = numbers[i];
            numbers[i] = numbers[temp];
            numbers[temp] = temp;
        }
    }

    return false;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值