最长递增子序列 -- 动规

300. 最长递增子序列

注意「⼦序列」和「⼦串」的区别,⼦串⼀定是连续的,⽽⼦序列不⼀定是连续的。

python 复制代码
class LengthOfLIS:
    """
    300. 最长递增子序列
    https://leetcode.cn/problems/longest-increasing-subsequence/description/
    """

    def solution(self, nums: List[int]):
        """
        方案一: 动态规划
        辅助数组 dp,dp[i] 表示以nums[i]结尾的子序列的最大长度
        时间复杂度O(N^2)
        空间复杂度O(N)
        :param nums:
        :return:
        """
        n = len(nums)
        dp = [1] * n
        for i in range(1, n):
            tmp = 0
            for j in range(0, i):
                if nums[j] < nums[i]:
                    tmp = max(tmp, dp[j])
            dp[i] = tmp + 1
        return max(dp)

    def solution2(self, nums: List[int]):
        """
        方案二:动态规划
        辅助数组 ends,ends[i] 代表目前所有长度为i+1的递增子序列的最小结尾。
        可推断出 ends 也是递增序列
        时间复杂度O(N*logN)
        空间复杂度O(N)
        :param nums:
        :return:
        """
        if not nums:
            return 0

        n = len(nums)
        ends = [0] * n
        ends[0] = nums[0]
        l, r, m, right = 0, 0, 0, 0
        res = 1
        for i in range(1, n):
            l = 0
            r = right
            while l <= r:
                m = (l + r) // 2
                if nums[i] > ends[m]:
                    l = m + 1
                else:
                    r = m - 1
            print(right, l)
            right = max(right, l)
            ends[l] = nums[i]
            res = max(res, l + 1)

        return res

    def solution3(self, nums: List[int]) -> int:
        """
        模拟蜘蛛纸牌
        :param nums:
        :return:
        """
        top = [0] * len(nums)
        #  牌堆数初始化为 0
        piles = 0
        for i in range(len(nums)):
            poker = nums[i]

            left, right = 0, piles
            while left < right:
                mid = left + (right - left) // 2
                if top[mid] > poker:
                    right = mid
                elif top[mid] < poker:
                    left = mid + 1
                else:
                    right = mid

            # 没找到合适的牌堆,新建⼀堆
            if left == piles:
                piles += 1
            # 把这张牌放到牌堆顶
            top[left] = poker

        return piles
相关推荐
小范自学编程19 小时前
算法训练营 Day38 - 动态规划part07
算法·动态规划
2301_766558651 天前
‘小批量快反’为何成功能母粒新护城河?福尔蒂柔性产线调度系统架构图首度开源
系统架构·动态规划·宽度优先
2501_924878731 天前
并非所有‘黑色母‘都是碳黑分散体:颜料包覆率、熔指匹配与螺杆剪切窗口解析(含实测对比)
深度优先·动态规划
liuyao_xianhui1 天前
优选算法_丢失的数字_位运算_C++
linux·数据结构·c++·算法·动态规划·哈希算法·散列表
沉睡的无敌雄狮1 天前
AIoT器件小型化:福尔蒂微米级荧光母粒实现0.3mm壁厚精准识别|项目实战
深度优先·动态规划
liuyao_xianhui1 天前
动态规划_最长递增子序列_C++
java·开发语言·数据结构·c++·算法·链表·动态规划
2501_924878732 天前
注塑件银纹/流痕/色差如何归因?汽车零部件厂故障分析与快响方案复盘
汽车·排序算法·动态规划
2501_924878732 天前
福尔蒂汽车内饰母粒通过大众VW80101全项验证的方法论
汽车·动态规划·宽度优先
liuyao_xianhui2 天前
动态规划_最大子数组和_C++
java·开发语言·数据结构·c++·算法·链表·动态规划
Darkwanderor3 天前
数据结构——ST表和RMQ问题
数据结构·c++·动态规划·st表·rmq问题