Z-функция — различия между версиями
| Glukos (обсуждение | вклад)  (→Описание алгоритма) | Glukos (обсуждение | вклад)  | ||
| Строка 56: | Строка 56: | ||
|      return ans; |      return ans; | ||
|   } |   } | ||
| + | |||
| + | == Источники == | ||
| + | [http://habrahabr.ru/post/113266/ Поиск подстроки и смежные вопросы — Хабр] | ||
| + | [http://ru.wikipedia.org/wiki/Z-%D1%84%D1%83%D0%BD%D0%BA%D1%86%D0%B8%D1%8F Z-функция — Википедия] | ||
Версия 02:18, 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;
}
Источники
Поиск подстроки и смежные вопросы — Хабр Z-функция — Википедия
