955.删列造序 II:模拟(O(mn)) + 提前退出
【LetMeFly】955.删列造序 II:模拟(O(mn)) + 提前退出
力扣题目链接:https://leetcode.cn/problems/delete-columns-to-make-sorted-ii/
给定由 n 个字符串组成的数组 strs,其中每个字符串长度相等。
选取一个删除索引序列,对于 strs 中的每个字符串,删除对应每个索引处的字符。
比如,有 strs = ["abcdef", "uvwxyz"],删除索引序列 {0, 2, 3},删除后 strs 为["bef", "vyz"]。
假设,我们选择了一组删除索引 answer,那么在执行删除操作之后,最终得到的数组的元素是按 字典序(strs[0] <= strs[1] <= strs[2] ... <= strs[n - 1])排列的,然后请你返回 answer.length 的最小可能值。
示例 1:
输入:strs = ["ca","bb","ac"] 输出:1 解释: 删除第一列后,strs = ["a", "b", "c"]。 现在 strs 中元素是按字典排列的 (即,strs[0] <= strs[1] <= strs[2])。 我们至少需要进行 1 次删除,因为最初 strs 不是按字典序排列的,所以答案是 1。
示例 2:
输入:strs = ["xc","yb","za"] 输出:0 解释: strs 的列已经是按字典序排列了,所以我们不需要删除任何东西。 注意 strs 的行不需要按字典序排列。 也就是说,strs[0][0] <= strs[0][1] <= ... 不一定成立。
示例 3:
输入:strs = ["zyx","wvu","tsr"] 输出:3 解释: 我们必须删掉每一列。
提示:
n == strs.length1 <= n <= 1001 <= strs[i].length <= 100strs[i]由小写英文字母组成
解题方法:模拟
解题思路
先来理解题意:
删除尽可能少的列,使得上面一行字符串的字典序不大于下面一行。
什么时候上面(a)字典序不大于下面一行(b)呢?
从前往后遍历两个字符串:
- 一旦遇到
a[i] > b[i],直接不符- 一旦遇到
a[i] > b[i],直接符合(后面不用看了)- 如果遇到
a[i] = b[i],目前符合,继续往后看
好了,现在开始一列一列地遍历,去保留满足条件的列,删去不满足条件的列吧!
具体做法
使用一个布尔数组canSkip,其中canSkip[i]为true则代表第i行可以跳过与上一行的比较。
第一层循环枚举列,一列一列开始遍历;
第二层循环枚举行,从第二行开始枚举到最后一行:
- 如果当前行被skip则直接跳过当前行
- 否则,如果当前行小于上一行,这一列必须移除,退出第二层循环,删除这行(实际是答案数量+1)
- 如果遍历到最后一行仍然满足条件,则这一列无需移除,再次遍历这一列,如果下一行大于上一行,则下一行永远不会比上一行字典序大,下次不用遍历“下一行”了,将
canSkip[下一行]设置为true。
特殊的,我们可以记录下一共有多少行可以跳过,当所有行都可以被跳过时,后续再也不用遍历了,直接返回答案即可。
时空复杂度
- 时间复杂度$O(size(strs))$
- 空间复杂度$O(n)$
AC代码
C++
1 | |
- 执行用时分布0ms击败100.00%
- 消耗内存分布12.57MB击败81.82%
同步发文于CSDN和我的个人博客,原创不易,转载经作者同意后请附上原文链接哦~
千篇源码题解已开源
955.删列造序 II:模拟(O(mn)) + 提前退出
https://blog.letmefly.xyz/2025/12/21/LeetCode 0955.删列造序II/