Z-функция — различия между версиями
(→Алгоритм поиска) |
Kasetkin (обсуждение | вклад) |
||
Строка 5: | Строка 5: | ||
Дана строка <tex>S</tex>. Необходимо построить массив <tex>Z</tex>, такой что <tex>Z[i]</tex> является префикс функцией данной строки с позиции <tex>i</tex> | Дана строка <tex>S</tex>. Необходимо построить массив <tex>Z</tex>, такой что <tex>Z[i]</tex> является префикс функцией данной строки с позиции <tex>i</tex> | ||
*'''Описание алгоритма''' | *'''Описание алгоритма''' | ||
+ | Для работы алгоритма заведём две переменные: <tex>left</tex> и <tex>right</tex> - начало и конец наибольшего префикса строки <tex>S</tex> с максимальным значением <tex>right</tex>. Изначально <tex>left=0</tex> и <tex>right=0</tex>. | ||
+ | |||
+ | Это динамический алгоритм. Пусть нам известны значения Z-функции от <tex>0</tex> до <tex>i-1</tex>. Найдём <tex>Z[i]</tex>. | ||
+ | Есть два случая: <tex>i > right</tex> и <tex>i \leq right</tex>. | ||
+ | |||
+ | Пусть <tex>i > right</tex>. Тогда просто пробегаемся по строке <tex>S</tex> и сравниваем символы из начала с символами после позиции <tex>i</tex>. (<tex>S[i+j] == S[j]</tex>) | ||
+ | Пусть <tex>j</tex> первая позиция в строке <tex>S</tex> для которой не выполняется равенство <tex>S[i+j] == S[j]</tex>, тогда <tex>j</tx> это и Z-функция для позиции <tex>i</tex>. Тогда <tex>left = i, right = i + j - 1</tex>. | ||
+ | |||
+ | Если <tex>i \leq right</tex> и | ||
+ | |||
*'''Код алгоритма''' | *'''Код алгоритма''' | ||
int[] z(String p) { | int[] z(String p) { |
Версия 22:24, 7 июня 2011
Определение
Z-функция от строки
и позиции , это длина максимального префикса подстроки, начинающейся с позиции в строке , который одновременно является и префиксом всей строки .Алгоритм поиска
- Задача
Дана строка
. Необходимо построить массив , такой что является префикс функцией данной строки с позиции- Описание алгоритма
Для работы алгоритма заведём две переменные:
и - начало и конец наибольшего префикса строки с максимальным значением . Изначально и .Это динамический алгоритм. Пусть нам известны значения 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; }