174. 地下城游戏(逆向DP)
Q:
一些恶魔抓住了公主(P)并将她关在了地下城的右下角。地下城是由 M x N 个房间组成的二维网格。我们英勇的骑士(K)最初被安置在左上角的房间里,他必须穿过地下城并通过对抗恶魔来拯救公主。
骑士的初始健康点数为一个正整数。如果他的健康点数在某一时刻降至 0 或以下,他会立即死亡。
有些房间由恶魔守卫,因此骑士在进入这些房间时会失去健康点数(若房间里的值为负整数,则表示骑士将损失健康点数);其他房间要么是空的(房间里的值为 0),要么包含增加骑士健康点数的魔法球(若房间里的值为正整数,则表示骑士将增加健康点数)。
为了尽快到达公主,骑士决定每次只向右或向下移动一步。
编写一个函数来计算确保骑士能够拯救到公主所需的最低初始健康点数。
例如,考虑到如下布局的地下城,如果骑士遵循最佳路径 右 -> 右 -> 下 -> 下,则骑士的初始健康点数至少为 7。
-2 (K) -3 3
-5 -10 1
10 30 -5 §
说明:
骑士的健康点数没有上限。
任何房间都可能对骑士的健康点数造成威胁,也可能增加骑士的健康点数,包括骑士进入的左上角房间以及公主被监禁的右下角房间。
A:
又自己想了一下为什么不能用正向DP。原因在于后效性这点不符合。假设我们正向DP,即从左上角向右下角前进,dp[i][j]表示从左上角到该位置最小需要的健康值。
例子:

