题目地址:https://leetcode.com/problems/knight-probability-in-chessboard/description/
题目描述:
Onan NxN
chessboard, a knight starts at the r
-th row and c
-th column and attempts to make exactly K moves. The rows and columns are 0 indexed, so the top-left square is (0, 0)
, and the bottom-right square is (N-1, N-1)
.
Achess knight has 8 possible moves it can make, as illustrated below. Each move is two squares in a cardinal direction, then one square in an orthogonal direction.
Each time the knight is to move, it chooses one of eight possible moves uniformly at random (even if the piece would go off the chessboard) and moves there.
Theknight continues moving until it has made exactly K moves or has moved off the chessboard. Return the probability that the knight remains on the board after it has stopped moving.
Example:
Input: 3, 2, 0, 0
Output: 0.0625
Explanation: There are two moves (to (1,2), (2,1)) that will keep the knight on the board.
From each of those positions, there are also two moves that will keep the knight on the board.
The total probability the knight stays on the board is 0.0625.
Note:
1、 N
willbebetween1and25.;
2、 K
willbebetween0and100.;
3、 Theknightalwaysinitiallystartsontheboard.;
题目大意
有个N* N的棋盘,上面有个马,马走日字象走田嘛,找出这个马走了K步之后依然在这个棋盘上的概率。
解题方法
如果dfs的话一定会超时的,所以还是得用dp来解。
这个dp数组代表在某一轮中,这个马能到这个位置的次数。
dp更新的策略是,我们遍历棋盘的每个位置,当前的数值是能走到这个位置的在上一轮dp的数值 + 1。
这个题的对称性让这个题变得简单而又有趣。最内层的for循环对dp进行更新的时候是不用考虑索引位置的,因为对称性太美了。
时间复杂度是O(K * N ^ 2),空间复杂度是O(N ^ 2)。
注意,python2里面的/默认的是地板除,需要用float再除得到概率。
代码如下:
class Solution(object):
def knightProbability(self, N, K, r, c):
"""
:type N: int
:type K: int
:type r: int
:type c: int
:rtype: float
"""
dp = [[0 for i in range(N)] for j in range(N)]
dp[r][c] = 1
directions = [(1, 2), (1, -2), (2, 1), (2, -1), (-2, 1), (-2, -1), (-1, 2), (-1, -2)]
for k in range(K):
new_dp = [[0 for i in range(N)] for j in range(N)]
for i in range(N):
for j in range(N):
for d in directions:
x, y = i + d[0], j + d[1]
if x < 0 or x >= N or y < 0 or y >= N:
continue
new_dp[i][j] += dp[x][y]
dp = new_dp
return sum(map(sum, dp)) / float(8 ** K)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
参考资料:
https://www.youtube.com/watch?v=MyJvMydR2G4
DDKK.COM 弟弟快看-教程,程序员编程资料站,版权归原作者所有
本文经作者:负雪明烛 授权发布,任何组织或个人未经作者授权不得转发