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个),你要按顺序将其分 ...
随机推荐
- 【sklearn】特征选择和降维
1.13 特征选择 sklearn.feature_selection模块中的类可以用于样本集上的特征选择/降维,以提高估计器的精度值,或提高其应用在高维数据集上的性能. 1.13.1 删除低方差的特 ...
- Linux中为什么执行自己的程序要在前面加./
前言 在Linux中,我们执行内置命令时,直接输入命令名称即可,如: $ mv a b #将a重命名为b 而在执行自己写好的程序时,却要带上./,例如: $ hello hello: comm ...
- PHP Help Guideds
how does php work with Apache? https://stillat.com/blog/2014/04/02/how-does-php-work-with-the-web-se ...
- Vue(三)--循环语句
v-for: v-for 指令需要以 site in sites 形式的特殊语法, sites 是源数据数组并且 site 是数组元素迭代的别名. demo1. <!DOCTYPE html&g ...
- XMind快捷键汇总
在 XMind: ZEN 中,快捷键是可以大大提高绘图效率的存在.掌握常用的快捷键组合,就可以在键盘上运指如飞,快速地进行思维导图的绘制.还在等什么?感兴趣的朋友,下面就和小编一起来看看吧! XMin ...
- Game with string CodeForces - 1104B
虽然只是B题,还是div2的 但感觉挺有意思,所以写一篇博客记录一下 最初的想法是利用DP去做,f[s]=true表示字符串s对应先手赢,否则对应后手赢,大致想了下发现是指数级别的算法,看了下范围直接 ...
- cookies欺骗-bugkuctf
解题思路: 打开链接是一串没有意义的字符串,查看源码没有发现什么,然后查看url,发现 filename的值是base64编码的,拿去解码 发现是一个文件,那么我们这里应该可以读取当前目录下的本地文件 ...
- PAT (Basic Level) Practice (中文)1030 完美数列 (25 分) (有点意思)
给定一个正整数数列,和正整数 p,设这个数列中的最大值是 M,最小值是 m,如果 M≤mp,则称这个数列是完美数列. 现在给定参数 p 和一些正整数,请你从中选择尽可能多的数构成一个完美数列. 输入格 ...
- A函数跨区域
ODerController.class.php内容如下 1 <?php /* * To change this license header, choose License Headers i ...
- hdu1005 矩阵快速幂
#include<iostream> #include<cstdio> #include<cstring> #include<set> #include ...