当前位置:   article > 正文

Leetcode 76 最小覆盖子串 java版

Leetcode 76 最小覆盖子串 java版

官网链接:

. - 力扣(LeetCode)

1. 问题

给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 ""

注意:

  • 对于 t 中重复字符,我们寻找的子字符串中该字符数量必须不少于 t 中该字符数量。
  • 如果 s 中存在这样的子串,我们保证它是唯一的答案。

示例 1:

输入:s = "ADOBECODEBANC", t = "ABC"
输出:"BANC"
解释:最小覆盖子串 "BANC" 包含来自字符串 t 的 'A'、'B' 和 'C'。

示例 2:

输入:s = "a", t = "a"
输出:"a"
解释:整个字符串 s 是最小覆盖子串。

示例 3:

输入: s = "a", t = "aa"
输出: ""
解释: t 中两个字符 'a' 均应包含在 s 的子串中,
因此没有符合条件的子字符串,返回空字符串。

提示:

  • m == s.length
  • n == t.length
  • 1 <= m, n <= 105
  • st 由英文字母组成

2. 滑动窗口(author: mcdnull):

     分为left, right 两个指针。先让right 往右侧移动。直至满足条件 包含了t子串 再移动left 右移。期间记录最短指针长度,两个指针位置

步骤一

不断增加right 指针使滑动窗口增大,直到窗口包含了T的所有元素

步骤二

不断增加left指针使滑动窗口缩小,因为是要求最小字串,所以将不必要的元素排除在外,使长度减小,直到碰到一个必须包含的元素,这个时候不能再扔了,再扔就不满足条件了,记录此时滑动窗口的长度,并保存最小值

步骤三

让left再增加一个位置,这个时候滑动窗口肯定不满足条件了,那么继续从步骤一开始执行,寻找新的满足条件的滑动窗口,如此反复,直到right超出了字符串S范围。

 3. 代码:

  1. package com.yunjing.mall.controller.app;
  2. import java.util.HashMap;
  3. import java.util.Map;
  4. /**
  5. * 描述:
  6. * 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 "" 。
  7. *
  8. * @Author: lbc
  9. * @Date: 2024-03-25 8:16
  10. * @email: 594599620@qq.com
  11. * @Description: keep coding
  12. */
  13. public class Solution {
  14. /**
  15. * 滑动窗口
  16. * 右侧窗口先滑动,左侧再进行滑动 右移
  17. *
  18. * @param s
  19. * @param t
  20. * @return
  21. */
  22. public static String minWindow(String s, String t) {
  23. long start = System.currentTimeMillis();
  24. if (s == null || t == null || s.length() < t.length()) {
  25. return "";
  26. }
  27. // 左滑指针
  28. int left = 0;
  29. // 判断窗口是否包含了t的所有元素,避免每次去遍历map,增加耗时
  30. int needCnt = t.length();
  31. int[] minResult = new int[]{0, Integer.MAX_VALUE};
  32. Map<Character, Integer> map = new HashMap<>();
  33. // 初始化map
  34. for (int i = 0; i < t.length(); i++) {
  35. map.put(t.charAt(i), map.getOrDefault(t.charAt(i), 0) + 1);
  36. }
  37. // 先走右边界
  38. for (int right = 0; right < s.length(); right++) {
  39. char c = s.charAt(right);
  40. if (map.getOrDefault(c, 0) > 0) {
  41. needCnt--;
  42. }
  43. map.put(c, map.getOrDefault(c, 0) - 1);
  44. if (needCnt == 0) {
  45. for (; ; ) {
  46. c = s.charAt(left);
  47. if (map.get(c) == 0) {
  48. break;
  49. }
  50. // 左边界
  51. map.put(c, map.getOrDefault(c, 0) + 1);
  52. left++;
  53. }
  54. // 更新结果, 覆盖res
  55. if (right - left < minResult[1] - minResult[0]) {
  56. minResult[1] = right;
  57. minResult[0] = left;
  58. }
  59. c = s.charAt(left);
  60. //将左边界右移,执行下一个窗口
  61. // 由于左边界是t需要的字符,右移后,需要更新tMap和needCnt,表示还需要增加一个字符
  62. map.put(c, map.getOrDefault(c, 0) + 1);
  63. needCnt++;
  64. left++;
  65. }
  66. }
  67. System.out.println("use: " + (System.currentTimeMillis() - start) + "ms");
  68. // 超过边界,返回空串
  69. return minResult[1] > s.length() ? "" : s.substring(minResult[0], minResult[1] + 1);
  70. }
  71. public static void main(String[] args) {
  72. System.out.println(minWindow("ABEFDGFGESFEWEFJKLEFJPFKPOEKGWPEGEFPDFJSDFJLKEJFKFJEIKJWOEIFGJWEIOGFEWOIFGJWEFOIJWEFWPOEJFFJ", "BFS"));
  73. System.out.println(minWindow("EFJEGFJEFJKDJFEOFGKJPEPEPFKMLFKFLEKFEFEFEFGEEFEFGGJKPOWKFPOKWEOPFKWPEOFD", "JJD"));
  74. }
  75. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/IT小白/article/detail/315449
推荐阅读
相关标签
  

闽ICP备14008679号