一道关于骑士救公主故事的题目。

一些恶魔抓住了公主(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 地下城游戏的更多相关文章

  1. 174 Dungeon Game 地下城游戏

    一些恶魔抓住了公主(P)并将她关在了地下城的右下角.地下城是由 M x N 个房间组成的二维网格布局.我们英勇的骑士(K)最初被安置在左上角的房间里,并且必须通过地下城对抗来拯救公主.骑士具有以正整数 ...

  2. [LeetCode] 174. Dungeon Game 地牢游戏

    The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dungeon. ...

  3. ✡ leetcode 174. Dungeon Game 地牢游戏 --------- java

    The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dungeon. ...

  4. [leetcode]174. Dungeon Game地牢游戏

    The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dungeon. ...

  5. leetcode@ [174] Dungeon Game (Dynamic Programming)

    https://leetcode.com/problems/dungeon-game/ The demons had captured the princess (P) and imprisoned ...

  6. Java for LeetCode 174 Dungeon Game

    The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dungeon. ...

  7. Leetcode#174 Dungeon Game

    原题地址 典型的地图寻路问题 如何计算当前位置最少需要多少体力呢?无非就是在向下走或向右走两个方案里做出选择罢了. 如果向下走,看看当前位置能提供多少体力(如果是恶魔就是负数,如果是草药就是正数),如 ...

  8. LeetCode 174. Dungeon Game (C++)

    题目: The demons had captured the princess (P) and imprisoned her in the bottom-right corner of a dung ...

  9. leetcode 174. 地下城游戏 解题报告

    leetcode 174. 地下城游戏 一些恶魔抓住了公主(P)并将她关在了地下城的右下角.地下城是由 M x N 个房间组成的二维网格.我们英勇的骑士(K)最初被安置在左上角的房间里,他必须穿过地下 ...

随机推荐

  1. React之简介

    官网链接React 用于构建用户界面的 JavaScript 库 特色 声明式: React 使创建交互式 UI 变得轻而易举.为你应用的每一个状态设计简洁的视图,当数据改变时 React 能有效地更 ...

  2. ES6深入浅出-2 新版函数:箭头函数 2 视频-1.视频 箭头函数

    ES3里面的写法 匿名函数是用不了的 还是需要给他赋值 选中的这一部分叫做匿名函数 等于号叫做赋值 let xxx叫做声明 所以说这个地方是三个语法,首先声明一个匿名函数,然后声明一个xxx变量,最后 ...

  3. java生成RSA公私钥字符串,简单易懂

    java生成RSA公私钥字符串,简单易懂   解决方法: 1.下载bcprov-jdk16-140.jar包,参考:http://www.yayihouse.com/yayishuwu/chapter ...

  4. Elasticsearch 7.x文档基本操作(CRUD)

    官方文档:https://www.elastic.co/guide/en/elasticsearch/reference/current/docs.html 1.添加文档 1.1.指定文档ID PUT ...

  5. Laravel 别名Redis 与 Redis 扩展冲突

    use Redis; //通过别名引用会报错 今天尝试使用了 Laravel 的 redis 结果报了如下错误. Non-static method Redis::xxx() cannot be ca ...

  6. 第十七章 OAuth2集成——《跟我学Shiro》

    目录贴:跟我学Shiro目录贴 目前很多开放平台如新浪微博开放平台都在使用提供开放API接口供开发者使用,随之带来了第三方应用要到开放平台进行授权的问题,OAuth就是干这个的,OAuth2是OAut ...

  7. D-Link系列路由器漏洞挖掘

    参考 http://www.freebuf.com/articles/terminal/153176.html https://paper.seebug.org/429/ http://www.s3c ...

  8. WebStorm 2018.2安装激活教程

    1.下载解压,得到jetbrains webstorm 2018.2主程序,破解文件和中文语言包: 2.运行“WebStorm-2018.2.exe”开始安装,默认安装目录[C:\Program Fi ...

  9. udevdm命令详解

    udevadm 后接一个命令和命令指定选项.它控制了udev运行的行为,处理内核事件,控制事件队列,并且提供简单的调试机制. 选项: --debug 打印错误信息 --version 打印版本信息 - ...

  10. python_网络编程_基础

    基本的架构有C/S架构 和B/S架构 B/S架构优于C/S架构? 因为统一入口 , 都是从浏览器开始访问 两台电脑实现通信, 需要网卡, 网卡上有全球唯一的mac地址 ARP协议 #通过ip地址就能找 ...