Z-функция — различия между версиями
Glukos (обсуждение | вклад) |
Glukos (обсуждение | вклад) (→Код алгоритма) |
||
Строка 24: | Строка 24: | ||
==Код алгоритма== | ==Код алгоритма== | ||
− | + | getZfunction(p) // p — исходная строка, n — её длина | |
− | + | answer[0] = 0 | |
− | + | left = 0 | |
− | + | right = 0 | |
− | + | for (i = 1..(n - 1)) | |
− | + | if (i > right) | |
− | for ( | + | 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 | |
− | while (j + right < n && p | + | right = right + j - 1 |
− | + | return answer | |
− | |||
− | |||
− | left = i | ||
− | right = right + j - 1 | ||
− | |||
− | |||
− | |||
− | return | ||
− | |||
== Источники == | == Источники == | ||
[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-функция — Википедия] |
Версия 17:21, 9 мая 2012
Определение
и позиции — это длина максимального префикса подстроки, начинающейся с позиции в строке , который одновременно является и префиксом всей строки . Значение не определено, поэтому его обычно приравнивают к нулю или к длине строки.Алгоритм поиска
Для работы алгоритма заведём две переменные:
и — начало и конец наибольшего префикса строки с максимальным значением . Изначально и .Пусть нам известны значения Z-функции от
до . Найдём . Рассмотрим два случая.
1) :
Просто пробегаемся по строке и сравниваем символы на позициях и .
Пусть первая позиция в строке для которой не выполняется равенство , тогда это и Z-функция для позиции . Тогда .
2) :
Сравним и . Если меньше, то надо просто пробежаться по строке начиная с позиции и вычислить значение .
Иначе мы уже знаем значение , так как оно равно значению .
Время работы
Этот алгоритм работает за
, так как каждая позиция пробегается не более двух раз: при попадании в диапазон от до и при высчитывании Z-функции простым циклом.Код алгоритма
getZfunction(p) // p — исходная строка, n — её длина 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-функция — Википедия