📑 题目:3. 无重复字符的最长子串

🚀 本题 LeetCode 传送门

题目大意

在一个字符串重寻找没有重复字母的最长子串。

解题思路

这一题和第 438 题,第 3 题,第 76 题,第 567 题类似,用的思想都是”滑动窗口”。

滑动窗口的右边界不断的右移,只要没有重复的字符,就持续向右扩大窗口边界。一旦出现了重复字符,就需要缩小左边界,直到重复的字符移出了左边界,然后继续移动滑动窗口的右边界。以此类推,每次移动需要计算当前长度,并判断是否需要更新最大长度,最终最大的值就是题目中的所求。

代码

  1. package leetcode
  2. // 解法一 位图
  3. func lengthOfLongestSubstring(s string) int {
  4. if len(s) == 0 {
  5. return 0
  6. }
  7. var bitSet [256]bool
  8. result, left, right := 0, 0, 0
  9. for left < len(s) {
  10. // 右侧字符对应的 bitSet 被标记 true,说明此字符在 X 位置重复,需要左侧向前移动,直到将 X 标记为 false
  11. if bitSet[s[right]] {
  12. bitSet[s[left]] = false
  13. left++
  14. } else {
  15. bitSet[s[right]] = true
  16. right++
  17. }
  18. if result < right-left {
  19. result = right - left
  20. }
  21. if left+result >= len(s) || right >= len(s) {
  22. break
  23. }
  24. }
  25. return result
  26. }
  27. // 解法二 滑动窗口
  28. func lengthOfLongestSubstring1(s string) int {
  29. if len(s) == 0 {
  30. return 0
  31. }
  32. var freq [256]int
  33. result, left, right := 0, 0, -1
  34. for left < len(s) {
  35. if right+1 < len(s) && freq[s[right+1]-'a'] == 0 {
  36. freq[s[right+1]-'a']++
  37. right++
  38. } else {
  39. freq[s[left]-'a']--
  40. left++
  41. }
  42. result = max(result, right-left+1)
  43. }
  44. return result
  45. }
  46. // 解法三 滑动窗口-哈希桶
  47. func lengthOfLongestSubstring2(s string) int {
  48. right, left, res := 0, 0, 0
  49. indexes := make(map[byte]int, len(s))
  50. for left < len(s) {
  51. if idx, ok := indexes[s[left]]; ok && idx >= right {
  52. right = idx + 1
  53. }
  54. indexes[s[left]] = left
  55. left++
  56. res = max(res, left-right)
  57. }
  58. return res
  59. }
  60. func max(a int, b int) int {
  61. if a > b {
  62. return a
  63. }
  64. return b
  65. }