news 2026/6/10 1:49:39

力扣983最低票价 - 一维DP - 值域爬楼梯与二分优化

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
力扣983最低票价 - 一维DP - 值域爬楼梯与二分优化

983. 最低票价
这题可以看成「爬楼梯」题目的变种。
有两种思考角度,每种角度有两种写法。

角度一

我们从旅游的第一天i ii开始思考,n nn为旅行的最后一天,寻找子问题,分类讨论:

  • 在第i ii天购买1 11天的车票,接下来要解决:从i + 1 i+1i+1n nn天的最小花费
  • 在第i ii天购买7 77天的车票,接下来要解决:从i + 7 i+7i+7n nn天的最小花费
  • 在第i ii天购买30 3030天的车票,接下来要解决:从i + 30 i+30i+30n nn天的最小花费
    这和「爬楼梯」选择爬几层的思路很像。

写法一

本题数据范围很小,n ≤ 365 n≤365n365,故可以直接以自然日为下标,用动态规划在自然日上推进。
定义d f s ( i ) dfs(i)dfs(i)表示i iin nn天的最小花费。

  • 若第i ii天不是旅游日,就有:
    d f s ( i ) = d f s ( i + 1 ) dfs(i) = dfs(i + 1)dfs(i)=dfs(i+1)
  • 若第i ii天是旅游日,则有:
    d f s ( i ) = m i n ( d f s ( i + 1 ) + c o s t s [ 0 ] , d f s ( i + 7 ) + c o s t s [ 1 ] , d f s ( i + 30 ) + c o s t s [ 2 ] ) dfs(i) = min(dfs(i+1)+costs[0], dfs(i+7)+costs[1], dfs(i+30)+costs[2])dfs(i)=min(dfs(i+1)+costs[0],dfs(i+7)+costs[1],dfs(i+30)+costs[2])
  • 递归边界:当i > n i>ni>nd f s ( i ) = 0 dfs(i)=0dfs(i)=0,此时没有要旅行的天数
  • 递归入口:d f s ( D f i r s t ) dfs(D_{first})dfs(Dfirst),其中D f i r s t D_{first}Dfirst为旅行的第一天
    当然也可以直接把旅行第一天定为1 11,最后一天定为365 365365

加上记忆化就有:

classSolution{boolean[]vis=newboolean[366];int[]cache=newint[366];int[]costs;publicintmincostTickets(int[]days,int[]costs){this.costs=costs;for(intx:days)vis[x]=true;Arrays.fill(cache,-1);returndfs(1);}privateintdfs(inti){if(i>365)return0;if(cache[i]!=-1)returncache[i];if(vis[i]!=true)returncache[i]=dfs(i+1);intc1=dfs(i+1)+costs[0];intc2=dfs(i+7)+costs[1];intc3=dfs(i+30)+costs[2];intres=Math.min(c1,Math.min(c2,c3));cache[i]=res;returnres;}}

时间复杂度:O ( 365 ) O(365)O(365)O ( D l a s t − D f i r s t ) O(D_{last}-D_{first})O(DlastDfirst)
空间复杂度:O ( 365 ) O(365)O(365)O ( D l a s t − D f i r s t ) O(D_{last}-D_{first})O(DlastDfirst)

递推形式:

classSolution{boolean[]vis=newboolean[366];publicintmincostTickets(int[]days,int[]costs){for(intx:days)vis[x]=true;int[]f=newint[370];f[366]=0;for(inti=365;i>=1;i--){if(vis[i]!=true){f[i]=f[i+1];continue;}intc1=f[Math.min(i+1,366)]+costs[0];intc2=f[Math.min(i+7,366)]+costs[1];intc3=f[Math.min(i+30,366)]+costs[2];f[i]=Math.min(c1,Math.min(c2,c3));}returnf[1];}}

写法二

d a y s [ i ] days[i]days[i]比较大时,比如≤ 1 0 9 ≤10^{9}109时,上面以值域推进的做法就不行了。
此时用日期索引上做「跳跃」,就能解决复杂度与d a y s [ i ] days[i]days[i]值域有关的问题了。

大体思路是一样的,我们定义d f s ( i ) dfs(i)dfs(i)为第d a y s [ i ] days[i]days[i]天到第d a y s [ d a y s . l e n g t h − 1 ] days[days.length-1]days[days.length1]的旅行最小花费。只需要在「跳跃」时,跳跃到第一个≥ d a y s [ i ] + ( 1 , 7 , 30 ) ≥days[i]+(1,7,30)days[i]+(1,7,30)的索引j jj就行,这可以用二分优化。

记忆化搜索:

