题解 | #最长无重复子数组#

最长无重复子数组

http://www.nowcoder.com/practice/b56799ebfd684fb394bd315e89324fb4

题目主要信息:
  • 题目给定一个数组,要找到其中最长的无重复的子数组的长度
  • 子数组必须是数组中连续的一段
举一反三:

学习完本题的思路你可以解决如下题目:

BM90. 最小覆盖子串

方法:滑动窗口(推荐使用)

知识点1:滑动窗口

滑动窗口是指在数组、字符串、链表等线性结构上的一段,类似一个窗口,而这个窗口可以依次在上述线性结构上从头到尾滑动,且窗口的首尾可以收缩。我们在处理滑动窗口的时候,常用双指针来解决,左指针维护窗口左界,右指针维护窗口右界,二者同方向不同速率移动维持窗口。

知识点2:哈希表

哈希表是一种根据关键码(key)直接访问值(value)的一种数据结构。而这种直接访问意味着只要知道key就能在O(1)O(1)时间内得到value,因此哈希表常用来统计频率、快速检验某个元素是否出现过等。

思路:

既然要找一段连续子数组的内不重复的长度,我们可以使用滑动窗口,保证窗口内都是不重复的,然后窗口右界不断向右滑,如果窗口内出现了重复数组,说明新加入的元素与之前的重复了,只需要窗口左界也向右收缩就可以保证窗口内都是不重复的。

而保证窗口内的元素不重复,我们可以使用根据key值快速访问的哈希表,key值为窗口内的元素,value为其出现次数,只要新加入窗口的元素出现次数不为1,就是重复。

while(mp.get(arr[right]) > 1) 
    //窗口左移,同时减去该数字的出现次数
    mp.put(arr[left],mp.get(arr[left++])-1); 

具体做法:

  • step 1:构建一个哈希表,用于统计数组元素出现的次数。
  • step 2:窗口左右界都从数组首部开始,每次窗口优先右移右界,并统计进入窗口的元素的出现频率。
  • step 3:一旦右界元素出现频率大于1,就需要右移左界直到窗口内不再重复,将左边的元素移除窗口的时候同时需要将它在哈希表中的频率减1,保证哈希表中的频率都是窗口内的频率。
  • step 4:每轮循环,维护窗口长度最大值。

图示:

alt

Java代码实现:

import java.util.*;
public class Solution {
    public int maxLength (int[] arr) {
        //哈希表记录窗口内非重复的数字
        HashMap<Integer, Integer> mp = new HashMap<>(); 
        int res = 0;
        //设置窗口左右边界
        for(int left = 0, right = 0; right < arr.length; right++){ 
            //窗口右移进入哈希表统计出现次数
            if(mp.containsKey(arr[right]))
                mp.put(arr[right],mp.get(arr[right])+1); 
            else
                mp.put(arr[right],1);
            //出现次数大于1,则窗口内有重复
            while(mp.get(arr[right]) > 1) 
                //窗口左移,同时减去该数字的出现次数
                mp.put(arr[left],mp.get(arr[left++])-1); 
            //维护子数组长度最大值
            res = Math.max(res, right - left + 1); 
        }
        return res;
    }
}

C++代码实现:

class Solution {
public:
    int maxLength(vector<int>& arr) {
        //哈希表记录窗口内非重复的数字
        unordered_map<int, int> mp; 
        int res = 0;
        //设置窗口左右边界
        for(int left = 0, right = 0; right < arr.size(); right++){ 
            //窗口右移进入哈希表统计出现次数
            mp[arr[right]]++; 
            //出现次数大于1,则窗口内有重复
            while(mp[arr[right]] > 1) 
                //窗口左移,同时减去该数字的出现次数
                mp[arr[left++]]--; 
            //维护子数组长度最大值
            res = max(res, right - left + 1); 
        }
        return res;
    }
};

Python实现代码:

class Solution:
    def maxLength(self , arr: List[int]) -> int:
        #哈希表记录窗口内非重复的数字
        mp = dict() 
        res = 0
        left = 0
        #设置窗口左右边界
        for right in range(len(arr)): 
            if arr[right] in mp:
                #窗口右移进入哈希表统计出现次数
                mp[arr[right]] += 1 
            else:
                mp[arr[right]] = 1
            #出现次数大于1,则窗口内有重复
            while mp[arr[right]] > 1: 
                #窗口左移,同时减去该数字的出现次数
                mp[arr[left]] -=1 
                left += 1
            #维护子数组长度最大值
            res = max(res, right - left + 1) 
        return res


复杂度分析:

  • 时间复杂度:O(n)O(n),外循环窗口右界从数组首右移到数组尾,内循环窗口左界同样如此,因此复杂度为O(n+n)=O(n)O(n+n)=O(n)
  • 空间复杂度:O(n)O(n),最坏情况下整个数组都是不重复的,哈希表长度就为数组长度nn
全部评论
while mp[arr[right]] > 1: #窗口左移,同时减去该数字的出现次数 mp[arr[left]] -=1 ?????这个地方是取巧了吗 这个并不是把窗口中出现两次的建一,而是把指针的最左边的值减一,为什么呢
1 回复 分享
发布于 2022-12-22 15:09 广东
这滑动窗口像一条向前蠕动的蠕虫,一探一缩,寻找能伸展最长的长度。
点赞 回复 分享
发布于 2024-08-23 16:10 上海

相关推荐

01-23 14:54
同济大学 Java
热爱敲代码的程序媛:给你提几点【专业技能】这个模块里面可优化的地方:1.【具备JVM调优经验】可以去b站上搜一下JVM调优的视频,估计一两个小时凭你的学习能力就能掌握JVM调优的实践方面的技能。2.【MySql优化】MySql这一栏,你去b站或者找个博客看看MySql优化,学一下,如果你本身比较熟悉MySql语句的话,那基本半天时间凭你的学习能力MySql语句优化方面的技能你也能掌握个差不多。以上1,2两点主要是因为我看你专业技能大部分都说的是偏理论,没有写应用。再就是最后,你结合你的项目,想一想你的项目中哪些sql语句是可以用MySql优化的,到时候你面试的时候也好结合着说一下。
点赞 评论 收藏
分享
评论
21
18
分享

创作者周榜

更多
牛客网
牛客企业服务