Leetcode659. 分割数组为连续子序列

Every day a Leetcode

题目来源:659. 分割数组为连续子序列

解法1:哈希 + 贪心

定义两个哈希表:

  • numsCount:统计数组 nums 中各元素出现次数。
  • tailCount:存储以数字 i 结尾的且符合题意的连续子序列个数。

算法:

  1. 先去寻找一个长度为3的连续子序列 i,i+1,i+2,找到后就将 numsCount[i],numsCount[i+1],numsCount[i+2] 中对应数字消耗 1 个(即 -1),并将 tail[i+2] 加 1,即以 i+2 结尾的子序列个数 +1。
  2. 如果后续发现有能够接在这个连续子序列的数字 i+3,则延长以 i+2 为结尾的连续子序列到 i+3,此时消耗 numsCount[i+3] 一个,由于子序列已延长,因此 tailCount[i+2] 减 1,tailCount[i+3] 加 1。
  3. 在不满足上面的情况下:
    • 如果 numsCount[i] == 0,说明这个数字已经消耗完,可以不管了。
    • 如果 numsCount[i] != 0,说明这个数字多出来了,且无法组成连续子序列,直接返回 false。

因此,只有检查到某个数时,这个数未被消耗完,且既不能和前面组成连续子序列,也不能和后面组成连续子序列时,无法分割。

关于上面 1 和 2 的优先度,也就是说,当遇到一个新数,是新建一个长度为 3 的子序列好,还是补充到原有子序列的末尾好?

优先开新序列,不够贪,会丢失部分子序列尾的位置,而这可能造成丢解。

优先补前面更贪心,这种做法会覆盖优先开新序列时的尾的位置,补这补这总能到,补的过程中出现的尾位置也不会丢失。

代码:

c 复制代码
/*
 * @lc app=leetcode.cn id=659 lang=cpp
 *
 * [659] 分割数组为连续子序列
 */

// @lc code=start
class Solution
{
public:
    bool isPossible(vector<int> &nums)
    {
        if (nums.size() < 3)
            return false;
        unordered_map<int, int> numsCount, tailCount;
        // 统计数组 nums 中各元素出现次数
        for (int &num : nums)
            numsCount[num]++;
        // tailCount[i] 表示以 i 结尾的符合条件的子序列个数
        for (int &num : nums)
        {
            if (numsCount[num] == 0)
                continue;
            else if (numsCount[num] > 0 && tailCount[num - 1] > 0)
            {
                // 1. 补充到已有子序列的尾部
                numsCount[num] -= 1;
                tailCount[num - 1] -= 1;
                tailCount[num] += 1;
            }
            else if (numsCount[num] > 0 && numsCount[num + 1] > 0 && numsCount[num + 2] > 0)
            {
                // 2. 新建一条长度为 3 的子序列
                // 注意 1 的优先级比 2 高
                numsCount[num] -= 1;
                numsCount[num + 1] -= 1;
                numsCount[num + 2] -= 1;
                tailCount[num + 2] += 1;
            }
            else
                return false;
        }
        return true;
    }
};
// @lc code=end

结果:

复杂度分析:

时间复杂度:O(n),其中 n 是数组 nums 的长度。

空间复杂度:O(n),其中 n 是数组 nums 的长度。

相关推荐
小飞猪Jay1 小时前
C++面试速通宝典——13
jvm·c++·面试
rjszcb2 小时前
一文说完c++全部基础知识,IO流(二)
c++
小字节,大梦想3 小时前
【C++】二叉搜索树
数据结构·c++
吾名招财3 小时前
yolov5-7.0模型DNN加载函数及参数详解(重要)
c++·人工智能·yolo·dnn
我是哈哈hh3 小时前
专题十_穷举vs暴搜vs深搜vs回溯vs剪枝_二叉树的深度优先搜索_算法专题详细总结
服务器·数据结构·c++·算法·机器学习·深度优先·剪枝
憧憬成为原神糕手3 小时前
c++_ 多态
开发语言·c++
郭二哈3 小时前
C++——模板进阶、继承
java·服务器·c++
Tisfy3 小时前
LeetCode 2187.完成旅途的最少时间:二分查找
算法·leetcode·二分查找·题解·二分
挥剑决浮云 -4 小时前
Linux 之 安装软件、GCC编译器、Linux 操作系统基础
linux·服务器·c语言·c++·经验分享·笔记
Mephisto.java4 小时前
【力扣 | SQL题 | 每日四题】力扣2082, 2084, 2072, 2112, 180
sql·算法·leetcode