leetcode-174. Dungeon Game 地下城游戏
一道关于骑士救公主故事的题目。
一些恶魔抓住了公主(P)并将她关在了地下城的右下角。地下城是由 M x N 个房间组成的二维网格。我们英勇的骑士(K)最初被安置在左上角的房间里,他必须穿过地下城并通过对抗恶魔来拯救公主。
骑士的初始健康点数为一个正整数。如果他的健康点数在某一时刻降至 0 或以下,他会立即死亡。
有些房间由恶魔守卫,因此骑士在进入这些房间时会失去健康点数(若房间里的值为负整数,则表示骑士将损失健康点数);其他房间要么是空的(房间里的值为 0),要么包含增加骑士健康点数的魔法球(若房间里的值为正整数,则表示骑士将增加健康点数)。
为了尽快到达公主,骑士决定每次只向右或向下移动一步。
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/dungeon-game
分析
这与之前的那道 64.最短路径和 颇为相似,不同的是,最短路径和是从左上角开始,取右&下最小一直算到右下角。而本题如果如法炮制(左上->右下)的话,并不能得到最初的最小健康点。
要算开始点的最小健康点,应该从右下->左上求值。对于这样的最优路径题目,我们一贯采用DP来解。
算法
1. 首先初始化一个二维数组DP[M+1][ N+1]值都是INT_MAX(额外的一行一列是为了确定DP中最后一行和最后一列使用的,当然你也可以不用额外的行列,首先算出右下角的数值,单独算最后一行和最后一列);
2. 从DP右下角[M][N]开始,一直算到左上角
状态方程:
ans = min(dp[i+1][j], dp[i][j+1]) - dungeon[i][j];
dp[i][j] = ans; (ans >0)
dp[i][j] = 1; (else)
3. 输出dp[0][0]
解释:我们根据当前点的右边和下边点来确定当前点。由于要求最小的生命值/健康点,所以我们取二者中的最小值(可以认为下一步的生命值越小,本点的生命值就越小)减去当前点(i, j)的损耗值,就是本点的最小生命值。(在简单点儿说就是,上一点的初始PH+损耗PH(有正有负) = 下一点初始PH,我们算的都是初始PH值,所以用下一点的初始PH-损耗=上一点的初始PH)
源码1 二维DP
class Solution {
public:
int calculateMinimumHP(vector<vector<int>>& dungeon) {
int row = dungeon.size();
int col = dungeon[].size();
if(row == || col == )
return ;
vector<vector<int>> dp(row+, vector<int>(col+, INT_MAX));
dp[row][col-] = ; dp[row-][col] = ;
for(int i=row-; i>=; i--)
{
for(int j=col-; j>=; j--)
{
int ph = min(dp[i+][j], dp[i][j+]) - dungeon[i][j];
dp[i][j] = (ph > )?ph:;
}
}
return dp[][];
}
};
源码2 原地操作,不使用额外空间
class Solution {
public:
int calculateMinimumHP(vector<vector<int>>& dungeon) {
int row = dungeon.size();
int col = dungeon[].size();
if(row == || col == )
return ;
//右下角点PH值
dungeon[row-][col-] = sub(, dungeon[row-][col-]);
//最后一行/一列单独计算
for(int i=row-; i>=; i--)
dungeon[i][col-] = sub(dungeon[i+][col-], dungeon[i][col-]);
for(int i=col-; i>=; i--)
dungeon[row-][i] = sub(dungeon[row-][i+], dungeon[row-][i]);
//其他点计算
for(int i=row-; i>=; i--)
{
for(int j=col-; j>=; j--)
{
dungeon[i][j] = sub(min(dungeon[i+][j], dungeon[i][j+]), dungeon[i][j]);
}
}
return dungeon[][];
}
//用于计算当前点的初始PH,参数为min(右边,下边),当前点损耗值
int sub(int ph, int sub)
{
int PH = ph - sub;
return (PH > ) ? PH : ;
}
};
源码3 使用一维DP数组
自己尝试吧!没有必要!
leetcode-174. Dungeon Game 地下城游戏的更多相关文章
- 174 Dungeon Game 地下城游戏
一些恶魔抓住了公主(P)并将她关在了地下城的右下角.地下城是由 M x N 个房间组成的二维网格布局.我们英勇的骑士(K)最初被安置在左上角的房间里,并且必须通过地下城对抗来拯救公主.骑士具有以正整数 ...
- [LeetCode] 174. Dungeon Game 地牢游戏
The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dungeon. ...
- ✡ leetcode 174. Dungeon Game 地牢游戏 --------- java
The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dungeon. ...
- [leetcode]174. Dungeon Game地牢游戏
The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dungeon. ...
- leetcode@ [174] Dungeon Game (Dynamic Programming)
https://leetcode.com/problems/dungeon-game/ The demons had captured the princess (P) and imprisoned ...
- Java for LeetCode 174 Dungeon Game
The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dungeon. ...
- Leetcode#174 Dungeon Game
原题地址 典型的地图寻路问题 如何计算当前位置最少需要多少体力呢?无非就是在向下走或向右走两个方案里做出选择罢了. 如果向下走,看看当前位置能提供多少体力(如果是恶魔就是负数,如果是草药就是正数),如 ...
- LeetCode 174. Dungeon Game (C++)
题目: The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dung ...
- leetcode 174. 地下城游戏 解题报告
leetcode 174. 地下城游戏 一些恶魔抓住了公主(P)并将她关在了地下城的右下角.地下城是由 M x N 个房间组成的二维网格.我们英勇的骑士(K)最初被安置在左上角的房间里,他必须穿过地下 ...
随机推荐
- spring bean容器加载后执行初始化处理@PostConstruct
先说业务场景,我在系统启动后想要维护一个List常驻内存,因为我可能经常需要查询它,但它很少更新,而且数据量不大,明显符合缓存的特质,但我又不像引入第三方缓存.现在的问题是,该List的内容是从数据库 ...
- 学习TypeScript 笔记
TypeScript 什么是TypeScript TypeScript 是 JavaScript 的一个超集,支持 ECMAScript 6 标准. TypeScript 由微软开发的自由和开源的编程 ...
- UI界面测试
概念:指测试用户界面的风格是否满足用户要求.文字是否正确.页面是否美观.文字与图片组合是否完美.操作是否友好等. 1.窗体测试 { 窗体大小. 移动窗体. 缩放窗体. 显示分辨率. 状态栏. 工具栏. ...
- 解决kalilinux:“下列签名无效: KEYEXPIRED 1425567400"
Kali linux由于太长时间未更新,而出现GPG错误 KEYEXPIRED 1425567400.经检查源未出现问题可以解析,deb也不冲突,就是密钥过期了. 解决方式 使用一条命令,添加新的密钥 ...
- Python(1)自动发送邮件
python发邮件需要掌握两个模块的用法,smtplib和email,这俩模块是python自带的,只需import即可使用.smtplib模块主要负责发送邮件,email模块主要负责构造邮件. sm ...
- CF C.Ivan the Fool and the Probability Theory【思维·构造】
题目传送门 题目大意: 一个$n*m$的网格图,每个格子可以染黑色.白色,问每个格子最多有一个相邻格子颜色相同的涂色方案数$n,m<=1e5$ 分析: 首先,考虑到如果有两个相邻的格子颜色相同, ...
- Can't connect to local MySQL server through socket '/var/lib/mysql/mysql.sock' (2)
mysql使用可视化界面登录使用的时候都没问题,只要使用xhell命令进入mysql就报异常 Can't connect to local MySQL server through socket '/ ...
- 事务的ACID
事务提供一种机制将一个活动涉及的所有操作纳入到一个不可分割的执行单元,组成事务的所有操作只有在所有操作均能正常执行的情况下方能提交,只要其中任一操作执行失败,都将导致整个事务的回滚. 简单地说,事务提 ...
- js文本公告滚动展示,图片轮播....
1.引入文件 <link rel="stylesheet" href="/css/liMarquee.css"> <script src=&q ...
- mysql常用操作及常见问题
常用操作 mysql备份: --整库备份 docker exec 容器ID mysqldump -uroot -p密码 --databases 库名 > 库名.sql --仅导出表和数据 mys ...