news 2026/4/30 11:35:59

D.二分查找-二分答案-求最大——1898. 可移除字符的最大数目

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
D.二分查找-二分答案-求最大——1898. 可移除字符的最大数目

题目链接:1898. 可移除字符的最大数目(中等)

算法原理:

此题的题意跟下面这题基本一模一样👇,只不过之前的求最小,这个是求最大

D.二分查找-二分答案-求最小——3639. 变为活跃状态的最小时间

解法:二分查找

96ms击败40.00%

时间复杂度O(Nlogn)

①目标变量:依次移除下标数

②目标条件:在保证p是否还是剩下子串的子序列的前提下,尽量多移除下标

③转换逻辑:移除前mid个下标对应的字符后,p是否还是剩下子串的子序列

具体步骤:

①确定边界:

left:0,最少就是一个都不移除

right:removable.length,最多就是把removable数组中的下标全移除

②确定二分模型:

移除下标数↑ p是剩余子串的子序列的概率↓ 呈负相关单调,由于是找最大移除下标数,因此采用最右端点模型

③check方法设计:

如果移除了mid个下标,如果p还是剩下子串的子序列就返回true,mid不动地方,否则就是移除多了,需要向左调整,这里我们采用标记数组,如果s数组中的字符被移除就标记为true,遍历时从左向右依次检查,如果未被移除,且能和p中字符对应,就相应指针往后挪,继续检查,当指针恰好把p走完,说明还有p的这个子序列

答疑

Q1:既然是一一对应检查,我把s和p都存进顺序表检查不可以吗?

有的小伙伴写check方法时可能会想到用顺序表,给s和p分别创建一个顺序表,利用Java自带的ArrayList移除中间元素后,后续元素会自动向前拼接的特性来判断s剩余字符串是否还存在子序列p,相关代码放下面了,但这有个致命错误,那就是下标偏移错误,举个例子👇

假设原始 s = "abcde"(下标 0-4),r = [2, 3](要移除原始下标 2 和 3,对应字符 'c' 和 'd')
第一步:tmp.remove(2) → tmp 变成 ["a","b","d","e"](原始 'd' 现在在 tmp 的下标 2 位置)
第二步:tmp.remove(3) → 你想删原始下标 3 的 'd',但现在 tmp 的下标 3 是 'e',结果删成了 'e',完全错了

Java代码:

class Solution { public int maximumRemovals(String s, String p, int[] r) { int left=0,right=r.length; while(left<right){ int mid=left+(right-left+1)/2; if(!check(mid,s,p,r)) right=mid-1; else left=mid; } return left; } //移除前mid个下标对应字符后,子序列还有p就返回true private boolean check(int mid,String s, String p, int[] r){ int n=s.length(); boolean[] removed=new boolean[n]; for(int i=0;i<mid;i++) removed[r[i]]=true; int pindex=0; for(int i=0;i<n&&pindex<p.length();i++) if(!removed[i]&&s.charAt(i)==p.charAt(pindex)) pindex++; return pindex==p.length(); } }
class Solution { //错误的示范代码 private static List<Character> listp=new ArrayList<>(); private static List<Character> lists=new ArrayList<>(); public int maximumRemovals(String s, String p, int[] r) { for(char c:p.toCharArray()) listp.add(c); for(char c:s.toCharArray()) lists.add(c); int left=0,right=r.length; while(left<right){ int mid=left+(right-left+1)/2; if(!check(mid,r)) right=mid-1; else left=mid; } //个数=下标+1 return left+1; } //移除前mid个下标对应字符后,子序列还有p就返回true private boolean check(int mid,int[] r){ List<Character> tmp=new ArrayList<>(lists); int index=0; for(int i=0;i<mid;i++) tmp.remove(r[i]); for(char c:tmp) if(c==listp.get(index)) index++; return index==listp.size()-1; } }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/29 6:44:39

CANN Runtime调试支持模块 算子中间结果保存与校验源码解析

摘要 调试AI模型就像医生做手术&#xff0c;得知道每个"器官"的运行状态。今天咱们就深入CANN Runtime的调试支持模块&#xff0c;看看它如何通过中间结果保存、数据校验、智能日志三大技术&#xff0c;让算子调试从"盲人摸象"变成"透明手术"。…

作者头像 李华
网站建设 2026/4/29 16:51:00

Java毕业设计新手避坑指南:从选题到部署的完整技术路径

Java毕业设计新手避坑指南&#xff1a;从选题到部署的完整技术路径 摘要&#xff1a;许多计算机专业学生在完成Java毕业设计时&#xff0c;常因缺乏工程经验陷入技术选型混乱、架构耦合严重或部署流程复杂等困境。本文面向零实战经验的新手&#xff0c;系统梳理从需求分析、技术…

作者头像 李华
网站建设 2026/4/23 14:14:30

Dify智能客服知识库回答限制实战:从配置到避坑指南

Dify智能客服知识库回答限制实战&#xff1a;从配置到避坑指南 一、背景与痛点&#xff1a;当客服“信口开河”时 在灰度测试阶段&#xff0c;我们曾把 Dify 智能客服直接接入官网 IM 通道&#xff0c;结果 30% 的访客提问得到的是“看似专业、实则 hallucination”的答案。典…

作者头像 李华
网站建设 2026/4/20 8:57:16

CosyVoice 技术解读:从零构建语音合成系统的核心原理与实战

CosyVoice 技术解读&#xff1a;从零构建语音合成系统的核心原理与实战 摘要&#xff1a;本文深入解析 CosyVoice 语音合成技术的核心架构与实现原理&#xff0c;针对开发者常见的语音自然度不足、延迟高等痛点&#xff0c;提供从模型训练到工程部署的完整解决方案。通过详细的…

作者头像 李华
网站建设 2026/4/29 9:41:22

基于Spring-AI-Alibaba构建智能客服系统的实战指南与架构解析

基于Spring-AI-Alibaba构建智能客服系统的实战指南与架构解析 摘要&#xff1a;本文针对传统客服系统开发效率低、智能化程度不足的痛点&#xff0c;详细讲解如何利用Spring-AI-Alibaba技术栈快速构建高可用智能客服系统。通过整合Alibaba NLP能力和Spring生态&#xff0c;开发…

作者头像 李华