Изменения

Перейти к: навигация, поиск

Сведение задачи LCA к задаче RMQ

3673 байта добавлено, 19:21, 4 сентября 2022
м
rollbackEdits.php mass rollback
{{Шаблон:Задача|definition == Постановка задачи LCA ==Пусть дано корневое дерево <tex>T</tex>. На вход подаются запросы вида <tex>(u,\;v)</tex>, для каждого запроса требуется найти их наименьшего общего предка.}}
{{Определение
|id=lca_suf_tree|definition = '''Наименьшим общим предком ''' (англ. ''least common ancestor)''' ) двух узлов <tex>u, </tex> и <tex>v</tex> в корневом дереве <tex>T</tex> называется узел <tex>w,</tex> , который среди всех узлов, являющихся предками как узла <tex>u,</tex> , так и <tex>v,</tex> , имеет наибольшую глубину.
}}
Пусть дано корневое дерево <tex>T.</tex> На вход подаются запросы вида <tex>(u,\;v),</tex> для каждого запроса требуется найти их наименьшего общего предка.
== Алгоритм ==
=== Идея ===
Будем решать задачу <tex>LCA</tex>, уже умея решать задачу <tex>RMQ</tex>. Тогда поиск наименьшего общего предка <tex>i</tex>-того и <tex>j</tex>-того элементов сводится к запросу минимума на отрезке массива, который будет введен позднее.
 
=== Препроцессинг ===
1) В каждом узле будет храниться глубина узла в корневом дереве Для каждой вершины <tex>T.</tex>определим глубину с помощью следующей рекурсивной формулы::<tex>\mathrm{depth}(u)= \begin{cases}0 & u = \mathrm{root}(T),\\\mathrm{depth}(v) + 1 & u = \mathrm{son}(v).
\end{cases}</tex>
Ясно, что глубина вершины элементарным образом поддерживается во время обхода в глубину.
 
Запустим [[Обход в глубину, цвета вершин|обход в глубину]] из корня, который будет вычислять значения следующих величин:
#Cписок глубин посещенных вершин <tex>d</tex>. Глубина текущей вершины добавляется в конец списка при входе в данную вершину, а также после каждого возвращения из её сына.
#Список посещений узлов <tex>\mathtt{vtx}</tex>, строящийся аналогично предыдущему, только добавляется не глубина а сама вершина.
#Значение функции <tex>\mathtt{I}[u]</tex>, возвращающей индекс в списке глубин <tex>d</tex>, по которому была записана глубина вершины <tex>u</tex> (например на момент входа в вершину).
2) Запустим обход Вот таким образом будут выглядеть эти три массива после обхода в глубину из корня, который будет строить список посещений узлов. Глубина текущей вершины добавляется в список при входе в эту вершину, а также после каждого возвращения из её сына.:
3) Построим дерево отрезков с операцией взятия минимума на отрезке по полученному списку узлов[[Файл:HoD8KSiOzTg.jpg‎ | мини | left | 700x500px| Пример массива <tex>\mathtt{vtx}</tex>]]<br clear="all">
=== Запрос ===
Пусть имеется Будем считать, что <tex>\mathrm{rmq}(d,l,r)</tex> возвращает индекс минимального элемента в <tex>d</tex> на отрезке <tex>[l..r]</tex>. Тогда ответом на запрос пара узлов <tex>\mathrm{lca}(u, v.)</tex> Обозначим , где <tex>\mathtt{I}[u] \leqslant \mathtt{I}[v]</tex> - индекс ячейки в списке глубин, в которой хранится глубина узла будет <tex>\mathtt{vtx}[\mathrm{rmq}(d,\mathtt{I}[u], \mathtt{I}[v])]</tex>.=== Доказательство корректности алгоритма ==={{Теорема|statement=Наименьшему общему предку вершин <tex>u, v</tex> Тогда запрос минимального элемента соответствует минимальная глубина на отрезке <tex>d[\mathtt{I}[u], \mathtt{I}[v]]</tex> будет равен глубине .|proof=Рассмотрим два узла <tex>u, v</tex> корневого дерева <tex>T</tex>. Рассмотрим отрезок <tex>d[\mathtt{I}[u]..\mathtt{I}[v]]</tex>. Поскольку этот отрезок {{---}} путь из <tex>u</tex> в <tex>v</tex>, он проходит через их наименьшего общего предка узлов <tex>w</tex> (в дереве есть только один простой путь между вершинами), а следовательно минимум на отрезке никак не больше глубины <tex>w</tex>. Заметим, что в момент добавления <tex>\mathtt{I}[u]</tex> обход посещал поддерево с корнем <tex>w</tex>. В момент добавления <tex>\mathtt{I}[v]</tex> мы все еще в поддереве с корнем <tex>w</tex>. Значит, и на отрезке между <tex>\mathtt{I}[u]</tex> и <tex>\mathtt{I}[v]</tex> мы находились внутри поддерева с корнем <tex>w</tex>.Отсюда сделаем заключение, что на рассматриваемом отрезке не посещалась вершина, отличная от <tex>w</tex>, с глубиной меньшей либо равной глубины <tex>w</tex>, т. к. подобной вершины нет в поддереве с корнем <tex>w</tex>.}}.
== Доказательство корректности алгоритма Пример ==Рассмотрим два узла <tex>uдерево на рисунке 1. Найдем наименьшего общего предка вершин, v</tex> корневого дерева <tex>T</tex>помеченных красным цветом. Пусть узел <tex>w</tex> Список глубин, получающийся в результате обхода в глубину {{--- наименьший общий предок узлов }} <tex>u[0, 1, 2, 1, 2, v</tex>1, 0, 1, 0]. Очевидно, что в поддереве с корнем <tex>w</tex> узел Глубина наименьшего общего предка красных вершин равна минимуму на отрезке <tex>w[2, 1, 0, 1].</tex> будет иметь наименьшую глубину.Осталось доказать, что всегда выполняется неравенство <tex>I[u] \le I[wФайл:Lca to rmq.png(2).png|thumb|center|400x200px|Рисунок к примеру] \le I[v]<div style='clear:left;'></texdiv>
== Сложность ==
=== Препроцессинг ===Длина массива глубин будет равна Для нахождения минимального элемента на отрезке можно использовать [[Алгоритм Фарака-Колтона и Бендера|алгоритм Фарака-Колтона и Бендера]] для <tex>(2 * n - 1),\pm 1RMQ</tex> , т.ек. дерево отрезков будет построено за соседние элементы в списке глубин отличаются не более чем на единицу. Длина списка глубин составляет <tex>O(n2n - 1).</tex> Таким , таким образом, препроцессинг работает за <tex>O(n).</tex>=== Запрос ===. Время выполнения запроса равно времени запроса минимального элемента на отрезке в дереве отрезков, т.е. {{---}} <tex>O(log n1).</tex>.
== См.также ==
*[[Алгоритм Фарака-Колтона и Бендера]]
*[[Сведение задачи RMQ к задаче LCA]]
== Ссылки Источники информации ==
*[http://e-maxx.ru/algo/lca Наименьший общий предок. Нахождение за O (sqrt (N)) и O (log N) с препроцессингом O (N)]
 
[[Категория: Алгоритмы и структуры данных]]
[[Категория: Задача о наименьшем общем предке]]
1632
правки

Навигация