-
Notifications
You must be signed in to change notification settings - Fork 1.6k
/
Copy pathminimum-changes-to-make-k-semi-palindromes.py
89 lines (83 loc) · 3.71 KB
/
minimum-changes-to-make-k-semi-palindromes.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
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
# Time: O(n * nlogn + n^3 + n^2 * k) = O(n^3)
# Space: O(n * nlogn) = O(n^2 * logn)
# number theory, dp
class Solution(object):
def minimumChanges(self, s, k):
"""
:type s: str
:type k: int
:rtype: int
"""
divisors = [[] for _ in xrange(len(s)+1)]
for i in xrange(1, len(divisors)): # Time: O(nlogn), Space: O(nlogn)
for j in xrange(i, len(divisors), i):
divisors[j].append(i)
dp = [[{} for _ in xrange(len(s))] for _ in xrange(len(s))]
for l in xrange(1, len(s)+1): # Time: O(n * nlogn + n^3), Space: O(n * nlogn)
for left in xrange(len(s)-l+1):
right = left+l-1
for d in divisors[l]:
dp[left][right][d] = (dp[left+d][right-d][d] if left+d < right-d else 0)+sum(s[left+i] != s[(right-(d-1))+i] for i in xrange(d))
dp2 = [[min(dp[i][j][d] for d in divisors[j-i+1] if d != j-i+1) if i < j else 0 for j in xrange(len(s))] for i in xrange(len(s))] # Time: O(n^2), Space: O(n^2)
dp3 = [len(s)]*(len(s)+1)
dp3[0] = 0
for l in xrange(k): # Time: O(k * n^2), Space: O(n)
new_dp3 = [len(s)]*(len(s)+1)
for i in xrange(len(s)):
for j in xrange(l*2, i): # optimized for the fact that the length of semi-palindrome is at least 2
new_dp3[i+1]= min(new_dp3[i+1], dp3[j]+dp2[j][i])
dp3 = new_dp3
return dp3[len(s)]
# Time: O(n * nlogn + n^3 + n^2 * k) = O(n^3)
# Space: O(n * nlogn) = O(n^2 * logn)
# number theory, dp
class Solution2(object):
def minimumChanges(self, s, k):
"""
:type s: str
:type k: int
:rtype: int
"""
divisors = [[] for _ in xrange(len(s)+1)]
for i in xrange(1, len(divisors)): # Time: O(nlogn), Space: O(nlogn)
for j in xrange(i, len(divisors), i):
divisors[j].append(i)
dp = [[{} for _ in xrange(len(s))] for _ in xrange(len(s))]
for l in xrange(1, len(s)+1): # Time: O(n * nlogn + n^3), Space: O(n * nlogn)
for left in xrange(len(s)-l+1):
right = left+l-1
for d in divisors[l]:
dp[left][right][d] = (dp[left+d][right-d][d] if left+d < right-d else 0)+sum(s[left+i] != s[(right-(d-1))+i] for i in xrange(d))
dp2 = [[len(s)]*(k+1) for _ in xrange(len(s)+1)]
dp2[0][0] = 0
for i in xrange(len(s)): # Time: O(n^2 * logn + n^2 * k), Space: O(n * k)
for j in xrange(i):
c = min(dp[j][i][d] for d in divisors[i-j+1] if d != i-j+1)
for l in xrange(k):
dp2[i+1][l+1] = min(dp2[i+1][l+1], dp2[j][l]+c)
return dp2[len(s)][k]
# Time: O(n^2 * nlogn + n^2 * k) = O(n^3 * logn)
# Space: O(nlogn + n * k)
# number theory, dp
class Solution3(object):
def minimumChanges(self, s, k):
"""
:type s: str
:type k: int
:rtype: int
"""
def min_dist(left, right): # Time: O(nlogn)
return min(sum(s[left+i] != s[right-((i//d+1)*d-1)+(i%d)] for i in xrange((right-left+1)//2))
for d in divisors[right-left+1])
divisors = [[] for _ in xrange(len(s)+1)]
for i in xrange(1, len(divisors)): # Time: O(nlogn), Space: O(nlogn)
for j in xrange(i+i, len(divisors), i):
divisors[j].append(i)
dp = [[len(s)]*(k+1) for _ in xrange(len(s)+1)]
dp[0][0] = 0
for i in xrange(len(s)): # Time: O(n^2 * nlogn + n^2 * k), Space: O(n * k)
for j in xrange(i):
c = min_dist(j, i)
for l in xrange(k):
dp[i+1][l+1] = min(dp[i+1][l+1], dp[j][l]+c)
return dp[len(s)][k]