前缀和优化DP,LeetCode 3177. 求出最长好子序列 II

目录

一、题目

1、题目描述

2、接口描述

python3

cpp

C#

3、原题链接

二、解题报告

1、思路分析

2、复杂度

3、代码详解

python3

cpp

C#


一、题目

1、题目描述

给你一个整数数组 nums 和一个 非负 整数 k 。如果一个整数序列 seq 满足在范围下标范围 [0, seq.length - 2] 中存在 不超过 k 个下标 i 满足 seq[i] != seq[i + 1] ,那么我们称这个整数序列为 序列。

请你返回 nums

子序列

的最长长度

2、接口描述

python3
复制代码
 ​
python 复制代码
class Solution:
    def maximumLength(self, nums: List[int], k: int) -> int:
cpp
复制代码
 ​
cpp 复制代码
class Solution {
public:
    int maximumLength(vector<int>& nums, int k) {
        
    }
};
C#
复制代码
 ​
cs 复制代码
public class Solution {
    public int MaximumLength(int[] nums, int k) {

    }
}

3、原题链接

3177. 求出最长好子序列 II


二、解题报告

1、思路分析

很容易想到状态定义:

f(x, j) 为 以x结尾有 j 个 坏pair 的最大好序列长度

那么有转移方程:

f(x, j) = f(y, j - 1) + 1,y != x

f(x, j) = f(x, j) + 1

每次从当前列或者左边列转移,我们考虑前缀优化

ma[j + 1] 代表 第 j 列最大值

那么 f(i, j) = max(f(i, j), ma(j)) + 1

ma(j) 可能会包含 f(x, j - 1),但是由于 f(x, j) >= f(x, j - 1),所以f(x, j) + 1 >= f(x, j - 1) + 1,不影响状态转移的正确性

2、复杂度

时间复杂度: O(NK)空间复杂度:O(NK)

3、代码详解

python3
python 复制代码
class Solution:
    def maximumLength(self, nums: list[int], k: int) -> int:
        f = defaultdict(list)
        ma = [0] * (k + 2)

        for x in nums:
            if x not in f:
                f[x] = [0] * (k + 1)
            nf = f[x]
            for j in range(k, -1, -1):
                nf[j] = max(nf[j], ma[j]) + 1
                ma[j + 1] = max(ma[j + 1], nf[j])

        return ma[-1]
cpp
复制代码
 ​
cpp 复制代码
class Solution {
public:
    int maximumLength(vector<int>& nums, int k) {
        std::unordered_map<int, std::vector<int>> f;
        std::vector<int> ma(k + 2);
        for (int x : nums) {
            for (int j = k; ~j; -- j) {
                if (!f.contains(x)) f[x] = std::vector<int>(k + 1);
                auto &nf = f[x];
                nf[j] = std::max(nf[j], ma[j]) + 1;
                ma[j + 1] = std::max(nf[j], ma[j + 1]);
            }
        }
        return ma.back();
    }
};
C#
复制代码
 ​
cs 复制代码
public class Solution {
    public int MaximumLength(int[] nums, int k) {
        int n = nums.Length;
        int[] ma = new int[k + 2];    // column max
        IDictionary<int, int[]> f = new Dictionary<int, int[]>();
        foreach(int x in nums) {
            if (!f.ContainsKey(x))
                f.Add(x, new int[k + 1]);
            var nf = f[x];
            for (int j = k; j >= 0; -- j) {
                nf[j] = Math.Max(nf[j], ma[j]) + 1;
                ma[j + 1] = Math.Max(ma[j + 1], nf[j]);
            }
        }
        return ma[^1];
    }
}
相关推荐
Dream it possible!17 分钟前
LeetCode 面试经典 150_数组/字符串_O(1)时间插入、删除和获取随机元素(12_380_C++_中等)(哈希表)
c++·leetcode·面试·哈希表
快去睡觉~22 分钟前
力扣137:只出现一次的数字Ⅱ
数据结构·算法·leetcode
阑梦清川25 分钟前
folo介绍和fluent reader阅读器的使用(RSS订阅技术)
算法
2501_924879361 小时前
密集表盘漏检率↓79%!陌讯多模态融合算法在电表箱状态识别的边缘优化
人工智能·算法·计算机视觉·目标跟踪·智慧城市
白白白鲤鱼2 小时前
Vue2项目—基于路由守卫实现钉钉小程序动态更新标题
服务器·前端·spring boot·后端·职场和发展·小程序·钉钉
Sunlightʊə2 小时前
05.LinkedList与链表
java·数据结构·算法·链表
qq_513970442 小时前
力扣 hot100 Day67
算法·leetcode·职场和发展
音符犹如代码2 小时前
《四种姿势用Java玩转AI大模型:从原生HTTP到LangChain4j》
java·人工智能·spring·职场和发展
Cx330❀2 小时前
【数据结构初阶】--单链表(二)
数据结构·经验分享·算法·leetcode
chy存钱罐3 小时前
模型拟合问题全解析:从欠拟合、过拟合到正则化(岭回归与拉索回归)
人工智能·算法·机器学习·数据挖掘·回归