classSolution{int[]cache,costs,days;intn;publicintmincostTickets(int[]days,int[]costs){this.costs=costs;this.n=days.length;this.days=days;cache=newint[n+10];Arrays.fill(cache,-1);returndfs(0);}privateintdfs(inti){if(i>=n)return0;if(cache[i]!=-1)returncache[i];intc1=dfs(lowerBound(i,1))+costs[0];intc2=dfs(lowerBound(i,7))+costs[1];intc3=dfs(lowerBound(i,30))+costs[2];intres=Math.min(c1,Math.min(c2,c3));cache[i]=res;returnres;}// 左闭右开privateintlowerBound(inti,intday){intl=i+1,r=n;while(l<r){intmid=l+r>>1;if(days[mid]>=days[i]+day)r=mid;elsel=mid+1;}returnr;}}

时间复杂度:O ( n l o g n ) O(nlogn)O(nlogn),其中n nnd a y s daysdays的长度。
空间复杂度:O ( n ) O(n)O(n)

角度二

上面我们是从第一天开始入题,从左往右思考。我们也可以从右往左思考。
我们从旅游的最后一天i ii开始思考,寻找子问题,分类讨论:

  • 在第i ii天购买1 11天的车票,接下来要解决:从1 11i − 1 i-1i1天的最小花费
  • 在第i − 7 + 1 i-7+1i7+1天购买7 77天的车票,接下来要解决:从1 11i − 7 i-7i7天的最小花费
  • 在第i − 30 + 1 i-30+1i30+1天购买30 3030天的车票,接下来要解决:从1 11i − 30 i-30i30天的最小花费
    这种方法更方便把递归翻译成递推。

定义d f s ( i ) dfs(i)dfs(i)为第1 11i ii天的最小花费,后续思考大同小异,不再赘述。给出以值域推进的代码:

classSolution{boolean[]vis=newboolean[366];int[]cache=newint[366];int[]costs;publicintmincostTickets(int[]days,int[]costs){this.costs=costs;for(intx:days)vis[x]=true;Arrays.fill(cache,-1);returndfs(365);}privateintdfs(inti){if(i<=0)return0;if(cache[i]!=-1)returncache[i];if(vis[i]!=true)returncache[i]=dfs(i-1);intc1=dfs(i-1)+costs[0];intc2=dfs(i-7)+costs[1];intc3=dfs(i-30)+costs[2];intres=Math.min(c1,Math.min(c2,c3));cache[i]=res;returnres;}}

#solutions

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/9 14:57:04

【Python新手村】集合(Set):一个强迫症晚期的“去重大师”

Python 集合(Set)&#xff1a;一个强迫症晚期的“去重大师”哈喽&#xff0c;各位 Python 探险家&#xff01;前面我们认识了列表&#xff08;什么都装的收纳箱&#xff09;和元组&#xff08;上了锁的保险箱&#xff09;。 今天&#xff0c;我们要介绍一位性格非常古怪的朋友—…

作者头像 李华
网站建设 2026/6/9 20:12:27

自动驾驶如何遵守交通规则?揭秘AI驾驶员的伦理与算法博弈

第一章&#xff1a;自动驾驶 Agent 的交通规则在自动驾驶系统中&#xff0c;Agent 必须严格遵守交通规则以确保行驶安全与合规。这些规则不仅包括通用的道路标志识别和信号灯响应&#xff0c;还涵盖动态环境中的行为决策逻辑。感知与决策协同机制 自动驾驶 Agent 依赖多传感器融…

作者头像 李华
网站建设 2026/6/9 20:06:50

RNOpenHarmony:本地化MQTT同行通信(系列二)-架构与消息流

延续系列一&#xff0c;这篇我们深入聊聊架构设计、主题命名、QoS 选择、会话管理这些“硬核”内容。还是用“客户端 SDK / 服务端 SDK”作为代称&#xff0c;避免暴露真实项目名称。 说实话&#xff0c;架构设计这块&#xff0c;我一开始也是“摸着石头过河”。主题怎么命名&a…

作者头像 李华
网站建设 2026/6/9 20:14:04

系统思考与科学决策

在老板电器供应链团队完成了《系统思考与科学决策》的训练。 很多管理者都有同一种感受&#xff1a;每天都在救火&#xff0c;而且越救&#xff0c;火好像越多。 系统思考不是教大家慢下来&#xff0c;而是帮助我们看清&#xff1a;哪些火是外部的&#xff0c;哪些火&#xff0…

作者头像 李华
网站建设 2026/6/9 20:12:39

探秘C#多态:函数重载与符号重载

第十四次一&#xff0c;多态之函数重载1&#xff0c;多态 : 同一个方法&#xff0c;不同形态体现2&#xff0c;多态分为 &#xff1a; 静态多态和动态多态3&#xff0c;静态多态&#xff1a; 函数重载和符号重载4&#xff0c;动态多态&#xff1a; 抽象和虚方法5&#xff0c;函…

作者头像 李华
网站建设 2026/6/8 17:52:29

为什么顶尖机构都在用气象 Agent?揭秘其预测精度领先行业30%的秘密

第一章&#xff1a;气象 Agent 的预测精度气象 Agent 作为智能环境感知系统的重要组成部分&#xff0c;其预测精度直接决定了后续决策的可靠性。高精度的气象预测不仅依赖于高质量的历史数据&#xff0c;还需要先进的算法模型与实时反馈机制协同工作。影响预测精度的关键因素 数…

作者头像 李华