-
Notifications
You must be signed in to change notification settings - Fork 1.6k
/
Copy pathnumber-of-ways-to-reach-destination-in-the-grid.py
71 lines (66 loc) · 2.16 KB
/
number-of-ways-to-reach-destination-in-the-grid.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
# Time: O(logn)
# Space: O(1)
# dp, matrix exponentiation
class Solution(object):
def numberOfWays(self, n, m, k, source, dest):
"""
:type n: int
:type m: int
:type k: int
:type source: List[int]
:type dest: List[int]
:rtype: int
"""
MOD = 10**9+7
def matrix_mult(A, B):
ZB = zip(*B)
return [[sum(a*b % MOD for a, b in itertools.izip(row, col)) % MOD for col in ZB] for row in A]
def matrix_expo(A, K):
result = [[int(i == j) for j in xrange(len(A))] for i in xrange(len(A))]
while K:
if K % 2:
result = matrix_mult(result, A)
A = matrix_mult(A, A)
K /= 2
return result
T = [[0, m-1, n-1, 0],
[1, m-2, 0, n-1],
[1, 0, n-2, m-1],
[0, 1, 1, (n-2)+(m-2)]]
dp = [0]*4 # dp[0] = both_same, dp[1] = row_same, dp[2] = col_same, dp[3] = no_same
if source == dest:
dp[0] = 1
elif source[0] == dest[0]:
dp[1] = 1
elif source[1] == dest[1]:
dp[2] = 1
else:
dp[3] = 1
dp = matrix_mult([dp], matrix_expo(T, k))[0] # [dp[0], dp[1], dp[2], dp[3]] * T^k
return dp[0]
# Time: O(n)
# Space: O(1)
# dp
class Solution2(object):
def numberOfWays(self, n, m, k, source, dest):
"""
:type n: int
:type m: int
:type k: int
:type source: List[int]
:type dest: List[int]
:rtype: int
"""
MOD = 10**9+7
both_same = row_same = col_same = no_same = 0
if source == dest:
both_same = 1
elif source[0] == dest[0]:
row_same = 1
elif source[1] == dest[1]:
col_same = 1
else:
no_same = 1
for _ in xrange(k):
both_same, row_same, col_same, no_same = (row_same+col_same)%MOD, (both_same*(m-1)+row_same*(m-2)+no_same)%MOD, (both_same*(n-1)+col_same*(n-2)+no_same)%MOD, (row_same*(n-1)+col_same*(m-1)+no_same*((n-2)+(m-2)))%MOD
return both_same