Задача о наибольшей подпоследовательности-палиндроме
Задача: |
Задача о наибольшей подпоследовательности — палиндроме (Largest Palindromic Subsequence) — это задача поиска длины наибольшей подпоследовательности — палиндроме которую можно получить вычеркиванием некоторых букв из данной последовательности. |
Содержание
Определения
Определение: |
Палиндромом (Palindrome) называется строка, которая одинаково читается как слева направо, так и справа налево. |
Определение: |
Подпоследовательностью — палиндромом данной строки (Largest Palindromic Subsequence) называется последовательность символов из данной строки, не обязательно идущих подряд, являющаяся палиндромом. |
Например, HELOLEH является подпоследовательностью-палиндромом строки HTEOLFEOLEH.
Решение
Обозначим данную последовательность через
, а ее элементы — через . Будем рассматривать возможные подпоследовательности данной последовательности с го по ый символ включительно, обозначив её как . Длины максимальных подпалиндромов для данной последовательности будем записывать в квадратный массив : — длина максимальной подпоследовательности — палиндрома, который можно получить из последовательности .Начнем решать задачу с простых подпоследовательностей. Для последовательности из одного элемента (то есть подпоследовательности вида
) ответ очевиден — ничего вычеркивать не надо, такая строка будет искомой подпоследовательностью-палиндромом. Для последовательности из двух элементов возможны два варианта: если элементы равны, то мы имеем подпоследовательность-палиндром, ничего вычеркивать не надо. Если же элементы не равны, то вычеркиваем любой.Пусть теперь нам дана подпоследовательность
. Если и элементы подпоследовательности не совпадают, то один из них нужно вычеркнуть. Тогда у нас останется подпоследовательность или — то есть мы сведем задачу к подзадаче: . Если же первый и последний элементы равны, то мы можем оставить оба, но необходимо знать решение задачи . Таким образом получаем следующее рекуррентное соотношение:
Асимптотика
Каждый элемент массива мы вычисляем
раз за обращаясь к уже вычисленным элементам. Так как размер массива , то алгоритм работает заПример
Рассмотрим решение на примере последовательности ABACCBA. Первым делом заполняем диагональ массива единицами, они будут соответствовать подпоследовательностями
из одного элемента. Затем начинаем рассматривать подпоследовательности длины два. Во всех подпоследовательностях, кроме , элементы различны, поэтому в соответствующие ячейки запишем , а в — .Получается, что мы будем заполнять массив по диагоналям, начиная с главной диагонали. Для подпоследовательностей длины
получаются следующие значения: в подпоследовательности ABA первый и последний элемент равны, поэтому . В остальных подпоследовательностях первый и последний элементы различны.
BAC:
ACC:
CCB:
CBA:
Продолжая далее аналогичные рассуждения, заполним все ячейки над диагональю и в ячейке
получим ответ — .Если же в задаче необходимо вывести не длину, а саму подпоследовательность — палиндром, то дополнительно к массиву длин мы должны построить массив переходов — для каждой ячейки запомнить, какой из случаев был реализован.
Последовательность заполнения массива и массив переходов см. на изображениях ниже.
Псевдокод
Перед вызовом процедуры заполняем
начальными значениями: если , , если , в остальных случаях . При первом вызове функции в качестве аргументов передаем индексы первого и последнего элементов исходной строки. Например для строки длиной вызов функции будет иметь следующий вид: . Искомая же длина будет записана в ячейке .Функция для вычисления длины палиндрома (
palSubSeq(left, right):
if L[left][right] == -1
if s[left] == s[right]
L[left][right] = palSubSeq(left + 1, right - 1) + 2
else
L[left][right] =
(palSubSeq(left + 1, right), palSubSeq(left, right - 1))
return L[left][right]
Функция для построения искомого палиндрома (
// palindrome — массив символов, где в palindrome[i] содержится символ искомой последовательности-палиндрома palChars(left, right, palLeft, palRight) while leftright if left == right and L[left][right] == 1 palindrome[palLeft++] = S[left++] else if S[left] == S[right] palindrome[palLeft++] = S[left++] palindrome[palRight--] = S[right--] else if L[left + 1][right] L[left][right - 1] left++ else right--