【力扣题解】【76. 最小覆盖子串】容易理解版
76. 最小覆盖子串
总结和复盘
这是时隔1年4个月之后,再次写的题解,比第一次要清晰很多。
我刚开始,就是用方法一做的,提交之后报超出内存限制;
对方法一进行优化,得到方法二,提交之后就AC了。
这是第二次做这道题,
第一次的题解,是写给自己看的,有很多思考的过程,作为题解是不清晰的。
这一次的题解,会更好理解。
首先,我新增了两个成员函数,相当于把功能拆分开了,这样更容易我们去理解;
其次,由于使用了其他函数,于是我把过程之中用到的变量,提到了类中,做成了成员变量,这样我就方便在其他成员函数之中使用。
先做,再优化。
解法一
// 解法一 //
// 由于使用substr从s之中截取子字符串,并保存在answer之中,导致内存超出了限制class Solution {
public:string answer; // 这道题的答案(由于保存了结果字符串,导致内存超出了限制)unordered_map<char, int> window; // 这就是滑动的窗口unordered_map<char, int> need; // 目标字符串 t 中每个字符的情况int left = 0, right = 0; // 用于滑动的指针int validCount = 0; // 「与字符串t中数量相等的字符」的数量string minWindow(string s, string t) {// 初始化 needfor (char c : t)need[c]++;// 核心思想: 不断增大right找到一个可行解;while (right < s.length()) {char R = s[right];// 先看这第一步:R 是我们需要的字符时if (need.count(R) == 1) { window[R]++; // 要放到窗口之中if (window[R] == need[R]) { // 该字符的数量已经达到要求validCount++; // 已就绪的字符数量+1if (validCount == need.size()) { // 如果所有字符都就绪,更新一下answerupdateAnswer(s);}}}// 最后看这第三步// 你想啊,如果只是right++,left不动,那这个窗口岂不是越来越长了// 所以,要考虑一下什么时候left也要++// 核心思想:当前的窗口中,已经包含了不止一个解的时候,我们让left++,去优化一下// 因为题目要求找字符串s的最小字串// 比如:s="A-B-C---BAC--", t="ABC"// 不断的增加right指针,当这个窗口达到「A-B-C---BAC」时,// 这也是s的字串 且 也涵盖t中的所有字符// 但这不是最优解// 所以要增加left,让窗口变成「BAC」// 才是最优解moveLeftFindThePerfectAnswer(s, t);// 再看这第二步:R 不是我们需要的字符,则 right++ 继续寻找right++;}return answer;}void moveLeftFindThePerfectAnswer(const string& s, const string& t) {while (validCount == need.size()) { // 已经找到解的情况之下,才会优化解char L = s[left];if (need.count(L) == 1) { // 字符L是目标的其中一个字符时,代表可能可以优化updateAnswer(s);window[L]--;if (window[L] < need[L])validCount--;if (window[L] == 0)window.erase(L);}// 不管是否优化了,left都要增加。但至少要让窗口中保留一个有用的解,也就是while的条件left++;}}void updateAnswer(const string& s) {string tmp_answer = s.substr(left, right - left + 1);if (answer.empty()) { // 首次情况下answer = tmp_answer;} else if (tmp_answer.length() < answer.length()) { // 找到了更优化解的情况answer = tmp_answer;}}
};
解法二
// 解法二 //
// 优化answer的计算,不使用substr
// 因为答案是一个子串,是连续的,
// 所以使用start和len来保存即可,
// 到最后一刻,才使用substr从s之中截取子串出来class Solution {
public:struct MyAnswer {int start = INT_MAX;int len = INT_MAX;};MyAnswer answer; // 这道题的答案unordered_map<char, int> window; // 这就是滑动的窗口unordered_map<char, int> need; // 目标字符串 t 中每个字符的情况int left = 0, right = 0; // 用于滑动的指针int validCount = 0; // 「与字符串t中数量相等的字符」的数量string minWindow(string s, string t) {// 初始化 needfor (char c : t)need[c]++;// 核心思想: 不断增大right找到一个可行解;while (right < s.length()) {char R = s[right];// 先看这第一步:R 是我们需要的字符时if (need.count(R) == 1) { window[R]++; // 放到窗口之中if (window[R] == need[R]) { // 该字符的数量已经达到要求validCount++; // 已就绪的字符数量+1if (validCount == need.size()) { // 如果所有字符都就绪,更新一下answerupdateAnswer(s);}}}// 最后看这第三步// 你想啊,如果只是right++,left不动,那这个窗口岂不是越来越长了// 所以,要考虑一下什么时候left也要++// 核心思想:当前的窗口中,已经包含了不止一个解的时候,我们让left++,去优化一下// 因为题目要求找字符串s的最小字串// 比如:s="A-B-C---BAC--", t="ABC"// 不断的增加right指针,当这个窗口达到「A-B-C---BAC」时,// 这也是s的字串 且 也涵盖t中的所有字符// 但这不是最优解// 所以要增加left,让窗口变成「BAC」// 才是最优解moveLeftFindThePerfectAnswer(s, t);// 再看这第二步:R 不是我们需要的字符,则 right++ 继续寻找right++;}// 要考虑没有找到解的情况,返回空字符串return (answer.start == INT_MAX && answer.len == INT_MAX)? "" : s.substr(answer.start, answer.len);}void moveLeftFindThePerfectAnswer(const string& s, const string& t) {while (validCount == need.size()) { // 已经找到解的情况之下,才会优化解char L = s[left];if (need.count(L) == 1) { // 字符L是目标的其中一个字符时,代表可能可以优化updateAnswer(s);window[L]--;if (window[L] < need[L])validCount--;if (window[L] == 0)window.erase(L);}// 不管是否优化了,left都要增加。但至少要让窗口中保留一个有用的解,也就是while的条件left++;}}void updateAnswer(const string& s) {int new_length = right - left + 1;if (answer.start == INT_MAX && answer.len == INT_MAX) { // 首次情况下answer.start = left;answer.len = new_length;} else if (new_length < answer.len) { // 找到了更优化解的情况answer.start = left;answer.len = new_length;}}
};
作者:坤坤学编程
链接:时隔1年4个月之后,再次写的题解,比第一次要清晰很多。
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
相关文章:
【力扣题解】【76. 最小覆盖子串】容易理解版
76. 最小覆盖子串 总结和复盘 这是时隔1年4个月之后,再次写的题解,比第一次要清晰很多。 我刚开始,就是用方法一做的,提交之后报超出内存限制; 对方法一进行优化,得到方法二,提交之后就AC了。…...

