-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy path005 Longest Palindromic Substring.py
110 lines (91 loc) · 3.28 KB
/
005 Longest Palindromic Substring.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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
"""
Given a string S, find the longest palindromic substring in S. You may assume that the maximum length of S is 1000, and
there exists one unique longest palindromic substring.
Author: Rajeev Ranjan
"""
class Solution(object):
def longestPalindrome(self, s):
"""
O(n^2)
:param s: string
:return: string
"""
if not s:
return
n = len(s)
if n == 1:
return s
ret = s[0]
for i in xrange(0, n):
cur = self.get_palindrome_from_center(s, i, i) # odd length
if len(cur) > len(ret): ret = cur
cur = self.get_palindrome_from_center(s, i, i+1)
if len(cur) > len(ret): ret = cur
return ret
def longestPalindrome_TLE(self, s):
"""
Algorithm: dp, O(n^2)
p[i,j] represents weather s[i:j] is palindrome. (incl. i-th while excl. j-th)
For example S = "abccb"
01234
p[0,1] = True, p[1,2] = True, etc. since single char is Palindrom
p[0,2] = s[0]==s[1],
p[0,3] = s[0]==s[2] && p[1,2]
p[0,4] = s[0]==s[3] && p[1,3]
p[0,5] = s[0]==s[4] && p[1,4]
thus,
p[i,j] = 1 if i+1==j
p[i,j] = s[i]==s[j-1] if i+1==j-1 else
p[i,j] = s[i]==s[j-1] && p[i+1, j-1]
:param s: string
:return: string
"""
length = len(s)
dp = [[False for _ in xrange(length+1)] for _ in xrange(length+1)]
for i in xrange(length+1):
dp[i][i] = True
longest = [0, 0]
for j in xrange(length+1):
for i in xrange(j-1, -1, -1):
if i+1 == j:
dp[i][j] = True
else:
dp[i][j] = s[i] == s[j-1] and dp[i+1][j-1] # pre-access? starting backward
if dp[i][j] == True and longest[1]-longest[0] < j-i:
longest[0], longest[1] = i, j
return s[longest[0]:longest[1]]
def longestPalindrome_TLE2(self, s):
"""
:param s: string
:return: string
"""
length = len(s)
longest = ""
dp = [[False for _ in xrange(length+1)] for _ in xrange(length+1)] # larger than usual
for i in xrange(length+1):
dp[i][i] = True # empty
for i in xrange(length):
dp[i][i+1] = True # single char
for i in xrange(length-1):
dp[i][i+2] = s[i] == s[i+1]
if dp[i][i+1]:
longest = s[i:i+2]
for l in xrange(3, length+1): # breadth
for i in xrange(0, length-l):
if s[i] == s[i+l-1]:
dp[i][i+l] = dp[i+1][i+l-1]
else:
dp[i][i+l] = False
if dp[i][i+l] and len(longest) < l:
longest = s[i:i+l]
return longest
def get_palindrome_from_center(self, s, begin, end):
"""
# [begin, end]
"""
while begin >= 0 and end < len(s) and s[begin] == s[end]:
begin -= 1
end += 1
return s[begin+1: end-1+1]
if __name__ == "__main__":
assert Solution().longestPalindrome("dfaaabbbaaac") == "aaabbbaaa"