当前位置: 首页 > news >正文

Leetcode 3049. Earliest Second to Mark Indices II

  • Leetcode 3049. Earliest Second to Mark Indices II
    • 1. 解题思路
    • 2. 代码实现
    • 3. 算法优化
  • 题目链接:3049. Earliest Second to Mark Indices II

1. 解题思路

这道题我看貌似难度报表,比赛的时候貌似只有36个人搞定了这道题目,然后最快的人耗时也花了40min以上,就很离谱,和平时基本15分钟就能有大佬全部做出来4道题的情况完全不懂,就很吓人。

所以基本上我对这道题也算是一个半放弃的状态,不过后来还是看了一下,然后发现其实也不是完全无法上手,本质上还是求一个处理的最优解,且显然这个操作依然是单调的,即从某一个位置开始,后续所有的操作序列都可以使得所有的位置都被mark上。

因此,最开始我的思路也是二分法来求任意一个位置作为操作序列的结束点时能否判断其是否可以使得所有的位置都被mark上,然后就和上一题一样,这里就变成了一个贪婪算法,要最优的分配点数使得所有的值都可以先降为0然后再mark上。

但是比较尴尬的是这里我们的贪婪算法并没有完成,因此有以下一个判断无法彻底想明白:

  • 已知任何一次直接归零操作都必须附加一次mark操作进行完成,因此对于任意一个点,我们需要判断是否要等待后续直接操作位然后再多加一轮操作使之mark还是直接使用当前富裕的点数(如果有的话)进行归零然后mark。

因此后续干脆就是直接绕开这个,走了最暴力的动态规划的方法,遍历了所有的可能,即每一个位置到底是走的快速清零还是进行-1或者mark操作,这里,显然如果要进行快速清零那必然是在某个位置在changeIndices当中最早出现的位置上。

但是,直接地实现会出现内存爆炸的情况,因此,我们做了一些不严谨的剪枝,即如果某次快速清零可以省出100以上的富裕,那么必走快速清零,反之考虑是否忽略掉这个快速清零的路线。

需要强调,这个方式是不严谨的,因为如果存在全面富裕了非常多的点数远大于100,而最后来一个可以清空100的操作,那不一定要走快速清零的,因为那样会需要额外多一次操作。

但是对于绝大多数的情况上述结果是对的,在这里也确实可以通过测试样例。

2. 代码实现

给出python代码实现如下:

class Solution:def earliestSecondToMarkIndices(self, nums: List[int], changeIndices: List[int]) -> int:n, m = len(nums), len(changeIndices)changeIndices = [x-1 for x in changeIndices]needed = sum(nums) + nfirst_seen = {}for i, x in enumerate(changeIndices):if x not in first_seen and nums[x] > 1:first_seen[x] = ifirst_seen = {v:k for k, v in first_seen.items()}@lru_cache(None)def dp(idx, need, extra):if need <= 1 and extra <= 1:return idxelif idx >= m:return mif idx not in first_seen:return dp(idx+1, need-1, max(extra-1, 0))i = first_seen[idx]if nums[i] >= 100:return dp(idx+1, need-nums[i], extra+1)else:return min(dp(idx+1, need-1, max(extra-1, 0)), dp(idx+1, need-nums[i], extra+1))ans = dp(0, needed, 0)return ans+1 if ans != m else -1

提交代码评测得到:耗时1219ms,占用内存288.9MB。

3. 算法优化

因为前面也反复强调了,上述解法有效,但是剪枝的存在使得上述解法并不严谨,因此,我们在这里摘录一下其他大佬们的解法作为补充,有兴趣的读者可以自行研读学习一下,这里就让我偷个懒了……

class Solution:def earliestSecondToMarkIndices(self, nums: List[int], changeIndices: List[int]) -> int:first_idx = [-1] * len(nums)for i in range(len(changeIndices)):if first_idx[changeIndices[i]-1] == -1:first_idx[changeIndices[i]-1] = ifor i, num in enumerate(nums):if num <= 1:first_idx[i] = -1default_needs = sum(nums) + len(nums)def is_possible(k):left = 0h = []pos_count = 0flag = Falsefor j in range(k, -1, -1):if flag:pos_count += 1flag = Falseelse:flag = Trueleft += 1i1 = first_idx[changeIndices[j]-1]if i1 == j:heapq.heappush(h, [nums[changeIndices[j]-1] - 1, changeIndices[j]-1])if len(h) > pos_count:heapq.heappop(h)return default_needs - sum(h1[0] for h1 in h) <= leftans = 0return ansl, r = len(nums)-1, len(changeIndices)-1while l < r:m = (l + r) // 2if is_possible(m):r = melse:l = m + 1if l > r or not is_possible(l):return -1return l+1
http://www.lryc.cn/news/311532.html

相关文章:

  • CrossOver 24下载-CrossOver 24 for Mac下载 v24.0.0中文永久版
  • 算法设计.
  • 20240304金融读报:票据贴现数据挖掘与新质生产力信贷创新
  • 05. Nginx入门-Nginx访问控制
  • S2---FPGA-A7板级原理图硬件实战
  • RK DVP NVP6158配置 学习
  • C++基础2:C++基本数据类型和控制结构
  • HFSS仿真双频微带天线学习笔记
  • 【十一】【SQL】外连接(左外连接,右外连接)
  • 敏捷开发模型:一种灵活、协作和持续的软件开发方法
  • 软件设计师10--计算机组成与体系结构章节回顾
  • 数据库分库分表中间件选择
  • 代码随想录算法训练营第22天|235.二叉搜索树的最近公共祖先、701.二叉搜索树中的插入操作、450.删除二叉搜索树中的节点
  • 基于SpringBoot的医护人员排班系统详细开题报告(源码)
  • CDH6.3.1离线安装
  • Pytorch之卷积操作
  • 2024年春招小红书前端实习面试题分享
  • 软件测试--性能测试工具JMeter
  • c++/c图的邻近矩阵表示
  • cocos-lua定时器用法
  • 激活函数Swish(ICLR 2018)
  • 【C++ 标准流,文件流】
  • 【排序】详解冒泡排序
  • 什么是Docker容器?
  • (C++练习)选择题+编程题
  • 【鸿蒙开发】第十五章 ArkTS基础类库-并发
  • 华为数通方向HCIP-DataCom H12-821题库(多选题:21-40)
  • 【简单模拟】第十三届蓝桥杯省赛C++ B组《刷题统计》(c++)
  • IO-DAY3
  • python实现常见一元随机变量的概率分布