Минимизация ДКА, алгоритм Хопкрофта (сложность O(n log n)) — различия между версиями
| Строка 33: | Строка 33: | ||
<tex>W</tex>.push(<tex>R_1</tex>) | <tex>W</tex>.push(<tex>R_1</tex>) | ||
<tex>W</tex>.push(<tex>R_2</tex>) | <tex>W</tex>.push(<tex>R_2</tex>) | ||
| − | Когда очередь станет пустой будет получено разбиение на классы эквивалентности, так как больше ни один класс | + | Когда очередь станет пустой будет получено разбиение на классы эквивалентности, так как больше ни один класс невозможно разбить. |
= Алгоритм Хопкрофта= | = Алгоритм Хопкрофта= | ||
| − | + | {{Лемма | |
| + | |statement = Класс <tex>R = R_1 \cup R_2</tex> и <tex>R_1 \cap R_2 = \varnothing</tex>, тогда разбиение всех классов (текущее разбиение) по символу <tex>a</tex> любыми двумя классами из <tex>R, R_1, R_2</tex> эквивалентно разбиению всех классов с помощью <tex>R, R_1, R_2</tex> по символу <tex>a</tex>. | ||
| + | |proof = | ||
| + | Разобьем все классы с помощью <tex>R </tex> и <tex> R_1</tex> по символу <tex>a</tex>, тогда для любого класса <tex>B</tex> из текущего разбиения выполняется | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \in R</tex> and <tex> \delta(r, a) \in R_1</tex> or | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \in R</tex> and <tex> \delta(r, a) \notin R_1</tex> or | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \notin R</tex> and <tex> \delta(r, a) \notin R_1</tex> | ||
| + | А так как <tex>R = R_1 \cup R_2</tex> и <tex>R_1 \cap R_2 = \varnothing</tex> то выполняется | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \in R_2</tex> | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \notin R_2</tex> | ||
| + | Из этого следует, что разбиение всех классов с помощью <tex>R_2</tex> никак не повлияет на текущее разбиение. <br/> | ||
| + | Аналогично доказывается и для разбиения с помощью <tex>R </tex> и <tex> R_2</tex> по символу <tex>a</tex>. <br/> | ||
| + | Разобьем все классы с помощью <tex>R_1</tex> и <tex> R_2</tex> по символу <tex>a</tex>, тогда для любого класса <tex>B</tex> из текущего разбиения выполняется | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \in R_1</tex> and <tex> \delta(r, a) \notin R_2</tex> or | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \notin R_1</tex> and <tex> \delta(r, a) \in R_2</tex> or | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \notin R_1</tex> and <tex> \delta(r, a) \notin R_2</tex> | ||
| + | А так как <tex>R = R_1 \cup R_2</tex> и <tex>R_1 \cap R_2 = \varnothing</tex> то выполняется | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \in R</tex> | ||
| + | :<tex>\forall r \in B \,\,\, \delta(r, a) \notin R</tex> | ||
| + | Из этого следует, что разбиение всех классов с помощью <tex>R</tex> никак не повлияет на текущее разбиение. | ||
| + | }} | ||
Алгорит Хопкрофта является улучшением простого алгоритма. | Алгорит Хопкрофта является улучшением простого алгоритма. | ||
Версия 22:49, 14 февраля 2012
Содержание
Постановка задачи
Пусть дан автомат, распознающий определенный язык. Требуется найти эквивалентный автомат с наименьшим количеством состояний.
Минимизация ДКА
Если в ДКА существуют два эквивалентных состояния, то при их объединении мы получим эквивалентный ДКА, так как распознаваемый язык не изменится. Основная идея минимизации состоит в разбиении множества состояний на классы эквивалентности, полученные классы и будут состояниями минимизированного ДКА.
Простой алгоритм
| Определение: |
| Класс разбивает класс по символу на и , если
|
Если класс может быть разбит по символу , то он содержит хотя бы одну пару неэквивалентных состояний (их можно различить любой строкой начинающейся с символа ). Если класс нельзя разбить, то он состоит из эквивалентных состояний. Поэтому самый простой алгоритм состоит в том, чтобы разбивать классы текущего разбиения до тех пор пока это возможно.
Псевдокод
— множество состояний ДКА. — множество терминальных состояний. — очередь. — разбиение множества состояний ДКА. — класс состояний ДКА.
while not .isEmpty() .pop() for all for all in if and replace in with and .push() .push()
Когда очередь станет пустой будет получено разбиение на классы эквивалентности, так как больше ни один класс невозможно разбить.
Алгоритм Хопкрофта
| Лемма: |
Класс и , тогда разбиение всех классов (текущее разбиение) по символу любыми двумя классами из эквивалентно разбиению всех классов с помощью по символу . |
| Доказательство: |
|
Разобьем все классы с помощью и по символу , тогда для любого класса из текущего разбиения выполняется
А так как и то выполняется Из этого следует, что разбиение всех классов с помощью никак не повлияет на текущее разбиение.
А так как и то выполняется |
Алгорит Хопкрофта является улучшением простого алгоритма.
Итеративно строим разбиение множества состояний следующим образом.
- Первоначальное разбиение множества состояний — класс допускающих состояний и класс недопускающих состояний.
- Меньший из них помещается в очередь.
- Из очереди извлекается класс, далее именуемый как сплиттер.
- Перебираются все символы из алфавита , где — текущий символ.
- Все классы текущего разбиения разбиваются на 2 подкласса (один из которых может быть пустым). Первый состоит из состояний, которые по символу переходят в сплиттер, а второй из всех оставшихся.
- Те классы, которые разбились на два непустых подкласса, заменяются этими подклассами в разбиении, а также меньший из двух подклассов добавляется в очередь.
- Пока очередь не пуста, алгоритм выполняет п.3 – п.6.
Псевдокод
— множество состояний ДКА. — множество терминальных состояний. — очередь. — разбиение множества состояний ДКА. — класс состояний ДКА.
if .push() else .push() while not .isEmpty() .pop() for all for all in if and replace in with and if in replace in with and else if .push() else .push()
Корректность алгоритма
Время работы алгоритма
Благодаря системе добавления классов состояний в очередь, каждое ребро будет рассмотрено не более чем раз. А так как ребер у нас порядка то получаем
Литература
- Хопкрофт Д., Мотвани Р., Ульман Д. Введение в теорию автоматов, языков и вычислений, 2-е изд. : Пер. с англ. — М.: Издательский дом «Вильямс», 2002. — С. 177 — ISBN 5-8459-0261-4 (рус.)
- J. E. Hopcroft. An n log n algorithm for minimizing states in a finite automaton. Technical Report CS-71-190, Stanford University, January 1971.