题目信息来源
作者:Krahets
链接:https://leetcode.cn/leetbook/read/illustration-of-algorithm
来源:力扣(LeetCode)
地上有一个m行n列的方格,从坐标 [0,0] 到坐标 [m-1,n-1] 。一个机器人从坐标 [0, 0] 的格子开始移动,它每次可以向左、右、上、下移动一格(不能移动到方格外),也不能进入行坐标和列坐标的数位之和大于k的格子。例如,当k为18时,机器人能够进入方格 [35, 37] ,因为3+5+3+7=18。但它不能进入方格 [35, 38],因为3+5+3+8=19。请问该机器人能够到达多少个格子?
示例 1:
输入:m = 2, n = 3, k = 1
输出:3
示例 2:
输入:m = 3, n = 1, k = 0
输出:1
提示:
1 <= n,m <= 100
0 <= k <= 20
昨天做了一题,总之能照猫画虎做出来一题
递归参数:行坐标i, 列坐标j
终止条件:超出边界,不符合规则,已经看过的地方。用了一个二维列表来标记,还踩了雷。
递推工作:标记已经选过的列表,搜索下一个单元格,计数
返回值:无。选择了一个全局变量来计数,说实话,我没太想好怎么在递归中计数,这个尝试失败了
相关知识点
行坐标和列坐标的数位之和计算,设行坐标为i,列坐标为j,则计算行坐标数位之和为i//10+i%10,即整除和取余。我这里可能钻了测试题的漏洞,如果出现m,n=100,可能就bug了。
class Solution:def movingCount(self, m: int, n: int, k: int) -> int:# 递归参数:行坐标i, 列坐标j# 终止条件:超出边界,不符合规则# 递推工作:标记已经选过的列表,搜索下一个单元格,计数# 返回值self.m, self.n, self.k = m, n, kself.result = 0self.myset = [[0 for x in range(n)] for x in range(m)]self.move(0, 0)return self.resultdef move(self, i, j):if i>=self.m or j>=self.n:return#if i<0 or i>=self.m or j<0 or j>=self.n:returnif self.myset[i][j]==1:returnif self.myset[i][j]==0:self.myset[i][j] = 1if (i//10+i%10+j//10+j%10)>self.k:returnelse:self.result = self.result + 1# self.move(i-1, j)self.move(i+1, j)# self.move(i, j-1)self.move(i, j+1)return
官方题解知识点重点
x和x+1不超过两位数set()来标记走过的点,来剪枝,但是失败了,因为我的语法是set.add([i,j]),看了解析用的是set.add((i,j))+,如果不符合条件的直接会返回 0,所以会没有变化,其他情况都是被1+but,官方解析里
si,sj的写法很容易混淆,其实就是 x 和 x+1 的数位之和
广度优先遍历 BFS
以平推的方式向前搜索,通常会利用队列实现。
尝试复现官方题解的广度优先遍历
class Solution:def movingCount(self, m: int, n: int, k: int) -> int:# 初始化需要队列遍历存储遍历的列表,需要一个列表存储已经标记过的格子myqueue = [(0, 0, 0, 0)]choosed = set()while myqueue:x, y, sum_x, sum_y = myqueue.pop(0)if x>=m or y>=n or sum_x+sum_y>k or (x,y) in choosed:continuechoosed.add((x, y))myqueue.append((x+1, y, sum_x+1 if (x+1)%10!=0 else sum_x-8, sum_y))myqueue.append((x, y+1, sum_x, sum_y+1 if (y+1)%10!=0 else sum_y-8))return len(choosed)
上一篇:汇丰银行主席王冬胜建言:加强顶层设计并加大政府资金和政策对科技创新的支持 汇丰银行主席王冬胜建言:加强顶层设计并加大政府资金和政策对科技创新的支持
下一篇:中央新规定,“房龄超过20年”的老小区,统统这样处理! 国家对房龄较老的小区如何处理 房龄超过20年的小区迎来新规处理