Поиск наибольшей общей подстроки двух строк с использованием хеширования

Материал из Викиконспекты
Перейти к: навигация, поиск

Постановка задачи

Имеются строки [math]S[/math] и [math]T[/math] такие, что элементы этих строк [math]-[/math] символы из конечного алфавита [math] \sum [/math]. Говорят, что строка [math]Z[/math][1 .. m] является подстрокой строки [math]S[/math][1 .. n], если существует такой индекс [math]k[/math] ∈ [1 .. n - m], что для любого [math]i[/math] ∈ [1 .. m] справедливо [math]S[k + i] = Z[i][/math]. Требуется найти такую строку [math]Z[/math], максимальной длины, что [math]Z[/math] является и подстрокой [math]S[/math], и подстрокой [math]T[/math].

Алгоритм

Данный алгоритм основывается на методе половинного деления. Пусть длина наибольшей общей подстроки будет [math]x[/math]. Заметим, что у строк [math]S[/math] и [math]T[/math] обязательно найдется общая подстрока длины [math]y[/math] ∈ [0 .. [math]x[/math]], так как в качестве такой строки можно взять префикс наибольшей общей подстроки. Рассмотрим функцию [math]f[/math] : [1 .. min(len([math]S[/math]), len([math]T[/math]))] → [math]\mathbb{Z}[/math], которая для [math]i[/math] из области определения равна [math]i[/math], если у строк [math]S[/math] и [math]T[/math] есть общая подстрока длины [math]i[/math], иначе она равна 0. Согласно замечанию, функция [math]f[/math] должна по мере возрастания [math]i[/math] строго монотонно возрастать до некоторого момента, а затем обращаться в 0. Таким образом на области определения у функции [math]f[/math] достигается максимум. Собственно, этот максимум и является длиной наибольшей общей подстроки у строк [math]S[/math] и [math]T[/math], так как функция [math]f[/math] специально так определена. Таким образом требуется с помощью бинарного поиска найти максимум функции [math]f[/math] на ее множестве определения. В ходе работы придется проверять наличие общей подстроки заданной длины. При этом предполагается использовать хэширование, чтобы улучшить асимптотику алгоритма. Делается это следующим образом: 1) у строки [math]S[/math] хэшируем подстроки заданной длины и полученные хэши записываем в Set. 2) у строки [math]T[/math] хэшируем подстроки заданной длины и в случае совпадения хэша с элементом Set выполняем посимвольную проверку на совпадение подстрок. Предполагается, что хэширование будет проводится так же, как и в алгоритме Рабина-Карпа.