Android10 音频参数导出合并
A10 设备录音时底噪过大,让音频同事校准了下,然后把校准好的参数需要导出来,集成到项目中,然后出包,导出方式在此记录 设备安装debug系统版本调试好后, adb root adb remount adb shell 进入设备目录 导…...

在 Windows 系统中如何快速进入安全模式的两种方法
在使用电脑的过程中,有时我们可能会遇到一些需要进入“安全模式”来解决的问题。安全模式是一种特殊的启动选项,它以最小化配置启动操作系统,仅加载最基本的驱动程序和服务,从而帮助用户诊断和修复系统问题。本文中简鹿办公将详细…...

计算机网络(1)基础篇
目录 1.TCP/IP 网络模型 2.键入网址--->网页显示 2.1 生成HTTP数据包 2.2 DNS服务器进行域名与IP转换 2.3 建立TCP连接 2.4 生成IP头部和MAC头部 2.5 网卡、交换机、路由器 3 Linux系统收发网络包 1.TCP/IP 网络模型 首先,为什么要有 TCP/IP 网络模型&a…...
自然语言处理NLP入门 -- 第四节文本分类
目标 本章的目标是帮助你理解文本分类的基本概念,并通过具体示例学习如何使用 scikit-learn 训练文本分类模型,以及如何利用 OpenAI API 进行文本分类。 5.1 什么是文本分类? 文本分类(Text Classification)是自然语…...

【redis】数据类型之bitmaps
Redis的Bitmaps是一种基于字符串的数据结构,用于处理位级别的操作。虽然Bitmaps在Redis中并不是一种独立的数据类型,而是基于字符串实现的,但它们提供了高效的位操作功能,适用于需要处理大量布尔值或二进制数据的场景。 基本概念…...

