Given a string, find the length of the longest substring without repeating characters.java
给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度。指针
示例 1:code
输入: "abcabcbb" 输出: 3 解释: 由于无重复字符的最长子串是 "abc",因此其长度为 3。
示例 2:blog
输入: "bbbbb" 输出: 1 解释: 由于无重复字符的最长子串是 "b",因此其长度为 1。
示例 3:索引
输入: "pwwkew" 输出: 3 解释: 由于无重复字符的最长子串是 "wke",因此其长度为 3。 请注意,你的答案必须是 **子串** 的长度,"pwke" 是一个子序列,不是子串。
<!--more-->ip
remove
掉 -- 后面发现只remove掉那个重复的字符的话,有些没有重复的字符在回退以后再次计算的时候,会发生混乱<font color=grey size=2>(回退后再次走到以前不重复的字符时候,由于hash
表中已经在第一次put
过了,因此此次必定会发生重复状况)</font>上面方法应该是 $ O(n^2) $ ,由于会发生例如
abcdea
在最后a
发生重复,就会彻底回退到b
位置---so low ;提交记录耗时大概80+ms内存
map<Character,Integer>
中存放的以前被重复字符的value(即字符所在的索引)换为当前发生重复的字符位置 -- 不是被重复字符max
最大j,k
表明的都是索引,因此最后结果 为max+1
Two和Three 提交的耗时6ms,占用内存35M--占用内存居然超过了 100%的java用户ヽ(°◇° )ノ资源
package com.dasnnj.practice.medium; import java.util.HashMap; import java.util.Map; /** * Description <p> TODO: * 给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度。 * <p> * 示例 1: * <p> * 输入: "abcabcbb" * 输出: 3 * 解释: 由于无重复字符的最长子串是 "abc",因此其长度为 3。 * 示例 2: * <p> * 输入: "bbbbb" * 输出: 1 * 解释: 由于无重复字符的最长子串是 "b",因此其长度为 1。 * 示例 3: * <p> * 输入: "pwwkew" * 输出: 3 * 解释: 由于无重复字符的最长子串是 "wke",因此其长度为 3。 * 请注意,你的答案必须是 子串 的长度,"pwke" 是一个子序列,不是子串。</p> * * @author DASNNJ <a href="mailto:dasnnj@outlook.com.com"> dasnnj@outlook.com </a> * @date 2019-05-08 13:17 */ public class LongestSubstringWithoutRepeatingCharacters { public static void main(String[] args) { LongestSubstringWithoutRepeatingCharacters l = new LongestSubstringWithoutRepeatingCharacters(); System.out.println(l.lengthOfLongestSubstring("")); } public int lengthOfLongestSubstring(String s) { //One /*char[] chars = s.toCharArray(); Map<Character, Integer> map = new HashMap<>(8); //计算非重复字符的长度 Integer len = 0; int max = 0; //索引 Integer index; for (int i = 0; i < chars.length; i++) { //若是是正常进行 //若是重复 if ((index = map.get(chars[i])) != null) { //回退到重复的位置,从重复位置的下一位从新算,至关于舍弃两个重复字符中的第一个 i = index; //长度归零 len = 0; //将map清空,从重复位置的下一位置从新计算 -- 清空是由于第一个重复的在上面提到是至关于舍弃,不清空的话会影响下次循环的判断 map.clear(); } else { //没重复固然正常put 正常++ map.put(chars[i], i); len++; } //每次循环都保持max最大 if (len > max) { max = len; } } return max;*/ if ("".equals(s)) { return 0; } char[] chars = s.toCharArray(); //j k,用于Two方法的两个指针---后面发现直接用for循环便可 int j = 0, k = 0, max = 0; Integer ele; Map<Character, Integer> sets = new HashMap<>(16); //Three for (int m = 0; m < chars.length; m++) { //若是发生重复 if ((ele = sets.get(chars[m])) != null) { // j指针指向两个重复字符中的第一个的下一位置,j指针不能后退,只能前进,因此下面有个判断 if (j < ele + 1) { j = ele + 1; } } //不重复则是正常put,重复状况1.将j指针指向原字符的下一位2.用新字符替换掉map中原字符(主要是为了替换map中key为此字符 的value值也就是索引) sets.put(chars[m], m); //每次循环保持max最大 if (max < m - j) { max = m - j; } } //Two 原理同Three /*while (j < chars.length && k < chars.length) { if ((ele = sets.get(chars[k])) != null) { if (j < ele + 1) { j = ele + 1; } } sets.put(chars[k], k); k++; if (max < k - j) { max = k - j; } }*/ return max + 1; } }