无重复字符的最长子串
2022/05/24
https://leetcode.cn/problems/longest-substring-without-repeating-characters/
示例
示例一
输入: s = "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是 "abc",所以其长度为 3。
示例二
输入: s = "bbbbb" 输出: 1 解释: 因为无重复字符的最长子串是 "b",所以其长度为 1。
示 例三
输入: s = "pwwkew" 输出: 3 解释: 因为无重复字符的最长子串是 "wke",所以其长度为 3。 请注意,你的答案必须是 子串 的长度,"pwke" 是一个子序列,不是子串。
思考过程
初步实现
- 用 currentString 记录当前的不连续的字符串,用 maxLength 记录当前最长的不连续字符串的长度
- 遍历原字符串 s,与 currentString 的每个字符比较,如果没有重复,则将该字符拼上去;否则裁剪 currentString
var lengthOfLongestSubstring = function (s) {
if (s === "") {
return 0;
}
var maxLength = 1;
var currentString = s[0];
for (var i = 1; i < s.length; i++) {
for (var j = 0; j < currentString.length; j++) {
if (s[i] === currentString[j]) {
currentString = currentString.slice(j + 1);
break;
}
}
currentString += s[i];
maxLength = Math.max(currentString.length, maxLength);
}
return maxLength;
};
利用 hashSet
官方题解利用了 es6 的 Set 对象
var lengthOfLongestSubstring = function (s) {
// 哈希集合,记录每个字符是否出现过
const occ = new Set();
const n = s.length;
// 右指针,初始值为 -1,相当于我们在字符串的左边界的左侧,还没有开始移动
let rk = -1,
ans = 0;
for (let i = 0; i < n; ++i) {
if (i != 0) {
// 左指针向右移动一格,移除一个字符
occ.delete(s.charAt(i - 1));
}
while (rk + 1 < n && !occ.has(s.charAt(rk + 1))) {
// 不断地移动右指针
occ.add(s.charAt(rk + 1));
++rk;
}
// 第 i 到 rk 个字符是一个极长的无重复字符子串
ans = Math.max(ans, rk - i + 1);
}
return ans;
};