mirror of
https://github.com/halfrost/LeetCode-Go.git
synced 2025-07-31 16:14:39 +08:00
125 lines
2.3 KiB
Markdown
125 lines
2.3 KiB
Markdown
# [3. Longest Substring Without Repeating Characters](https://leetcode.com/problems/longest-substring-without-repeating-characters/)
|
||
|
||
## 题目
|
||
|
||
Given a string, find the length of the longest substring without repeating characters.
|
||
|
||
|
||
|
||
**Example 1**:
|
||
|
||
```
|
||
|
||
Input: "abcabcbb"
|
||
Output: 3
|
||
Explanation: The answer is "abc", with the length of 3.
|
||
|
||
```
|
||
|
||
**Example 2**:
|
||
|
||
```
|
||
|
||
Input: "bbbbb"
|
||
Output: 1
|
||
Explanation: The answer is "b", with the length of 1.
|
||
|
||
```
|
||
|
||
**Example 3**:
|
||
|
||
```
|
||
|
||
Input: "pwwkew"
|
||
Output: 3
|
||
Explanation: 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.
|
||
|
||
```
|
||
|
||
## 题目大意
|
||
|
||
|
||
在一个字符串重寻找没有重复字母的最长子串。
|
||
|
||
## 解题思路
|
||
|
||
这一题和第 438 题,第 3 题,第 76 题,第 567 题类似,用的思想都是"滑动窗口"。
|
||
|
||
滑动窗口的右边界不断的右移,只要没有重复的字符,就持续向右扩大窗口边界。一旦出现了重复字符,就需要缩小左边界,直到重复的字符移出了左边界,然后继续移动滑动窗口的右边界。以此类推,每次移动需要计算当前长度,并判断是否需要更新最大长度,最终最大的值就是题目中的所求。
|
||
|
||
|
||
## 代码
|
||
|
||
```go
|
||
|
||
package leetcode
|
||
|
||
// 解法一 位图
|
||
func lengthOfLongestSubstring(s string) int {
|
||
if len(s) == 0 {
|
||
return 0
|
||
}
|
||
var bitSet [256]bool
|
||
result, left, right := 0, 0, 0
|
||
for left < len(s) {
|
||
// 右侧字符对应的 bitSet 被标记 true,说明此字符在 X 位置重复,需要左侧向前移动,直到将X标记为 false
|
||
if bitSet[s[right]] {
|
||
bitSet[s[left]] = false
|
||
left++
|
||
} else {
|
||
bitSet[s[right]] = true
|
||
right++
|
||
}
|
||
if result < right-left {
|
||
result = right - left
|
||
}
|
||
if left+result >= len(s) || right >= len(s) {
|
||
break
|
||
}
|
||
}
|
||
return result
|
||
}
|
||
|
||
// 解法二 滑动窗口
|
||
func lengthOfLongestSubstring_(s string) int {
|
||
if len(s) == 0 {
|
||
return 0
|
||
}
|
||
var freq [256]int
|
||
result, left, right := 0, 0, -1
|
||
|
||
for left < len(s) {
|
||
if right+1 < len(s) && freq[s[right+1]-'a'] == 0 {
|
||
freq[s[right+1]-'a']++
|
||
right++
|
||
} else {
|
||
freq[s[left]-'a']--
|
||
left++
|
||
}
|
||
result = max(result, right-left+1)
|
||
}
|
||
return result
|
||
}
|
||
|
||
func max(a int, b int) int {
|
||
if a > b {
|
||
return a
|
||
}
|
||
return b
|
||
}
|
||
|
||
|
||
```
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|