【贪心算法-LeetCode3:无重复字符的最长子串(Java实现)】

个人社区:https://bbs.csdn.net/forums/smile
个人主页:https://blog.csdn.net/qq_43665602
欢迎各位志同道合的朋友,一起学习!

一、题目描述

1.题目内容

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

2.样例

在这里插入图片描述

二、解决方案

看见这种求最优解的问题,可以尝试使用贪心算法,尤其对于集合覆盖问题、区间问题以及分配问题。

贪心算法:求局部最优(当前最优),以使最终结果为最优解或者近似最优解。

本题目的是求出给定字符串S中不含重复字符的最长子串,很自然地可以想到,我们可以通过逐步求解当前无重复最长子串,最后得到字符串S中不含重复字符的最长子串。
需要注意子串必须是连续的,要与子序列区分。

1.算法流程

1)分析

以样例1和样例2为例,我通过图解分析一下求解无重复字符最长子串的过程:
样例1,S=“abcabcbb”,从字符串开头开始分别以当前字符为起点,确定此时的无重复字符的最长子串:
(1)i=0:起点为a,最长子串为abc,长度为3;
(2)i=1:起点为b,最长子串为bca,长度为3;
(3)i=2:起点为c,最长子串为cab,长度为3;
(4)i=3:起点为a,最长子串为abc,长度为3;
(5)i=4:起点为b,最长子串为bc,长度为2;
(6)i=5:起点为c,最长子串为cb,长度为2;
(7)i=6:起点为b,最长子串为b,长度为1;
(8)i=7:起点为b,最长子串为b,长度为1;
可看到第一次确定的子串长度为3,此后子串长度均小于等于3,所以最长子串为abc,其长度为3。
在这里插入图片描述
样例2,S=“pwwkew”,从字符串开头开始分别以当前字符为起点,确定此时的无重复字符的最长子串:
(1)i=0:起点为p,最长子串为pw,长度为2;
(2)i=1:起点为w,最长子串为w,长度为1;
(3)i=2:起点为w,最长子串为wke,长度为3;
(4)i=3:起点为k,最长子串为kew,长度为3;
(5)i=4:起点为e,最长子串为ew,长度为2;
(6)i=5:起点为w,最长子串为w,长度为1;
在这里插入图片描述

可看到第三次确定的子串长度为3,此后子串长度均小于等于3,所以最长子串为wke,其长度为3。
从上面的图示中可以清晰看到最长子串的确定过程,由此可归纳出我们的算法流程,便于代码编写。

2)算法流程

选择策略:确定当前最长无重复子串,遇到重复字符即说明可确定当前最长无重复子串.

  • (1)依次遍历字符串,使用集合存储每个元素(这里使用HashSet);
  • (2)如果集合中存在当前指向的字符,说明有重复:比较此时maxLen与set.size的大小决定是否更新maxLen;
  • (3)循环(1-2),如果遇到更长子串,则更新maxLen; 如果剩余子串长度小于等于maxLen则停止遍历;

需要注意特殊情况:如果字符串为空串或者其长度为1,说明其最长无重复子串为其本身。

2.Java代码

1)核心代码

class Solution {
    /**
     * 选择策略:当前最长无重复子串,遇到重复字符就重置
     */
    public int lengthOfLongestSubstring(String s) {
        if (s.length()<=1){  // 如果字符串为空串或者其长度为1,说明其最长无重复子串为其本身
            return s.length();
        }
        int n=s.length();
        Set<Character> withoutRepeat=new HashSet<>();  // 存储当前子串
        int i=0,maxLen=0;
        while (i<n){  // 依次遍历每个字符
            withoutRepeat.clear();  // 存储新的子串需要重置Set
            int j=i; 
            while (j<n){ // 寻找当前字符开始的最长子串
                if(withoutRepeat.contains(s.charAt(j))){  // 遇到重复字符就停止寻找
                    break;
                }
                withoutRepeat.add(s.charAt(j));
                j++;
            }
            if((j-i+1)>maxLen){  // 确定是否更新maxLen,j-i+1为当前子串长度
                maxLen=j-i;
            }
            if (maxLen>=(n-i+1)){ // 如果剩余子串长度小于等于maxLen则停止遍历
                break;
            }
            i++;
        }
        return maxLen;
    }
}

2)完整测试代码

import java.util.HashSet;
import java.util.Set;

/**
 * LeetCode3:无重复字符的最长子串
 */
public class WithoutRepeat {
    public static void main(String[] args) {
        String s = "abcabcbb";
//        String s = "bbbbbb";
//        String s = "pwwkew";
//        String s = "au";
        System.out.println(new RepeatSolution().lengthOfLongestSubstring(s));
    }
}

class RepeatSolution {
    /**
     * 选择策略:当前最长无重复子串,遇到重复字符就重置
     * @param s
     * @return
     */
    public int lengthOfLongestSubstring(String s) {
        if (s.length()<=1){
            return s.length();
        }
        int n=s.length();
        Set<Character> withoutRepeat=new HashSet<>();
        int i=0,maxLen=0;
        while (i<n){  // 依次遍历每个字符
            withoutRepeat.clear();
            int j=i;  // 寻找当前字符开始的最长子串
            while (j<n){
                if(withoutRepeat.contains(s.charAt(j))){
                    break;
                }
                withoutRepeat.add(s.charAt(j));
                j++;
            }
            if((j-i+1)>maxLen){
                maxLen=j-i;
            }
            if (maxLen>=(n-i+1)){
                break;
            }
            i++;
        }
        return maxLen;
    }
}
  • 7
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

NorthSmile

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值