LeetCode 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.length
  • 1 <= n <= 100
  • 1 <= strs[i].length <= 100
  • strs[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 ( s i z e ( s t r s ) ) O(size(strs)) O(size(strs))
  • 空间复杂度 O ( n ) O(n) O(n)

AC代码

C++
cpp 复制代码
/*
 * @LastEditTime: 2025-12-21 20:46:33
 */
class Solution {
public:
    int minDeletionSize(vector<string>& strs) {
        int ans = 0;
        int n = strs.size(), m = strs[0].size();
        int totalSkip = 1;  // 第一行不用校验
        vector<bool> canSkip(n);
        for (int j = 0; j < m; j++) {
            bool can = true;
            for (int i = 1; i < n; i++) {
                if (canSkip[i]) {
                    continue;
                }
                if (strs[i][j] < strs[i-1][j]) {
                    can = false;
                    break;
                }
            }
            if (can) {
                for (int i = 1; i < n; i++) {
                    if (strs[i][j] != strs[i-1][j] && !canSkip[i]) {
                        canSkip[i] = true;
                        totalSkip++;
                        if (totalSkip == n) {
                            return ans;
                        }
                    }
                }
            } else {
                ans++;
            }
        }
        return ans;
    }
};
  • 执行用时分布0ms击败100.00%
  • 消耗内存分布12.57MB击败81.82%

同步发文于CSDN和我的个人博客,原创不易,转载经作者同意后请附上原文链接哦~

千篇源码题解已开源

相关推荐
A_nanda8 小时前
c# MOdbus rto读写串口,如何不相互影响
算法·c#·多线程
代码雕刻家10 小时前
2.4.蓝桥杯-分巧克力
算法·蓝桥杯
Ulyanov10 小时前
顶层设计——单脉冲雷达仿真器的灵魂蓝图
python·算法·pyside·仿真系统·单脉冲
智者知已应修善业12 小时前
【查找字符最大下标以*符号分割以**结束】2024-12-24
c语言·c++·经验分享·笔记·算法
91刘仁德12 小时前
c++类和对象(下)
c语言·jvm·c++·经验分享·笔记·算法
diediedei12 小时前
模板编译期类型检查
开发语言·c++·算法
阿杰学AI12 小时前
AI核心知识78——大语言模型之CLM(简洁且通俗易懂版)
人工智能·算法·ai·语言模型·rag·clm·语境化语言模型
mmz120713 小时前
分治算法(c++)
c++·算法
睡一觉就好了。13 小时前
快速排序——霍尔排序,前后指针排序,非递归排序
数据结构·算法·排序算法
Tansmjs13 小时前
C++编译期数据结构
开发语言·c++·算法