Leetcode 1671. Minimum Number of Removals to Make Mountain Array

本文介绍了如何解决最小山峰移除问题,通过两种版本的算法实现,Version1和Version2,分别利用了二分查找和最长递增子序列(LIS)来确定山脉长度,从而计算出需要删除的元素数量。Version2的优化显著提高了速度。博客作者Tyan分享了详细的代码实现,并提供了LeetCode题目链接作为参考。

文章作者:Tyan
博客:noahsnail.com  |  优快云  |  简书

1. Description

Minimum Number of Removals to Make Mountain Array

2. Solution

**解析:**Version 1,分别以数组中的元素作为中心点,在左右两侧分别求最长递增子序列,根据左右两侧的最长递增子序列的长度求出山脉的长度,则要删除的元素个数为数组长度减去最长的山脉长度,速度太慢。Version 2在Version 1的基础上进行了优化,分别求出数组正序和逆序各个位置的最长递增子序列,然后跟Version 1类似,累加左右对应位置的最长递增子序列的长度,即为山脉的长度,则要删除的元素个数为数组长度减去最长的山脉长度,速度明显有了大幅提升。

  • Version 1
class Solution:
    def minimumMountainRemovals(self, nums: List[int]) -> int:
        n = len(nums)
        maximum = 0
        for index in range(1, n - 1):
            left = [nums[index]]
            for i in range(index - 1, -1, -1):
                if nums[i] >= nums[index] or nums[i] == left[0]:
                    continue
                elif nums[i] < left[0]:
                    bisect.insort(left, nums[i])
                else:
                    pos = bisect.bisect(left, nums[i])
                    left[pos-1] = nums[i]
            if len(left) < 2:
                continue
            right = [nums[index]]
            for i in range(index+1, n):
                if nums[i] >= nums[index] or nums[i] == right[0]:
                    continue
                if nums[i] < right[0]:
                    bisect.insort(right, nums[i])
                else:
                    pos = bisect.bisect(right, nums[i])
                    right[pos-1] = nums[i]
            if len(right) > 1:
                maximum = max(maximum, len(left) + len(right) - 1)
        return n - maximum
  • Version 2
class Solution:
    def minimumMountainRemovals(self, nums: List[int]) -> int:
        n = len(nums)
        maximum = 0
        left = self.LIS(nums)
        right = self.LIS(nums[::-1])
        for i in range(n):
            if left[i] > 1 and right[n-i-1] > 1:
                maximum = max(maximum, left[i] + right[n-i-1] - 1)
        return n - maximum


    def LIS(self, nums):
        n = len(nums)
        dp = [1] * n
        arr = [nums[0]]
        for i in range(1, n):
            if nums[i] > arr[-1]:
                arr.append(nums[i])
            else:
                pos = bisect.bisect_left(arr, nums[i])
                arr[pos] = nums[i]
            dp[i] = len(arr)
        return dp

Reference

  1. https://leetcode.com/problems/minimum-number-of-removals-to-make-mountain-array/
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值