​LeetCode刷题实战174:地下城游戏

算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试。所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 !

今天和大家聊的问题叫做 地下城游戏  ,我们先来看题面:

https://leetcode-cn.com/problems/dungeon-game/

Write a function to determine the knight's minimum initial health so that he is able to rescue the princess.

For example, given the dungeon below, the initial health of the knight must be at least 7 if he follows the optimal path RIGHT-> RIGHT -> DOWN -> DOWN.

题意

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

骑士的初始健康点数为一个正整数。如果他的健康点数在某一时刻降至 0 或以下,他会立即死亡。

有些房间由恶魔守卫,因此骑士在进入这些房间时会失去健康点数(若房间里的值为负整数,则表示骑士将损失健康点数);其他房间要么是空的(房间里的值为 0),要么包含增加骑士健康点数的魔法球(若房间里的值为正整数,则表示骑士将增加健康点数)。

为了尽快到达公主,骑士决定每次只向右或向下移动一步。

解题

https://blog.csdn.net/qq_17550379/article/details/86243097

一看到这个问题首先想到的就是动态规划,其次想的就是自顶向下,但是这样想的话就会有一个问题,就是我们需要判断每次到达一个房间后,我们的血量不能是负数,这给我们编写代码造成了很大的困难。如果采用自底向上的思路去做的话,那么问题就会变得简单很多(而且我觉得你应该做到,拿到这个问题首先想到的就是自底向上,这是一种直觉)。我们只需要取右和下的最小值(我们需要计算最少的能量),然后减去dungoen[i][j],并且保证到达当前房间后的血量大于等于0就行了

f(i,j)=max(0,min(f(i+1,j),f(i,j+1))−dungeon[i][j])

对于边界问题我们单独处理。最后我们只需要取结果f(0,0)+1即可(根据题目中的例子)。Python代码如下:

class Solution:
    def calculateMinimumHP(self, dungeon):
        """
        :type dungeon: List[List[int]]
        :rtype: int
        """
        row, col = len(dungeon), len(dungeon[0])
        mem = [[0]*col for _ in range(row)]

        for i in range(row-1,-1,-1):
            for j in range(col-1,-1,-1):
                if i == row-1 and j == col-1:
                    mem[i][j] = max(0, -dungeon[i][j])
                elif i == row-1:
                    mem[i][j] = max(0, mem[i][j+1] - dungeon[i][j])
                elif j == col-1:
                    mem[i][j] = max(0, mem[i+1][j] - dungeon[i][j])
                else:
                    mem[i][j] = max(0, min(mem[i+1][j], mem[i][j+1]) - dungeon[i][j])
                
        return mem[0][0] + 1

好了,今天的文章就到这里,如果觉得有所收获,请顺手点个在看或者转发吧,你们的支持是我最大的动力 。

上期推文:

LeetCode1-160题汇总,希望对你有点帮助!

LeetCode刷题实战161:相隔为1的编辑距离

LeetCode刷题实战162:寻找峰值

LeetCode刷题实战163:缺失的区间

LeetCode刷题实战164:最大间距

LeetCode刷题实战165:比较版本号

LeetCode刷题实战166:分数到小数

LeetCode刷题实战167:两数之和 II - 输入有序数组

LeetCode刷题实战168:Excel表列名称

LeetCode刷题实战169:多数元素

LeetCode刷题实战170:两数之和 III - 数据结构设计

LeetCode刷题实战171:Excel表列序号

LeetCode刷题实战172:阶乘后的零

LeetCode刷题实战173:二叉搜索树迭代器

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值