-
Notifications
You must be signed in to change notification settings - Fork 29
/
Copy pathsufix_automation.py
75 lines (58 loc) · 2.41 KB
/
sufix_automation.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
class SuffixAutomatonNode:
def __init__(self):
self.next = {} # Transition to next states based on character
self.length = 0 # Length of the node's substring
self.link = -1 # Suffix link to another state
class SuffixAutomaton:
def __init__(self):
self.suffix_automaton = []
self.last = 0 # Index of the last state in the automaton
# Initialize the suffix automaton
def initialize(self):
initial_node = SuffixAutomatonNode()
self.suffix_automaton = [initial_node]
self.last = 0
# Extend the automaton with a new character
def extend_automaton(self, c):
new_node = SuffixAutomatonNode()
new_node.length = self.suffix_automaton[self.last].length + 1
current = self.last
while current != -1 and c not in self.suffix_automaton[current].next:
self.suffix_automaton[current].next = len(self.suffix_automaton) # Create a new state
current = self.suffix_automaton[current].link
if current == -1:
new_node.link = 0 # The root state
else:
next_state = self.suffix_automaton[current].next
if self.suffix_automaton[current].length + 1 == self.suffix_automaton[next_state].length:
new_node.link = next_state
else:
clone_node = SuffixAutomatonNode()
clone_node = self.suffix_automaton[next_state]
clone_node.length = self.suffix_automaton[current].length + 1
self.suffix_automaton.append(clone_node) # Clone the state
while current != -1 and self.suffix_automaton[current].next == next_state:
self.suffix_automaton[current].next = len(self.suffix_automaton) - 1
current = self.suffix_automaton[current].link
new_node.link = len(self.suffix_automaton) - 1
self.suffix_automaton[next_state].link = new_node.link
self.suffix_automaton.append(new_node)
self.last = len(self.suffix_automaton) - 1
# Traverse the suffix automaton
def traverse_automaton(self):
print("Traversing Suffix Automaton:")
for i, state in enumerate(self.suffix_automaton):
print(f"State {i}, Length: {state.length}, Suffix Link: {state.link}")
for char, next_state in state.next.items():
print(f" Transition on '{char}' to State {next_state}")
# Main function
def main():
input_str = "abab"
suffix_automaton_instance = SuffixAutomaton()
suffix_automaton_instance.initialize()
for char in input_str:
suffix_automaton_instance.extend_automaton(char)
# Traverse the constructed suffix automaton
suffix_automaton_instance.traverse_automaton()
if __name__ == "__main__":
main()