<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="ru">
		<id>http://neerc.ifmo.ru/wiki/api.php?action=feedcontributions&amp;feedformat=atom&amp;user=178.178.11.56&amp;*</id>
		<title>Викиконспекты - Вклад участника [ru]</title>
		<link rel="self" type="application/atom+xml" href="http://neerc.ifmo.ru/wiki/api.php?action=feedcontributions&amp;feedformat=atom&amp;user=178.178.11.56&amp;*"/>
		<link rel="alternate" type="text/html" href="http://neerc.ifmo.ru/wiki/index.php?title=%D0%A1%D0%BB%D1%83%D0%B6%D0%B5%D0%B1%D0%BD%D0%B0%D1%8F:%D0%92%D0%BA%D0%BB%D0%B0%D0%B4/178.178.11.56"/>
		<updated>2026-05-15T11:44:59Z</updated>
		<subtitle>Вклад участника</subtitle>
		<generator>MediaWiki 1.30.0</generator>

	<entry>
		<id>http://neerc.ifmo.ru/wiki/index.php?title=%D0%9F%D0%BE%D0%B8%D1%81%D0%BA_%D0%BF%D0%BE%D0%B4%D1%81%D1%82%D1%80%D0%BE%D0%BA%D0%B8_%D0%B2_%D1%81%D1%82%D1%80%D0%BE%D0%BA%D0%B5_%D1%81_%D0%B8%D1%81%D0%BF%D0%BE%D0%BB%D1%8C%D0%B7%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D0%B5%D0%BC_%D1%85%D0%B5%D1%88%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D1%8F._%D0%90%D0%BB%D0%B3%D0%BE%D1%80%D0%B8%D1%82%D0%BC_%D0%A0%D0%B0%D0%B1%D0%B8%D0%BD%D0%B0-%D0%9A%D0%B0%D1%80%D0%BF%D0%B0&amp;diff=20728</id>
		<title>Поиск подстроки в строке с использованием хеширования. Алгоритм Рабина-Карпа</title>
		<link rel="alternate" type="text/html" href="http://neerc.ifmo.ru/wiki/index.php?title=%D0%9F%D0%BE%D0%B8%D1%81%D0%BA_%D0%BF%D0%BE%D0%B4%D1%81%D1%82%D1%80%D0%BE%D0%BA%D0%B8_%D0%B2_%D1%81%D1%82%D1%80%D0%BE%D0%BA%D0%B5_%D1%81_%D0%B8%D1%81%D0%BF%D0%BE%D0%BB%D1%8C%D0%B7%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D0%B5%D0%BC_%D1%85%D0%B5%D1%88%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D1%8F._%D0%90%D0%BB%D0%B3%D0%BE%D1%80%D0%B8%D1%82%D0%BC_%D0%A0%D0%B0%D0%B1%D0%B8%D0%BD%D0%B0-%D0%9A%D0%B0%D1%80%D0%BF%D0%B0&amp;diff=20728"/>
				<updated>2012-04-15T15:10:12Z</updated>
		
		<summary type="html">&lt;p&gt;178.178.11.56: &lt;/p&gt;
