-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfinite-automaton.js
96 lines (80 loc) · 2.11 KB
/
finite-automaton.js
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
class FiniteAutomaton {
#transitionFunction
#startState
#acceptStates
constructor({ transitions, startState, acceptStates }) {
this.#transitionFunction = new TransitionFunction(transitions)
this.#startState = startState
this.#acceptStates = [...acceptStates]
}
isAccepting(string) {
const { isAccepting } = this.process(string)
return isAccepting
}
process(string) {
const stateSequence = [this.#startState]
let state = this.#startState
for (const symbol of string) {
state = this.#transitionFunction.get(state, symbol)
stateSequence.push(state)
}
const isAccepting = this.#isAcceptingState(state)
return { isAccepting, stateSequence }
}
#isAcceptingState(state) {
return this.#acceptStates.includes(state)
}
states() {
return this.#transitionFunction.states()
}
alphabet() {
return this.#transitionFunction.alphabet()
}
transitions() {
return this.#transitionFunction.transitions()
}
startState() {
return this.#startState
}
acceptStates() {
return [...this.#acceptStates]
}
}
class TransitionFunction {
#map
constructor(transitions) {
this.#map = this.#createTransitionMap(transitions)
}
#createTransitionMap(transitions) {
const map = new Map()
for (const { state, symbol, resultState } of transitions) {
let singleStateMap = map.get(state)
if (!singleStateMap) {
map.set(state, (singleStateMap = new Map()))
}
singleStateMap.set(symbol, resultState)
}
return map
}
get(state, symbol) {
const singleStateMap = this.#map.get(state)
return singleStateMap.get(symbol)
}
states() {
return Array.from(this.#map.keys())
}
alphabet() {
const arbitraryState = this.states()[0]
const singleStateMap = this.#map.get(arbitraryState)
return Array.from(singleStateMap.keys())
}
transitions() {
const transitions = []
for (const [state, singleStateMap] of this.#map) {
for (const [symbol, resultState] of singleStateMap) {
transitions.push({ state, symbol, resultState })
}
}
return transitions
}
}