计算机网络-MPLS转发原理
在上一篇关于 MPLS 基础的文章中,我们了解了 MPLS 的基本概念、术语以及它在网络中的重要性。今天,我们将深入探讨 MPLS 转发的原理与流程,帮助大家更好地理解 MPLS 是如何在实际网络中工作的。 一、MPLS 转发概述 MPLS 转发的本质是将数据…...

5. 【.NET 8 实战--孢子记账--从单体到微服务--转向微服务】--微服务基础工具与技术--Nacos
一、什么是Nacos Nacos 是阿里巴巴开源的一款云原生应用基础设施,它旨在简化微服务架构中服务治理和配置管理的复杂性。通过 Nacos,服务在启动时可以自动注册,而其他服务则可以通过名称来查找并访问这些注册好的实例。同时,Nacos…...
【每日关注】科技圈重要动态
时代新动态 2025 年 2 月 12 日科技圈重要动态总结全球 AI 治理新进展巴黎 AI 宣言签署,美英缺席 科技巨头合作与竞争苹果联姻阿里开发中国版AI功能DeepSeek生态持续扩展OpenAI拒绝马斯克收购,矛盾公开化 汽车行业动态小米汽车销量跃居新势力第二比亚迪智…...
【算法】用C++实现A*算法
A*算法的背景与原理 A*(A-Star)算法是一种广泛应用于路径规划和图搜索问题中的启发式搜索算法。它结合了Dijkstra算法的广度优先搜索和贪心最佳优先搜索的优点,通过引入启发式函数来估计从当前节点到目标节点的成本,从而有效地减少搜索空间。A*算法的核心思想是使用一个评…...

细胞计数专题 | LUNA-FX7™新自动对焦算法提高极低细胞浓度下的细胞计数准确性
现代细胞计数仪采用自动化方法,在特定浓度范围内进行细胞计数。其上限受限于在高浓度条件下准确区分细胞边界的能力,而相机视野等因素则决定了下限。在图像中仅包含少量可识别细胞或特征的情况下,自动对焦可能会失效,从而影响细胞…...

记一次Self XSS+CSRF组合利用
视频教程在我主页简介或专栏里 (不懂都可以来问我 专栏找我哦) 目录: 确认 XSS 漏洞 确认 CSRF 漏洞 这个漏洞是我在应用程序的订阅表单中发现的一个 XSS 漏洞,只能通过 POST 请求进行利用。通常情况下,基于 POST 的…...
JVM 类加载子系统在干什么?
JVM 类加载子系统是什么? 类加载子系统(Class Loader Subsystem)是 JVM 负责 加载、链接和初始化 .class 文件的组件。它的主要作用是将字节码文件加载进 JVM 并准备执行。 类加载器(ClassLoader)是 字节码的搬运工&…...
Golang轻松实现消息模板变量替换:text/template
text/template 是 Go 语言标准库中的一个包,用于生成文本输出。它通过解析模板并根据给定的数据执行模板来生成最终的文本。text/template 提供了强大的模板引擎,支持条件判断、循环、变量替换等功能。 基本概念 模板:模板是一个文本文件或…...

DeepSeek模型R1服务器繁忙,怎么解决?
在当今科技飞速发展的时代,人工智能领域不断涌现出令人瞩目的创新成果,其中DeepSeek模型无疑成为了众多关注焦点。它凭借着先进的技术和卓越的性能,在行业内掀起了一股热潮,吸引了无数目光。然而,如同许多前沿技术在发…...
《探秘Windows 10驱动开发:从入门到实战》
《探秘Windows 10驱动开发:从入门到实战》 为什么要在 Windows 10 编写驱动程序 在当今数字化时代,计算机已成为人们生活和工作中不可或缺的工具 ,而 Windows 10 作为一款广泛使用的操作系统,其生态系统的丰富性和复杂性不言而喻。在这个庞大的体系中,驱动程序扮演着举足…...

Golang的容器化部署流程
# Golang的容器化部署流程 什么是容器化部署 容器化部署是将应用程序、运行环境及其依赖项打包在一起,以便可以在任何环境中快速、一致地运行的技术。它提供了更高效的资源利用、更便捷的部署和更稳定的环境。 的容器化支持 天生支持跨平台编译,使得将Go…...
计算机网络,大白话
好嘞,咱就从头到尾,给你好好说道说道计算机网络里这些“门门道道”的概念: 1. 网络(Network) 啥是网络? 你可以把网络想象成一个“大Party”,大家(设备)聚在一起&#…...