我们假设考虑dp[2][0],从上面-2,-5过来,最少需要8滴血,那么dp[i][j]就填8吗,还是填1(毕竟这个房间可以加10滴血,只要来的时候不死就行了)?这里就是问题所在。对于某个状态(即某个确切的i,j坐标),它对于之后的路径是有影响的。因为除了考虑安全到达当前状态之外,还需要考虑从当前状态不死安全抵达公主。(即怎么到达当前状态会对之后的状态造成影响,这就不符合动态规划的无后效性了)
顺便粘贴一个知乎的回答:https://www.zhihu.com/question/43361359/answer/129799087
从不同的路径走到一个共同状态,而后续的状态变迁都是一样的,和之前采用何种路径到这个状态没有关系,即前面的各种决策结果由这个状态表示,在考虑后半段的决策方面没有任何区别
那么为什么逆序就可以?因为动态规划是要求无后效性,而不要求无前效性(虽然没有这个名词)。我们逆序的dp[i][j]表示从当前位置开始,安全抵达公主所需的最少健康值,不考虑怎么到达i,j,而是只关心从i,j开始如何抵达公主。可以看到,我们对于状态的定义不包含该状态之前的事件,也就不会有什么当前状态之前的事件对当前状态之后 造成影响!
这是我的想法,如果有懂这方面的大佬麻烦评论指点一哈
代码:
class Solution:
def calculateMinimumHP(self, dungeon ) -> int:
m,n=len(dungeon),
if not m:
return
n=len(dungeon[])
dp=[[ for i in range(n)] for j in range(m)] #m行n列
#dp[i][j]为从当前房间出发能不死抵达公主所需的最低健康点数
dp[-][-]=max(,-dungeon[-][-])
for i in range(m-,-,-):
for j in range(n-,-,-):
if i==m- and j==n-:
continue
x1,x2=float('inf'),float('inf')
if i+<m:
x1=max(,dp[i+][j]-dungeon[i][j])
if j+<n:
x2=max(,dp[i][j+]-dungeon[i][j])
dp[i][j]=min(x1,x2)
# for x in dp:
# print(x)
return dp[][]
把最后一行、最后一列单独判断一下,别人的代码:
class Solution:
def calculateMinimumHP(self, dungeon: List[List[int]]) -> int:
dp = [[ for i in range(len(dungeon[]))] for j in range(len(dungeon))]
m = len(dungeon)
n = len(dungeon[])
dp[-][-] = max(, - dungeon[-][-])
for i in range(m-,-,-):
dp[i][-] = max(,dp[i+][-]-dungeon[i][-])
for j in range(n-,-,-):
dp[-][j] = max(, dp[-][j+] - dungeon[-][j])
for i in range(m-,-,-):
for j in range(n-,-,-):
dp[i][j] = max(,min(dp[i+][j],dp[i][j+])-dungeon[i][j])
return dp[][]
174. 地下城游戏(逆向DP)的更多相关文章
- leetcode 174. 地下城游戏 解题报告
leetcode 174. 地下城游戏 一些恶魔抓住了公主(P)并将她关在了地下城的右下角.地下城是由 M x N 个房间组成的二维网格.我们英勇的骑士(K)最初被安置在左上角的房间里,他必须穿过地下 ...
- Java实现 LeetCode 174 地下城游戏
174. 地下城游戏 一些恶魔抓住了公主(P)并将她关在了地下城的右下角.地下城是由 M x N 个房间组成的二维网格.我们英勇的骑士(K)最初被安置在左上角的房间里,他必须穿过地下城并通过对抗恶魔来 ...
- Leetcode 174.地下城游戏
地下城游戏 一些恶魔抓住了公主(P)并将她关在了地下城的右下角.地下城是由 M x N 个房间组成的二维网格.我们英勇的骑士(K)最初被安置在左上角的房间里,他必须穿过地下城并通过对抗恶魔来拯救公主. ...
- P2016 战略游戏——树形DP大水题
P2016 战略游戏 树形DP 入门题吧(现在怎么是蓝色标签搞不懂): 注意是看见每一条边而不是每一个点(因为这里错了好几次): #include<cstdio> #include< ...
- [JLOI2013]卡牌游戏 概率DP
[JLOI2013]卡牌游戏 概率DP 题面 \(dfs\)复杂度爆炸,考虑DP.发现决策时,我们只用关心当前玩家是从庄家数第几个玩家与当前抽到的牌是啥.于是设计状态\(f[i][j]\)表示有\(i ...
- Lua游戏逆向及破解方法介绍
Lua游戏逆向及破解方法介绍 背景介绍 随着手游的发展,越来越多的Cocos-lua端游开发者转移到手游平台.Lua脚本编写逻辑的手游也是越来越多,如梦幻西游.刀塔传奇.开心消消乐.游龙英雄.奇迹 ...
- FPS游戏逆向-方框透视(三角函数)
本套课程主要学习FPS类游戏安全 由于FPS类游戏本身的特性问题,可能产生一些通用的游戏安全问题 在通过逆向与正向对FPS类游戏分析之后,找到其可能出现的不安全点 才能更好的保护游戏不被外部力量侵犯 ...
- 174 Dungeon Game 地下城游戏
一些恶魔抓住了公主(P)并将她关在了地下城的右下角.地下城是由 M x N 个房间组成的二维网格布局.我们英勇的骑士(K)最初被安置在左上角的房间里,并且必须通过地下城对抗来拯救公主.骑士具有以正整数 ...
- NOIP2003pj数字游戏[环形DP]
题目描述 丁丁最近沉迷于一个数字游戏之中.这个游戏看似简单,但丁丁在研究了许多天之后却发觉原来在简单的规则下想要赢得这个游戏并不那么容易.游戏是这样的,在你面前有一圈整数(一共n个),你要按顺序将其分 ...
随机推荐
- 正确安装Windows server 2012 r2的方法
正确安装Windows server 2012 r2的方法,请看下面的步骤 方法/步骤 1 准备好镜像文件,安装 2 输入密钥,下一步 选择下面(带有GUI的服务器).不要选上面的(服务器核心安装), ...
- php安装xdebug扩展,PHPStorm+XDebug单步调试
(一)php安装xdebug扩展,PHPStorm+XDebug单步调试 (二)PHPStorm配置XDebug (三)PHPStorm使用XDebug调试 (四)PhpStorm+Xdebug配置单 ...
- source insight增加tab标签页的方法之sihook
1.效果如下 2.方法见如下博客 http://www.cnblogs.com/Red_angelX/archive/2013/01/23/2873603.html
- ssh配置跳板机-带密钥
ssh配置跳板机堡垒机带密钥 ~/.ssh/config 添加以下配置: # 跳板机地址 Host jumper HostName jumper.com User jumper port 23333 ...
- ECMAScript基本语法——⑤运算符 赋值运算符
左边的变量等于等号左边的内容移到右边
- 第十周 11.28 psp0
课堂测试总结 学生:马小心 日期:2017.11.28 作业号 日期 过程 估计数据 实际数据 累计数据 时间 ...
- 洛谷P1308 统计单词数
原题链接:https://www.luogu.org/problem/P1308 #include<iostream> #include<cstring> #include&l ...
- 数据预处理 | 使用 Pandas 统一同一特征中不同的数据类型
出现的问题:如图,总消费金额本应该为float类型,此处却显示object 需求:将 TotalCharges 的类型转换成float 使用 pandas.to_numeric(arg, errors ...
- POJ1273【网络流】
Drainage Ditches Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 91824 Accepted ...
- Vue中常见参数传递方式
文章内容:这里只有vue中父子组件传参.路由间的传参 (另外还有vuex.储存本地.中央bus等方式) 一.父子组件 1.1父传子(props) <!-- 父组件father.vue --> ...