在长度 2N 的数组中找出重复 N 次的元素

在长度 2N 的数组中找出重复 N 次的元素

在长度 2N 的数组中找出重复 N 次的元素

这道题出自LeetCode,题目如下:

给你一个整数数组 nums ,该数组具有以下属性:

  • nums.length == 2 * n.
  • nums 包含 n + 1 个 不同的 元素
  • nums 中恰有一个元素重复 n

找出并返回重复了 n 次的那个元素。

示例 1:

输入:nums = [1,2,3,3]
输出:3

示例 2:

输入:nums = [2,1,2,5,3,2]
输出:2

示例 3:

输入:nums = [5,1,5,2,5,3,5,4]
输出:5

首先从题目中可以知道,长度为2n的数组中有n+1个不同的元素,其中只有1个元素重复了n次,那么最坏情况下我们需要遍历这个数组到n+1个元素的时候,才能找到这个重复的元素。但是如果顺序遍历的话,我们就需要一个大小为n的map,来记录当前遍历过的元素。那么有没有办法在O(1)空间解决这个问题呢?

我们可以通过分组的方式来解决这个问题。长度为2n的数组,可以两两一组,分成n组。最差的情况下,就是每组的元素一个是重复元素,一个是不重复元素。这种情况我们无法直接通过遍历n个分组来找到这个重复元素。但是这样,我们可以在最坏假设的基础上,将每两个分组进行合并,就是以4个元素为一组。通过之前分析可以知道,这4个元素中必然至少存在2个重复元素,那意味着我们只需要用3个标记就能找到这个重复元素,也就是O(1)的空间。

最后通过的代码如下:

class Solution {
public:
    int repeatedNTimes(vector<int>& nums) {
        for(int i = 1; i <= 3; i++)
        {
            for(int j = 0; j < nums.size() - i; j++)
            {
                if(nums[j] == nums[j + i])
                {
                    return nums[j];
                }
            }
        }

        return 0;
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 中位数是指一组数据中处于中间位置的数,即将数据从小到大排序后,位于中间位置的数。对于两个已排好序的数组x和y,可以使用归并排序的思想,将它们合并成一个有序数组,然后找出这个数组的中位数即可。具体步骤如下: 1. 定义一个新的数组z,长度2n。 2. 使用归并排序的思想,将x和y合并成一个有序数组z。具体方法是,定义两个指针i和j,分别指向x和y的起始位置,比较x[i]和y[j]的大小,将较小的数放入z中,并将指向该数的指针向后移动一位,直到其中一个数组的所有数都放入了z中。 3. 如果x和y的长度不相等,将剩余的数依放入z中。 4. 如果z的长度为偶数,中位数为z[n-1]和z[n]的平均值;如果z的长度为奇数,中位数为z[n]。 5. 返回中位数。 注意:在实际编程中,需要考虑数组越界的情况,以及处理浮点数的精度问题。 ### 回答2: 中位数是一个序列中居于中间位置的数,对于已经排好序的数组而言,容易求得中位数,如果n为奇数,中位数就是第(n+1)/2个数;如果n为偶数,则中位数是第n/2和n/2+1个数的平均数。但是对于两个数组,如何求得它们合并后的中位数呢? 一种思路是直接将两个数组合并成一个有序数组,然后再按照上述方法求出中位数。但是时间复杂度为O(n),无法满足要求。 另一种思路是采用递归的方式,不断缩小求解问题的规模。首先找到x和y的中位数,分别记为m1和m2,比较m1和m2的大小,如果m1<m2,则中位数必然在x[m1:n-1]和y[0:m2]之间,即问题规模缩小为原来的一半;反之,中位数在x[0:m1]和y[m2:n-1]之间。进一步,如果问题规模为奇数,则找到第(n+1)/2小的数,如果为偶数,则找到第n/2和n/2+1小的数,然后求它们的平均数。 递归结束条件为,x和y各自缩减到一个元素时,比较它们的大小,较小的数即为中位数。 该算法的时间复杂度为O(logn),满足要求。 ### 回答3: 中位数是指一组数中大小排列中处于中间位置的数,当数的个数为偶数时,中位数为中间两个数的平均数。设x[ 0 : n - 1]和y[ 0 : n – 1 ]为两个数组,每个数组中含有n个已排好序的数,要求找到这两个数组合并后的2n个数的中位数。 一种简单而直接的做法是将两个数组合并成一个有序数组,然后找到这个新数组的中位数。可以用归并排序的思想,在比较两个数组中的数的大小时,将较小的数放到新数组中,对应的数组索引+1,直到找到中位数或者新数组中存储了2n个数。如果2n为奇数,则新数组中第n个数即为中位数;如果2n为偶数,则新数组中第n和第n+1个数的平均数即为中位数。 时间复杂度为O(n),由归并排序中每个元素只比较一导致。该做法虽然简单有效,但需要额外的存储空间来存储新数组,占用的空间大小为O(n)。如果不希望占用额外的空间,还可以使用双指针的方法,分别从x和y两个数组的中位数开始往两边扩展,每排除掉一半的数据,直到找到合并后的2n个数的中位数。 具体做法为,首先找到两个数组自身的中位数,分别为xmid和ymid,若xmid小于ymid,则x数组的前xmid个元素和y数组的后n-xmid个元素一定在2n个数的中位数的左边;若xmid大于ymid,则x数组的后n-xmid个元素和y数组的前xmid个元素一定在2n个数的中位数的左边。根据这个规律,可以反复排除一半的数据,直到找到合并后的2n个数的中位数。 时间复杂度为O(logn),每一排除了一半的数据,类似于二分查找的思想。这种做法不需要额外的存储空间,但需要特殊处理边界情况,比如xmid=0或n-1的情况。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值