智慧城市V4系统小程序源码独立版全插件全开源
智慧城市V4系统小程序源码:多城市代理同城信息服务的全域解决方案 在数字化浪潮的推动下,智慧城市已成为全球发展的核心战略。作为这一领域的革新者,智慧城市V4系统小程序源码凭借其多城市代理同城信息服务能力与多商家营销功能,…...
SpringBoot分布式应用程序和数据库在物理位置分配上、路由上和数量上的最佳实践是什么?
在设计和部署Spring Boot分布式应用程序时,物理位置分配、路由和数据库数量的最佳实践对系统性能、可用性和可维护性至关重要。以下是相关建议: 1. 物理位置分配 最佳实践: 靠近用户部署:将应用实例部署在靠近用户的数据中心&a…...
云原生核心技术 (7/12): K8s 核心概念白话解读(上):Pod 和 Deployment 究竟是什么?
大家好,欢迎来到《云原生核心技术》系列的第七篇! 在上一篇,我们成功地使用 Minikube 或 kind 在自己的电脑上搭建起了一个迷你但功能完备的 Kubernetes 集群。现在,我们就像一个拥有了一块崭新数字土地的农场主,是时…...
树莓派超全系列教程文档--(62)使用rpicam-app通过网络流式传输视频
使用rpicam-app通过网络流式传输视频 使用 rpicam-app 通过网络流式传输视频UDPTCPRTSPlibavGStreamerRTPlibcamerasrc GStreamer 元素 文章来源: http://raspberry.dns8844.cn/documentation 原文网址 使用 rpicam-app 通过网络流式传输视频 本节介绍来自 rpica…...
FFmpeg 低延迟同屏方案
引言 在实时互动需求激增的当下,无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作,还是游戏直播的画面实时传输,低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架,凭借其灵活的编解码、数据…...
spring:实例工厂方法获取bean
spring处理使用静态工厂方法获取bean实例,也可以通过实例工厂方法获取bean实例。 实例工厂方法步骤如下: 定义实例工厂类(Java代码),定义实例工厂(xml),定义调用实例工厂ÿ…...

【2025年】解决Burpsuite抓不到https包的问题
环境:windows11 burpsuite:2025.5 在抓取https网站时,burpsuite抓取不到https数据包,只显示: 解决该问题只需如下三个步骤: 1、浏览器中访问 http://burp 2、下载 CA certificate 证书 3、在设置--隐私与安全--…...

CocosCreator 之 JavaScript/TypeScript和Java的相互交互
引擎版本: 3.8.1 语言: JavaScript/TypeScript、C、Java 环境:Window 参考:Java原生反射机制 您好,我是鹤九日! 回顾 在上篇文章中:CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...

Psychopy音频的使用
Psychopy音频的使用 本文主要解决以下问题: 指定音频引擎与设备;播放音频文件 本文所使用的环境: Python3.10 numpy2.2.6 psychopy2025.1.1 psychtoolbox3.0.19.14 一、音频配置 Psychopy文档链接为Sound - for audio playback — Psy…...
css3笔记 (1) 自用
outline: none 用于移除元素获得焦点时默认的轮廓线 broder:0 用于移除边框 font-size:0 用于设置字体不显示 list-style: none 消除<li> 标签默认样式 margin: xx auto 版心居中 width:100% 通栏 vertical-align 作用于行内元素 / 表格单元格ÿ…...
是否存在路径(FIFOBB算法)
题目描述 一个具有 n 个顶点e条边的无向图,该图顶点的编号依次为0到n-1且不存在顶点与自身相连的边。请使用FIFOBB算法编写程序,确定是否存在从顶点 source到顶点 destination的路径。 输入 第一行两个整数,分别表示n 和 e 的值(1…...

排序算法总结(C++)
目录 一、稳定性二、排序算法选择、冒泡、插入排序归并排序随机快速排序堆排序基数排序计数排序 三、总结 一、稳定性 排序算法的稳定性是指:同样大小的样本 **(同样大小的数据)**在排序之后不会改变原始的相对次序。 稳定性对基础类型对象…...