Z-функция — различия между версиями
(→Алгоритм поиска) |
Glukos (обсуждение | вклад) (→Определение) |
||
Строка 1: | Строка 1: | ||
==Определение== | ==Определение== | ||
− | Z-функция от строки <tex>S</tex> и позиции <tex>x</tex> — это длина максимального префикса подстроки, начинающейся с позиции <tex>x</tex> в строке <tex>S</tex>, который одновременно является и префиксом всей строки <tex>S</tex>. | + | Z-функция от строки <tex>S</tex> и позиции <tex>x</tex> — это длина максимального префикса подстроки, начинающейся с позиции <tex>x</tex> в строке <tex>S</tex>, который одновременно является и префиксом всей строки <tex>S</tex>.<br> |
+ | [[Файл:Zfunc-examp.jpg]] | ||
+ | |||
==Алгоритм поиска== | ==Алгоритм поиска== | ||
Для работы алгоритма заведём две переменные: <tex>left</tex> и <tex>right</tex> — начало и конец наибольшего префикса строки <tex>S</tex> с максимальным значением <tex>right</tex>. Изначально <tex>left=0</tex> и <tex>right=0</tex>. | Для работы алгоритма заведём две переменные: <tex>left</tex> и <tex>right</tex> — начало и конец наибольшего префикса строки <tex>S</tex> с максимальным значением <tex>right</tex>. Изначально <tex>left=0</tex> и <tex>right=0</tex>. |
Версия 16:43, 9 мая 2012
Определение
Z-функция от строки
Файл:Zfunc-examp.jpg
Алгоритм поиска
Для работы алгоритма заведём две переменные:
и — начало и конец наибольшего префикса строки с максимальным значением . Изначально и .Пусть нам известны значения 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; }
Источники
Поиск подстроки и смежные вопросы — Хабр
Z-функция — Википедия