摘要:用来记录字母出现的可以快速查找出字母是否重复出现,并定位到它出现的位置,方便做删除之前字母的操作。然后把这个字母的更新。
题目:
Given a string, find the length of the longest substring without repeating characters.
Examples:
Given "abcabcbb", the answer is "abc", which the length is 3.
Given "bbbbb", the answer is "b", with the length of 1.
Given "pwwkew", the answer is "wke", with the length of 3. Note that the answer must be a substring, "pwke" is a subsequence and not a substring.
解法1:
核心思想是当发现有重复的字母,就把重复字母第一次出现位置以及之前的字母都删掉,这样就保证剩下的string中没有重复的字母了。用hashmap来记录字母出现的index,可以快速查找出字母是否重复出现,并定位到它出现的位置,方便做删除之前字母的操作。
代码1:
public void deletePrevious(Mapmap, String s, int start, int end) { for (int i = start; i <= end; i++) { char c = s.charAt(i); map.remove(c); } } public int lengthOfLongestSubstring(String s) { int result = 0; if (s.length() == 0) return result; Map map = new HashMap (); int start = 0; int len = 0; for (int i = 0; i < s.length(); i++) { char c = s.charAt(i); if (map.containsKey(c)) { int index = map.get(c); //把从重复的index开始,前面所有的数都在map里去掉,并记下新的start deletePrevious(map, s, start, index); start = index + 1; map.put(c, i); //删掉后的len是当前i减去删掉数的index len = i - index; } else { map.put(c, i); len++; } result = Math.max(result, len); } return result; }
又看了其他人的解法,感觉有很好的思路就拷贝过来了。
解法2:
我用j, i来记录最后不重复string的开始点和当前点,当我找到一个重复出现的字母,我判断一下j是否要更新,如果这个重复字母出现第一次的位置比j小,那么说明至少从j开始才能保证没有重复字母;如果这个重复字母出现第一次的位置比j大,那么将j更新到这个位置的后一位,作为新的开始点。然后把这个字母的index更新。这里省去了我上面说的删除之前所以map中的值,因为就算出现了重复的值,我也还是从j开始算,并且我更新这个值的新index后,原来那个就相当于自动删除了。
代码2:
public int lengthOfLongestSubstring(String s) { if (s.length()==0) return 0; HashMapmap = new HashMap (); int max=0; for (int i = 0, j = 0; i < s.length(); i++) { if (map.containsKey(s.charAt(i))) { //这里j是记录满足条件的start,当发现有重复的数时,应从重复数第一次出现的位置向后一位开始记,但同时要保持从j开始向后所有数都不重复,就得取最大值 j = Math.max(j, map.get(s.charAt(i)) + 1); } //更新重复出现数的index值 map.put(s.charAt(i), i); max = Math.max(max, i - j + 1); } return max; }
解法3:
这里思路跟解法2一样,只不过他巧妙利用cache,因为所以字母就127个,那个我用256的空间就肯定能记录所有的点。比如就算最极端,cache(0)是‘a",一直到cache(255)才双出现‘a’,我的空间也是够的。
public int lengthOfLongestSubstring(String s) { int result = 0; int[] cache = new int[256]; for (int i = 0, j = 0; i < s.length(); i++) { j = (cache[s.charAt(i)] > 0) ? Math.max(j, cache[s.charAt(i)]) : j; //注意因为stirng的index是从0开始,但是我们判断这个字母是否在cache里也用这个字母的cache值是否来判断,所以我们把每个存进cache里的字母的index + 1, 这样存的就是它的下一个数的index。 cache[s.charAt(i)] = i + 1; result = Math.max(result, i - j + 1); } return result; }
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/64862.html
摘要:哈希表是最自然的想法。在遍历字符串时,我们先根据哈希表找出该字符上次出现的位置,如果大于等于子字符串首,便更新子字符串首。结束后,将该字符新的位置放入哈希表中。 Longest Substring Without Repeating Characters 最新更新解法:https://yanjia.me/zh/2018/12/... Given a string, find the ...
摘要:解题思路本题借助实现。如果字符未出现过,则字符,如果字符出现过,则维护上次出现的遍历的起始点。注意点每次都要更新字符的位置最后返回时,一定要考虑到从到字符串末尾都没有遇到重复字符的情况,所欲需要比较下和的大小。 Longest Substring Without Repeating CharactersGiven a string, find the length of the lon...
摘要:哈希表法双指针法只有当也就是时上面的循环才会结束,,跳过这个之前的重复字符再将置为 Problem Given a string, find the length of the longest substring without repeating characters. Example For example, the longest substring without repeat...
摘要:原问题我的沙雕解法无重复字母存在重复字母挨打最暴力的无脑解法,耗时。。。 原问题 Given a string, find the length of the longest substring without repeating characters. Example 1: Input: abcabcbb Output: 3 Explanation: The answer is a...
摘要:建立数组,存储个字符最近一次出现的位置。首次出现某字符时,其位置标记为,并用无重复字符计数器记录无重复字符的长度,再在更新其最大值。循环完整个字符串后,返回最大值。 Problem Given a string, find the length of the longest substring without repeating characters. Examples: Given ...
阅读 3054·2023-04-26 02:27
阅读 2769·2021-11-22 13:54
阅读 907·2021-11-12 10:36
阅读 3758·2021-10-09 09:44
阅读 3184·2021-10-09 09:41
阅读 1229·2021-09-22 10:02
阅读 2838·2019-08-30 15:56
阅读 3109·2019-08-30 11:02