Изменения

Перейти к: навигация, поиск

Алгоритм Кнута-Морриса-Пратта

500 байт добавлено, 18:52, 10 марта 2018
м
Замечание
Дана цепочка <tex>T</tex> и образец <tex>P</tex>. Требуется найти все позиции, начиная с которых <tex>P</tex> входит в <tex>T</tex>.
<br>
Построим строку <tex>S = P\#T</tex>, где <tex>\#</tex> — любой символ, не входящий в алфавит <tex>P</tex> и <tex>T</tex>. Посчитаем на ней значение [[Префикс-функция|префикс-функции]]<tex> p </tex>. Благодаря разделительному символу <tex>\#</tex>, выполняется <tex>\forall i: \pi(p[i) ] \leqslant |P|</tex>. Заметим, что по определению [[Префикс-функция|префикс-функции]] при <tex>i > |P|</tex> и <tex>\pi(p[i) ] = |P|</tex> подстроки длины <tex>P</tex>, начинающиеся с позиций <tex>0</tex> и <tex>i - |P| + 1</tex>, совпадают. Соберем все такие позиции <tex>i - |P| + 1</tex> строки <tex>S</tex>, вычтем из каждой позиции <tex>|P| + 1</tex>, это и будет ответ. Другими словами, если в какой-то позиции <tex>i</tex> выполняется условие <tex>\pi(p[i)]=|P|</tex>, то в этой позиции начинается очередное вхождение образца в цепочку.<br> 
[[Файл:kmp_pict2.png|640px]]
==Псевдокод==
'''int'''[] kmp('''string''' TP, '''string''' PT): '''int''' p pl = P.length '''int''' t tl = T.length
'''int'''[] answer
'''int'''[] <tex>\pi</tex> p = [[Префикс-функция#Эффективный_алгоритм|prefixFunction(P + "#" + T)]]
'''int''' count = 0
'''for''' i = 0 .. (t tl - 1) '''if''' <tex>\pi</tex>p[p pl + i + 1] == ppl
answer[count++] = i
'''return''' answer
==Оценка по памяти==
Предложенная реализация имеет оценку по памяти <tex>O(P+T)</tex>. Оценки <tex>O(P)</tex> можно добиться за счет отказа от запоминания значений префикс-функции для позиций в <tex>S</tex>, меньших <tex>p |P| + 1</tex> (т.е. то есть до начала цепочки <tex>T</tex>). Это возможно, так как значение префикс -функции не может превысить длину образца, благодаря разделительному символу <tex>\#</tex>. ==Замечание==Вместо [[Префикс-функция|префикс-функции]] в алгоритме Кнута-Морриса-Пратта можно использовать [[Z-функция|Z-функцию]]. Оценки времени работы и памяти при этом не изменятся.
==См. также==
*[[Алгоритм Бойера-Мура|Алгоритм Бойера-Мура]]
*[[Алгоритм Колусси|Алгоритм Колусси]]
*[[Префикс-функция|Префикс-функция]]
*[[Z-функция|Z-функция]]
==Источникиинформации==*[[wikipedia:en:Knuth–Morris–Pratt algorithm | Wikipedia {{---}} Knuth–Morris–Pratt algorithm]]
*[[wikipedia:ru:Алгоритм Кнута — Морриса — Пратта | Википедия {{---}} Алгоритм Кнута — Морриса — Пратта]]
*[[wikipedia:en:Knuth–Morris–Pratt algorithm | Wikipedia {{---}} Knuth–Morris–Pratt algorithm]]
*Кормен, Т., Лейзерсон, Ч., Ривест, Р., Штайн — Алгоритмы: построение и анализ / пер. с англ. — изд. 2-е — М.: Издательский дом «Вильямс», 2009. — с.1036. — ISBN 978-5-8459-0857-5.
[[Категория: Алгоритмы и структуры данных]]
[[Категория: Поиск подстроки в строке]]
[[Категория:Точный поиск]]
7
правок

Навигация