图解leetcode697.数组的度

1.题目描述:

给定一个非空且只包含非负数的整数数组nums,数组的度的定义是指数组里任一元素出现频数的最大值。你的任务是在nums中找到与nums拥有相同大小的度的最短连续子数组,返回其长度。

 2.使用数组记录频次...

其实思路和图解leetcode645.错误的集合(未完)包括图解leetcode287.寻找重复数(未完)类似,但是此题还需后续计算原数组的最短连续子数组的长度,直接看图解结合代码来理清此解法思路。使用数组的空间复杂度相比用哈希表低,内存消耗很小。

class Solution {
    public int findShortestSubArray(int[] nums) {
        int maxNums = 0;
        for(int ele:nums){
            if(ele >= maxNums) maxNums = ele;
        }
        int n = nums.length;
        int max = Math.max(n,maxNums);//取最大值考虑到类似[5,5]数组
        int[] freArr = new int[max + 1];//加一考虑到0元素
        int[] lenArr = new int[max + 1];
        for(int ele:nums){
            freArr[ele]++;
        }
        //Arrays.sort(freArr);不能用sort找到频次最高的元素,破坏freArr索引的对应关系
        int maxFre = 0;
        for(int ele:freArr){
            if(ele >= maxFre) maxFre = ele;
        }
        for(int i = 0;i < max + 1;i++){
            int left = 0;
            int right = 0;
            if(freArr[i] == maxFre){
                for(int j = 0;j < n;j++){
                    if(nums[j] == i){
                        left = j;
                        break;
                    }
                }
                for(int k = n - 1;k >= 0;k--){
                    if(nums[k] == i){
                        right = k;
                        break;
                    }
                }
                lenArr[i] = right - left + 1;//循环得到所有满足条件子数组的长度
            }
        }
        Arrays.sort(lenArr);
        for(int ele:lenArr){
            if(ele != 0) return ele;
        }
        return -1;
    }
}

3.哈希表:

直接提供官解的代码,使用哈希表,value关联为长度为3的一维数组,可以直接依次记录元素的频次,出现的最早位置,出现的最晚位置,代码简洁易懂。

class Solution {
    public int findShortestSubArray(int[] nums) {
        Map<Integer, int[]> map = new HashMap<Integer, int[]>();
        int n = nums.length;
        for (int i = 0; i < n; i++) {
            if (map.containsKey(nums[i])) {
                map.get(nums[i])[0]++;
                map.get(nums[i])[2] = i;
            } else {
                map.put(nums[i], new int[]{1, i, i});
            }
        }
        int maxNum = 0, minLen = 0;
        for (Map.Entry<Integer, int[]> entry : map.entrySet()) {
            int[] arr = entry.getValue();
            if (maxNum < arr[0]) {
                maxNum = arr[0];
                minLen = arr[2] - arr[1] + 1;
            } else if (maxNum == arr[0]) {
                if (minLen > arr[2] - arr[1] + 1) {
                    minLen = arr[2] - arr[1] + 1;
                }
            }
        }
        return minLen;
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值