题目

给你一个整数数组 nums 和一个整数 k ,判断数组中是否存在两个 不同的索引 i 和 j ,满足 nums[i] == nums[j] 且 abs(i - j) <= k 。

如果存在,返回 true ;否则,返回 false 。

示例 1:

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

示例 2:

输入:nums = [1,0,1,1], k = 1 输出:true

示例 3:

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

提示:

1 <= nums.length <= 10^5

-10^9 <= nums[i] <= 10^9

0 <= k <= 10^5

v1-HashMap

思路

HashMap 计数,这个应该很容易想到

1) 用 map 存储,key 对应值,value 对应 indexList

2) 我们找到相同的之后,只需要判断当前的 i 和上一次的 j 对比,是否满足条件即可。

因为 i 递增,不需要考虑 abs 的问题。

实现

public boolean containsNearbyDuplicate(int[] nums, int k) {
        Map<Integer, List<Integer>> countMap = new HashMap<>();
        for(int i = 0; i < nums.length; i++){
            int num = nums[i];
            List<Integer> indexList = countMap.getOrDefault(num, new ArrayList<>());

            // 
            if(!indexList.isEmpty()) {
                int minLen = i - indexList.get(indexList.size()-1);
                if(minLen <= k) {
                    return true;
                }
            }
            indexList.add(i);

            // i-j <= k
            // 判断是否满足条件
            countMap.put(num, indexList);
        }

        return false;
    }

效果

27ms 13.36%

效果一般

v2-滑动窗口

思路

我们其实这样考虑

因为要求 满足 nums[i] == nums[j] 且 abs(i - j) <= k

我们可以规定一个窗口,假设 i 开始,那么 窗口的大小最多为 k。

只要判断这个窗口内是否存在重复元素即可。

是否存在重复元素,是 T217 的问题,这里不再赘述。

实现方式

想实现这个窗口方式也比较多,我们采用 Set

代码

public boolean containsNearbyDuplicate(int[] nums, int k) {
    Set<Integer> set = new HashSet<>();
    for(int i = 0; i < nums.length; i++){
        // 固定窗口大小
        if(i > k) {
            // 移除最开始的元素  保证窗口最多k
            set.remove(nums[i - k - 1]);
        }
        if(set.contains(nums[i])) {
            return true;
        }
        set.add(nums[i]);
    }
    return false;
}

效果

18ms 87.83%

效果还不错。

小结

看了一下最优解也是这样的算法,估计只是执行的时候服务器计时的差别。

滑动窗口有时候需要我们转换一下视角,挺有意思的一个角度。