其他分享
首页 > 其他分享> > leetcode-174. Dungeon Game 地下城游戏

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)

源码

class Solution {
public:
    int calculateMinimumHP(vector<vector<int>>& dungeon) {
        int row = dungeon.size();
        int col = dungeon[0].size();
        if(row == 0 || col == 0)
            return 1;
        vector<vector<int>> dp(row+1, vector<int>(col+1, INT_MAX));
        dp[row][col-1] = 1; dp[row-1][col] = 1;
        for(int i=row-1; i>=0; i--)
        {
            for(int j=col-1; j>=0; j--)
            {
                int ph = min(dp[i+1][j], dp[i][j+1]) - dungeon[i][j];
                dp[i][j] = (ph > 0)?ph:1;
            }
        }
        return dp[0][0];
    }
};

标签:Dungeon,dungeon,int,Game,骑士,地下城,col,dp,row
来源: https://www.cnblogs.com/yocichen/p/11107025.html