Z-функция — различия между версиями

Материал из Викиконспекты
Перейти к: навигация, поиск
(Код алгоритма)
Строка 24: Строка 24:
  
 
==Код алгоритма==
 
==Код алгоритма==
  int[] z(String p) {
+
getZfunction(p) // p — исходная строка, n — её длина
    int[] ans = new int[p.length()];
+
     answer[0] = 0
     ans[0] = 0;
+
     left = 0
     int n = p.length();
+
     right = 0
    int left = 0;
+
     for (i = 1..(n - 1))
     int right = 0;
+
      if (i > right)
     for (int i = 1; i < n; i++) {
+
          j = 0
      if (i > right) {
+
          while (i + j < n && p[i + j] == p[j])
        int j = 0;
+
            j++
        while (i + j < n && p.charAt(i+j) == p.charAt(j)) {
+
          answer[i] = j
          j++;
+
          left = i
        }
+
          right = i + j - 1
        ans[i] = j;
+
      else if (answer[i - left] < right - i + 1)
        left = i;
+
           answer[i] = answer[i - left]
        right = i + j - 1;
+
      else  
      } else {
+
           j = 1
        if (ans[i - left] < right - i + 1) {
+
           while (j + right < n && p[j + right - i] == p[right + j])
           ans[i] = ans[i - left];
+
            j++
        } else {
+
           answer[i] = right + j - i
           int j = 1;
+
           left = i
           while (j + right < n && p.charAt(j+right-i) == p.charAt(right + j)) {
+
           right = right + j - 1
            j++;
+
     return answer
           }
 
          ans[i] = right + j - i;
 
           left = i;
 
           right = right + j - 1;
 
        }
 
      }
 
    }
 
     return ans;
 
}
 
  
 
== Источники ==
 
== Источники ==
 
[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-функция от строки [math]S[/math] и позиции [math]x[/math] — это длина максимального префикса подстроки, начинающейся с позиции [math]x[/math] в строке [math]S[/math], который одновременно является и префиксом всей строки [math]S[/math]. Значение [math]Z[0][/math] не определено, поэтому его обычно приравнивают к нулю или к длине строки. Zfunc-examp.png

Алгоритм поиска

Для работы алгоритма заведём две переменные: [math]left[/math] и [math]right[/math] — начало и конец наибольшего префикса строки [math]S[/math] с максимальным значением [math]right[/math]. Изначально [math]left=0[/math] и [math]right=0[/math].

Пусть нам известны значения Z-функции от [math]0[/math] до [math]i-1[/math]. Найдём [math]Z[i][/math]. Рассмотрим два случая.


1) [math]i \gt right[/math]:
Просто пробегаемся по строке [math]S[/math] и сравниваем символы на позициях [math]S[i+j][/math] и [math]S[j][/math]. Пусть [math]j[/math] первая позиция в строке [math]S[/math] для которой не выполняется равенство [math]S[i+j] == S[j][/math], тогда [math]j[/math] это и Z-функция для позиции [math]i[/math]. Тогда [math]left = i, right = i + j - 1[/math].


2) [math]i \le right[/math]:
Сравним [math]Z[i - left] + i[/math] и [math]right[/math]. Если [math]right[/math] меньше, то надо просто пробежаться по строке начиная с позиции [math]right[/math] и вычислить значение [math]Z[i][/math]. Иначе мы уже знаем значение [math]Z[i][/math], так как оно равно значению [math]Z[i - left][/math].
Z-func.png

Время работы

Этот алгоритм работает за [math]O(\lvert S\rvert)[/math], так как каждая позиция пробегается не более двух раз: при попадании в диапазон от [math]left[/math] до [math]right[/math] и при высчитывании 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-функция — Википедия