Z-функция — различия между версиями
(→Алгоритм поиска) |
Glukos (обсуждение | вклад) (→Описание алгоритма) |
||
Строка 8: | Строка 8: | ||
Пусть нам известны значения Z-функции от <tex>0</tex> до <tex>i-1</tex>. Найдём <tex>Z[i]</tex>. | Пусть нам известны значения Z-функции от <tex>0</tex> до <tex>i-1</tex>. Найдём <tex>Z[i]</tex>. | ||
− | + | Рассмотрим два случая. | |
<br> | <br> |
Версия 01:57, 16 апреля 2012
Содержание
Определение
Z-функция от строки
и позиции — это длина максимального префикса подстроки, начинающейся с позиции в строке , который одновременно является и префиксом всей строки .Алгоритм поиска
Задача
Дана строка
. Необходимо построить массив , такой, что является префикс функцией данной строки с позицииОписание алгоритма
Для работы алгоритма заведём две переменные:
и — начало и конец наибольшего префикса строки с максимальным значением . Изначально и .Пусть нам известны значения Z-функции от
до . Найдём . Рассмотрим два случая.
1) :
Просто пробегаемся по строке и сравниваем символы на позициях и .
Пусть первая позиция в строке для которой не выполняется равенство , тогда это и Z-функция для позиции . Тогда .
2) :
Сравним и . Если меньше, то надо просто пробежаться по строке начиная с позиции и вычислить значение .
Иначе мы уже знаем значение , так как оно равно значению .
Время работы алгоритма
Этот алгоритм работает за
, так как каждая позиция пробегается не более двух раз: при попадании в диапазон от до и при высчитывании Z-функции простым циклом.Код алгоритма
int[] z(String p) { int[] ans = new int[p.length()]; ans[0] = 0; int n = p.length(); int left = 0; int right = 0; for (int i = 1; i < n; i++) { if (i > right) { int j = 0; while (i + j < n && p.charAt(i+j) == p.charAt(j)) { j++; } ans[i] = j; left = i; right = i + j - 1; } else { if (ans[i - left] < right - i + 1) { ans[i] = ans[i - left]; } else { int j = 1; while (j + right < n && p.charAt(j+right-i) == p.charAt(right + j)) { j++; } ans[i] = right + j - i; left = i; right = right + j - 1; } } } return ans; }