&lt;hr /&gt;
&lt;div&gt;Алгоритм Рабина {{---}} Карпа {{---}} это алгоритм поиска подстроки в строке с использованием хеширования.&lt;br /&gt;
&lt;br /&gt;
==Метод хеширования==&lt;br /&gt;
&lt;br /&gt;
Следует использовать полиномиальный хеш {{---}} &amp;lt;tex&amp;gt;hash(s[1..n]) = (p^{n - 1} s[1] + ... + p^{0} s[n])&amp;lt;/tex&amp;gt; mod &amp;lt;tex&amp;gt;r&amp;lt;/tex&amp;gt;, где &amp;lt;tex&amp;gt;p&amp;lt;/tex&amp;gt; {{---}} это некоторое простое число, а &amp;lt;tex&amp;gt;r&amp;lt;/tex&amp;gt; {{---}} некоторое большое число, чтобы было меньше коллизий (обычно берётся &amp;lt;tex&amp;gt;2^{32}&amp;lt;/tex&amp;gt; или &amp;lt;tex&amp;gt;2^{64}&amp;lt;/tex&amp;gt;, чтобы модуль брался автоматически при переполнении типов. Стоит обратить внимание, что если 2 строчки имеют одинаковый хэш, то они в большинстве таких случаев равны.&lt;br /&gt;
&lt;br /&gt;
При удалении первого символа строки и добавлении символа в конец считать хеш новой строки при помощи хеша изначальной строки возможно за &amp;lt;tex&amp;gt;O(1)&amp;lt;/tex&amp;gt;:&lt;br /&gt;
&lt;br /&gt;
&amp;lt;tex&amp;gt;hash(s[i + 1..i + m - 1]) = hash(s[i..i + m - 1]) - p^{m - 1} s[i]&amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
&amp;lt;tex&amp;gt;hash(s[i + 1..i + m]) = p \cdot hash(s[i + 1..i + m - 1]) + s[i + m]&amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
Получается : &amp;lt;tex&amp;gt;hash(s[i + 1..i + m]) = p \cdot hash(s[i..i + m - 1]) - p^{m} s[i] + s[i + m]&amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
==Алгоритм==&lt;br /&gt;
&lt;br /&gt;
Есть шаблон {{---}} &amp;lt;tex&amp;gt;p[1..m]&amp;lt;/tex&amp;gt; и строка {{---}} &amp;lt;tex&amp;gt;s[1..n]&amp;lt;/tex&amp;gt;. Нужно найти все вхождения шаблона в строку.&lt;br /&gt;
&lt;br /&gt;
В начале вычисляются &amp;lt;tex&amp;gt;hash(s[1..m])&amp;lt;/tex&amp;gt; и &amp;lt;tex&amp;gt;hash(p[1..m])&amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
Для &amp;lt;tex&amp;gt;i \in [1..n - m + 1]&amp;lt;/tex&amp;gt; вычисляется &amp;lt;tex&amp;gt;hash(s[i..i + m - 1]&amp;lt;/tex&amp;gt; и сравнивается с &amp;lt;tex&amp;gt;hash(p[1..m])&amp;lt;/tex&amp;gt;. Если они получаются равными {{---}} то считается, что подстрока &amp;lt;tex&amp;gt;p&amp;lt;/tex&amp;gt; входит в строку &amp;lt;tex&amp;gt;s&amp;lt;/tex&amp;gt; (начиная с позиции &amp;lt;tex&amp;gt;i&amp;lt;/tex&amp;gt;;) или проверяется, что подстрока является шаблоном, для этого выбираются и сравниваются случайные символы из строк.&lt;br /&gt;
&lt;br /&gt;
Следует предподсчитать &amp;lt;tex&amp;gt;p^{m}&amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
==Псевдокод==&lt;br /&gt;
  '''RabinKarp''' (s[1..n], p[1..m])&lt;br /&gt;
       hp = hash(p[1..m])&lt;br /&gt;
       h = hash(s[1..m])&lt;br /&gt;
       '''for''' i = 1 '''to''' n - m + 1&lt;br /&gt;
            '''if''' h = hp&lt;br /&gt;
                 answer.add(i)&lt;br /&gt;
            h = p * h - p&amp;lt;tex&amp;gt;^{m}&amp;lt;/tex&amp;gt; * hash(s[i]) + hash(s[i + m)&lt;br /&gt;
       '''if''' answer.size() == 0&lt;br /&gt;
            '''return''' not found&lt;br /&gt;
       '''else'''&lt;br /&gt;
            '''return''' answer&lt;br /&gt;
&lt;br /&gt;
Новый хеш &amp;lt;tex&amp;gt;h&amp;lt;/tex&amp;gt; был получен с помощью быстрого пересчёта. Следует считать, что &amp;lt;tex&amp;gt;s[n + 1]&amp;lt;/tex&amp;gt; {{---}} пустой символ.&lt;br /&gt;
&lt;br /&gt;
==Время работы==&lt;br /&gt;
&lt;br /&gt;
Изначальный подсчёт хешей {{---}} &amp;lt;tex&amp;gt;O(m)&amp;lt;/tex&amp;gt;. В цикле всего &amp;lt;tex&amp;gt;n - m + 1&amp;lt;/tex&amp;gt; итераций {{---}} каждая выполняется за &amp;lt;tex&amp;gt;O(1)&amp;lt;/tex&amp;gt;. Итого {{---}} &amp;lt;tex&amp;gt;O(n + m)&amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
== Литература ==&lt;br /&gt;
Кормен Т., Лейзерсон Ч., Ривест Р. Алгоритмы: построение и анализ. {{---}} 2-е изд. {{---}} М.: Издательский дом «Вильямс», 2007. {{---}} С. 1296.&lt;/div&gt;</summary>
		<author><name>178.178.11.56</name></author>	</entry>

	<entry>
		<id>http://neerc.ifmo.ru/wiki/index.php?title=%D0%9D%D0%B0%D0%B8%D0%B2%D0%BD%D1%8B%D0%B9_%D0%B0%D0%BB%D0%B3%D0%BE%D1%80%D0%B8%D1%82%D0%BC_%D0%BF%D0%BE%D0%B8%D1%81%D0%BA%D0%B0_%D0%BF%D0%BE%D0%B4%D1%81%D1%82%D1%80%D0%BE%D0%BA%D0%B8_%D0%B2_%D1%81%D1%82%D1%80%D0%BE%D0%BA%D0%B5&amp;diff=20727</id>
		<title>Наивный алгоритм поиска подстроки в строке</title>
		<link rel="alternate" type="text/html" href="http://neerc.ifmo.ru/wiki/index.php?title=%D0%9D%D0%B0%D0%B8%D0%B2%D0%BD%D1%8B%D0%B9_%D0%B0%D0%BB%D0%B3%D0%BE%D1%80%D0%B8%D1%82%D0%BC_%D0%BF%D0%BE%D0%B8%D1%81%D0%BA%D0%B0_%D0%BF%D0%BE%D0%B4%D1%81%D1%82%D1%80%D0%BE%D0%BA%D0%B8_%D0%B2_%D1%81%D1%82%D1%80%D0%BE%D0%BA%D0%B5&amp;diff=20727"/>
				<updated>2012-04-15T15:07:47Z</updated>
		
		<summary type="html">&lt;p&gt;178.178.11.56: /* Псевдокод */&lt;/p&gt;
&lt;hr /&gt;
&lt;div&gt;==Постановка задачи==&lt;br /&gt;
Имеются строки &amp;lt;tex&amp;gt;T[1 .. n]&amp;lt;/tex&amp;gt; и &amp;lt;tex&amp;gt;P[1 .. m]&amp;lt;/tex&amp;gt; такие, что &amp;lt;tex&amp;gt;n&amp;lt;/tex&amp;gt; &amp;lt;tex&amp;gt;\ge&amp;lt;/tex&amp;gt; &amp;lt;tex&amp;gt;m&amp;lt;/tex&amp;gt; и элементы этих строк  &amp;lt;tex&amp;gt;-&amp;lt;/tex&amp;gt; символы из конечного алфавита &amp;lt;tex&amp;gt; \sum &amp;lt;/tex&amp;gt;. Говорят, что строка &amp;lt;tex&amp;gt;P&amp;lt;/tex&amp;gt; встречается в строке &amp;lt;tex&amp;gt;T&amp;lt;/tex&amp;gt; со сдвигом &amp;lt;tex&amp;gt;s&amp;lt;/tex&amp;gt;, если &amp;lt;tex&amp;gt; 0 \le s \le n-m&amp;lt;/tex&amp;gt; и &amp;lt;tex&amp;gt;T[s + 1 .. s + m]&amp;lt;/tex&amp;gt; = &amp;lt;tex&amp;gt;P[1..m]&amp;lt;/tex&amp;gt;. Если строка &amp;lt;tex&amp;gt;P&amp;lt;/tex&amp;gt; встречается в строке &amp;lt;tex&amp;gt;T&amp;lt;/tex&amp;gt;, то &amp;lt;tex&amp;gt;P&amp;lt;/tex&amp;gt; является подстрокой &amp;lt;tex&amp;gt;T&amp;lt;/tex&amp;gt;. Требуется проверить, является ли строка &amp;lt;tex&amp;gt;P&amp;lt;/tex&amp;gt; подстрокой &amp;lt;tex&amp;gt;T&amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
==Алгоритм==&lt;br /&gt;
В наивном алгоритме поиск всех допустимых сдвигов производится с помощью цикла, в котором проверяется условие &amp;lt;tex&amp;gt;T[s + 1 .. s + m]&amp;lt;/tex&amp;gt; = &amp;lt;tex&amp;gt;P[1..m]&amp;lt;/tex&amp;gt; для каждого из &amp;lt;tex&amp;gt; n-m+1&amp;lt;/tex&amp;gt;  возможных значений &amp;lt;tex&amp;gt;s&amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
==Псевдокод==&lt;br /&gt;
&lt;br /&gt;
 '''naiveStringMatcher''' (T, P)&lt;br /&gt;
    n = length(T)&lt;br /&gt;
    m = length(P)&lt;br /&gt;
    '''for''' s = 0 '''to''' n - m&lt;br /&gt;
       '''if''' T[s + 1 .. s + m] = P[1..m]&lt;br /&gt;
          print()&lt;br /&gt;
&lt;br /&gt;
==Время работы==&lt;br /&gt;
Алгоритм работает за &amp;lt;tex&amp;gt;O(m * (n - m))&amp;lt;/tex&amp;gt;. В худшем случае &amp;lt;tex&amp;gt; m = n / 2 &amp;lt;/tex&amp;gt;, что дает &amp;lt;tex&amp;gt; O(n^2/4) = O(n^2) &amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
== Литература ==&lt;br /&gt;
* ''Кормен Т., Лейзерсон Ч., Ривест Р.'' Алгоритмы: построение и анализ.[http://wmate.ru/ebooks/?dl=380&amp;amp;mirror=1] — 2-е изд. — М.: Издательский дом «Вильямс», 2007. — С. 1296.&lt;/div&gt;</summary>
		<author><name>178.178.11.56</name></author>	</entry>

	</feed>