官网链接:
. - 力扣(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
s
和t
由英文字母组成
2. 滑动窗口(author: mcdnull):
分为left, right 两个指针。先让right 往右侧移动。直至满足条件 包含了t子串 再移动left 右移。期间记录最短指针长度,两个指针位置
步骤一
不断增加right 指针使滑动窗口增大,直到窗口包含了T的所有元素
步骤二
不断增加left指针使滑动窗口缩小,因为是要求最小字串,所以将不必要的元素排除在外,使长度减小,直到碰到一个必须包含的元素,这个时候不能再扔了,再扔就不满足条件了,记录此时滑动窗口的长度,并保存最小值
步骤三
让left再增加一个位置,这个时候滑动窗口肯定不满足条件了,那么继续从步骤一开始执行,寻找新的满足条件的滑动窗口,如此反复,直到right超出了字符串S范围。
3. 代码:
package com.yunjing.mall.controller.app;
import java.util.HashMap;
import java.util.Map;
/**
* 描述:
* 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 "" 。
*
* @Author: lbc
* @Date: 2024-03-25 8:16
* @email: 594599620@qq.com
* @Description: keep coding
*/
public class Solution {
/**
* 滑动窗口
* 右侧窗口先滑动,左侧再进行滑动 右移
*
* @param s
* @param t
* @return
*/
public static String minWindow(String s, String t) {
long start = System.currentTimeMillis();
if (s == null || t == null || s.length() < t.length()) {
return "";
}
// 左滑指针
int left = 0;
// 判断窗口是否包含了t的所有元素,避免每次去遍历map,增加耗时
int needCnt = t.length();
int[] minResult = new int[]{0, Integer.MAX_VALUE};
Map<Character, Integer> map = new HashMap<>();
// 初始化map
for (int i = 0; i < t.length(); i++) {
map.put(t.charAt(i), map.getOrDefault(t.charAt(i), 0) + 1);
}
// 先走右边界
for (int right = 0; right < s.length(); right++) {
char c = s.charAt(right);
if (map.getOrDefault(c, 0) > 0) {
needCnt--;
}
map.put(c, map.getOrDefault(c, 0) - 1);
if (needCnt == 0) {
for (; ; ) {
c = s.charAt(left);
if (map.get(c) == 0) {
break;
}
// 左边界
map.put(c, map.getOrDefault(c, 0) + 1);
left++;
}
// 更新结果, 覆盖res
if (right - left < minResult[1] - minResult[0]) {
minResult[1] = right;
minResult[0] = left;
}
c = s.charAt(left);
//将左边界右移,执行下一个窗口
// 由于左边界是t需要的字符,右移后,需要更新tMap和needCnt,表示还需要增加一个字符
map.put(c, map.getOrDefault(c, 0) + 1);
needCnt++;
left++;
}
}
System.out.println("use: " + (System.currentTimeMillis() - start) + "ms");
// 超过边界,返回空串
return minResult[1] > s.length() ? "" : s.substring(minResult[0], minResult[1] + 1);
}
public static void main(String[] args) {
System.out.println(minWindow("ABEFDGFGESFEWEFJKLEFJPFKPOEKGWPEGEFPDFJSDFJLKEJFKFJEIKJWOEIFGJWEIOGFEWOIFGJWEFOIJWEFWPOEJFFJ", "BFS"));
System.out.println(minWindow("EFJEGFJEFJKDJFEOFGKJPEPEPFKMLFKFLEKFEFEFEFGEEFEFGGJKPOWKFPOKWEOPFKWPEOFD", "JJD"));
}
}