Z-функция — различия между версиями
Glukos (обсуждение | вклад) (→Код алгоритма) |
(→Псевдокод) |
||
| Строка 24: | Строка 24: | ||
==Псевдокод== | ==Псевдокод== | ||
| − | + | '''Zfunction'''(p) | |
answer[0] = 0 | answer[0] = 0 | ||
left = 0 | left = 0 | ||
right = 0 | right = 0 | ||
| − | for (i = 1..(n - 1)) | + | '''for''' (i = 1..(n - 1)) |
| − | if (i > right) | + | '''if''' (i > right) |
j = 0 | j = 0 | ||
| − | while (i + j < n && p[i + j] == p[j]) | + | '''while''' (i + j < n && p[i + j] == p[j]) |
j++ | j++ | ||
answer[i] = j | answer[i] = j | ||
left = i | left = i | ||
right = i + j - 1 | right = i + j - 1 | ||
| − | else if (answer[i - left] < right - i + 1) | + | '''else if''' (answer[i - left] < right - i + 1) |
answer[i] = answer[i - left] | answer[i] = answer[i - left] | ||
| − | else | + | '''else''' |
j = 1 | j = 1 | ||
| − | while (j + right < n && p[j + right - i] == p[right + j]) | + | '''while''' (j + right < n && p[j + right - i] == p[right + j]) |
j++ | j++ | ||
answer[i] = right + j - i | answer[i] = right + j - i | ||
left = i | left = i | ||
right = right + j - 1 | right = right + j - 1 | ||
| − | return answer | + | '''return''' answer |
== Источники == | == Источники == | ||
[http://habrahabr.ru/post/113266/ Поиск подстроки и смежные вопросы — Хабр]<br> | [http://habrahabr.ru/post/113266/ Поиск подстроки и смежные вопросы — Хабр]<br> | ||
[http://ru.wikipedia.org/wiki/Z-%D1%84%D1%83%D0%BD%D0%BA%D1%86%D0%B8%D1%8F Z-функция — Википедия] | [http://ru.wikipedia.org/wiki/Z-%D1%84%D1%83%D0%BD%D0%BA%D1%86%D0%B8%D1%8F Z-функция — Википедия] | ||
Версия 21:16, 28 мая 2012
Определение
Z-функция от строки и позиции — это длина максимального префикса подстроки, начинающейся с позиции в строке , который одновременно является и префиксом всей строки . Значение не определено, поэтому его обычно приравнивают к нулю или к длине строки.
Алгоритм поиска
Для работы алгоритма заведём две переменные: и — начало и конец наибольшего префикса строки с максимальным значением . Изначально и .
Пусть нам известны значения Z-функции от до . Найдём . Рассмотрим два случая.
1) :
Просто пробегаемся по строке и сравниваем символы на позициях и .
Пусть первая позиция в строке для которой не выполняется равенство , тогда это и Z-функция для позиции . Тогда .
2) :
Сравним и . Если меньше, то надо просто пробежаться по строке начиная с позиции и вычислить значение .
Иначе мы уже знаем значение , так как оно равно значению .
Время работы
Этот алгоритм работает за , так как каждая позиция пробегается не более двух раз: при попадании в диапазон от до и при высчитывании Z-функции простым циклом.
Псевдокод
Zfunction(p)
answer[0] = 0
left = 0
right = 0
for (i = 1..(n - 1))
if (i > right)
j = 0
while (i + j < n && p[i + j] == p[j])
j++
answer[i] = j
left = i
right = i + j - 1
else if (answer[i - left] < right - i + 1)
answer[i] = answer[i - left]
else
j = 1
while (j + right < n && p[j + right - i] == p[right + j])
j++
answer[i] = right + j - i
left = i
right = right + j - 1
return answer
Источники
Поиск подстроки и смежные вопросы — Хабр
Z-функция — Википедия