【LeetCode 手撕算法】(滑动窗口) 3-无重复字符的最长子串、438-找到字符串中所有字母异位词

3-无重复字符的最长子

给定一个字符串 s ,请你找出其中不含有重复字符的最长子串的长度。

示例 1

输入: s = "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是 "abc",所以其长度为 3。注意 "bca" 和 "cab" 也是正确答案。

示例 2:

输入: s = "bbbbb" 输出: 1 解释: 因为无重复字符的最长子串是 "b",所以其长度为 1。

示例 3:

输入: s = "pwwkew" 输出: 3 解释: 因为无重复字符的最长子串是 "wke",所以其长度为 3。

请注意,你的答案必须是 子串 的长度,"pwke"是一个*子序列,*不是子串。

思路: 求子串的最长值,说明是变化的,选用滑动窗口。两指针都在最左端,右指针持续 探路,窗口从0开始加入元素 并持续判断是否重复。若出现重复 则删除最左端,直到窗口内不重复为止。右指针走完就结束,这个过程中记录窗口的长度。

java 复制代码
//----滑动窗口模版-----
//外层循环扩展右边界,内层循环扩展左边界
for (int l = 0, r = 0 ; r < n ; r++) {
	//当前考虑的元素
	while (l <= r && check()) {//区间[left,right]不符合题意
        //扩展左边界
    }
    //区间[left,right]符合题意,统计相关信息
}
java 复制代码
class Solution {
    public int lengthOfLongestSubstring(String s) {
        char[]ch= s.toCharArray();       //把string变成char
        Set<Character> win=new HashSet<>();  //窗口需要hash去重
        int max=0;
        for(int l=0,r=0;r<ch.length;r++){
            while(win.contains(ch[r])){   //一直判断窗口win内是否含有右指针
                win.remove(ch[l]);       //有就删除左指针,直到干净
                l++;
            }
            win.add(ch[r]);     //若窗口win无,则右指针加入窗口
            max=Math.max(max,r-l+1); //记录窗口最大时的值
        }
        return max;
    }
}

细节注意:

1、contains( ) 拼写别少了s

2、char[ ]ch= s.toCharArray(); 必须把string长串变成char数组,方便用双指针

3、Set<Character> win=new HashSet<>(); 滑动窗口必须用集合去重,别忘记<>, HashSet为固定实现类,Character为格式

4、Math 大写

5、左指针要一直去重,if只能判断一次,while可以一直判断。 举例 abcac,abcac,abca****c ,++a++++bcac,abcac,abcac++ 其中左指针连续减了两次


438-找到字符串中所有字母异位词

给定两个字符串 sp,找到 s 中所有 p 的 异位词 的子串,返回这些子串的起始索引。不考虑答案输出的顺序。

示例 1:

输入: s = "cbaebabacd", p = "abc"

输出: [0,6]

解释: 起始索引等于 0 的子串是 "cba", 它是 "abc" 的异位词。 起始索引等于 6 的子串是 "bac", 它是 "abc" 的异位词。

示例 2:

输入: s = "abab", p = "ab"

输出: [0,1,2]

解释: 起始索引等于 0 的子串是 "ab", 它是 "ab" 的异位词。 起始索引等于 1 的子串是 "ba", 它是 "ab" 的异位词。 起始索引等于 2 的子串是 "ab", 它是 "ab" 的异位词。

思路:

异位词即每个字母出现次数相同,因此选用26位int数组。由于异位词的二者长度相同,由此可以确定窗口大小,即p的长度。s的窗口从i=0开始移动,到s长-p长+1为止,每移动一次进行哈希统计,并于hashP比较,是否相同。

java 复制代码
class Solution {
    public List<Integer> findAnagrams(String s, String p) {
        char[] winS=s.toCharArray();
        char[] winP=p.toCharArray();
        int [] hashP=new int[26];
        ArrayList<Integer> list= new ArrayList<>();
        for(int i=0;i<winP.length;i++){
            hashP[winP[i]-'a']++;
        }
        for(int i=0;i<winS.length-winP.length+1;i++){
            int [] hashS=new int[26];
            for(int j=i;j<i+winP.length;j++){
                hashS[winS[j]-'a']++;
            }
            if(Arrays.equals(hashS,hashP)){list.add(i);}
            
        }
        return list;
    }
}

注意事项:

1、比较地址内容用 a.equals(b) 比较数组内容用Arrays.equals(a,b)

2、每次查完hashS必须进行清空,不能直接hashS=0,因为左右两种数据类型不一致,可以重新初始化,int [ ] hashS=new int[26];

3、动态添加数组用ArrayList<Integer> list= new ArrayList<>();

4、String类型不能用 .length算长度,必须转成char

相关推荐
汀、人工智能1 小时前
[特殊字符] 第21课:最长有效括号
数据结构·算法·数据库架构·图论·bfs·最长有效括号
Boop_wu1 小时前
[Java 算法] 字符串
linux·运维·服务器·数据结构·算法·leetcode
故事和你912 小时前
洛谷-算法1-2-排序2
开发语言·数据结构·c++·算法·动态规划·图论
Fcy6482 小时前
算法基础详解(三)前缀和与差分算法
算法·前缀和·差分
kvo7f2JTy2 小时前
基于机器学习算法的web入侵检测系统设计与实现
前端·算法·机器学习
List<String> error_P3 小时前
蓝桥杯最后几天冲刺:暴力大法(一)
算法·职场和发展·蓝桥杯
迈巴赫车主4 小时前
蓝桥杯3500阶乘求和java
java·开发语言·数据结构·职场和发展·蓝桥杯
流云鹤4 小时前
Codeforces Round 1090 (Div. 4)
c++·算法
wljy14 小时前
第十三届蓝桥杯大赛软件赛省赛C/C++ 大学 B 组(个人见解,已完结)
c语言·c++·算法·蓝桥杯·stl
高一要励志成为佬4 小时前
【数据结构】算法复杂度
数据结构