<?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=80.79.247.164&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=80.79.247.164&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/80.79.247.164"/>
		<updated>2026-05-19T17:59:55Z</updated>
		<subtitle>Вклад участника</subtitle>
		<generator>MediaWiki 1.30.0</generator>

	<entry>
		<id>http://neerc.ifmo.ru/wiki/index.php?title=%D0%94%D0%B5%D0%BA%D0%B0%D1%80%D1%82%D0%BE%D0%B2%D0%BE_%D0%B4%D0%B5%D1%80%D0%B5%D0%B2%D0%BE_%D0%BF%D0%BE_%D0%BD%D0%B5%D1%8F%D0%B2%D0%BD%D0%BE%D0%BC%D1%83_%D0%BA%D0%BB%D1%8E%D1%87%D1%83&amp;diff=24380</id>
		<title>Декартово дерево по неявному ключу</title>
		<link rel="alternate" type="text/html" href="http://neerc.ifmo.ru/wiki/index.php?title=%D0%94%D0%B5%D0%BA%D0%B0%D1%80%D1%82%D0%BE%D0%B2%D0%BE_%D0%B4%D0%B5%D1%80%D0%B5%D0%B2%D0%BE_%D0%BF%D0%BE_%D0%BD%D0%B5%D1%8F%D0%B2%D0%BD%D0%BE%D0%BC%D1%83_%D0%BA%D0%BB%D1%8E%D1%87%D1%83&amp;diff=24380"/>
				<updated>2012-06-07T17:39:11Z</updated>
		
		<summary type="html">&lt;p&gt;80.79.247.164: &lt;/p&gt;
&lt;hr /&gt;
&lt;div&gt;==Основная идея==&lt;br /&gt;
Возьмем структуру данных [[Саморасширяющийся массив|вектор]]. В её стандартной реализации мы умеем добавлять элемент в конец вектора, узнавать значение элемента, стоящего на определенной позиции, изменять элемент по номеру и удалять последний элемент. Предположим, что нам необходима структура данных с вышеуказанными свойствами, а также с операциями: добавить элемент в любое место (с соответствующим изменением нумерации элементов) и удалить любой элемент (также с соответствующим изменением нумерации). Такая структура существует и называется ''декартово дерево по неявному ключу''.&lt;br /&gt;
[[Файл:Tree_1.png|right|250px|thumb|Пример описанного дерева с демонстрацией определения ключа &amp;lt;tex&amp;gt;X&amp;lt;/tex&amp;gt;]]&lt;br /&gt;
Как известно, [[декартово дерево]] {{---}} это структура данных, объединяющая в себе бинарное дерево поиска и бинарную кучу. При реализации же декартова дерева по неявному ключу модифицируем эту структуру. А именно, оставим в нем только приоритет &amp;lt;tex&amp;gt;Y&amp;lt;/tex&amp;gt;, а вместо ключа &amp;lt;tex&amp;gt;X&amp;lt;/tex&amp;gt; будем использовать следующую величину: '''количество элементов в нашей структуре, находящихся левее нашего элемента'''. Иначе говоря, будем считать ключом порядковый номер нашего элемента в дереве, уменьшенный на единицу. &lt;br /&gt;
&lt;br /&gt;
Заметим, что при этом сохранится структура [[Дерево_поиска,_наивная_реализация|двоичного дерева поиска]] по этому ключу (то есть модифицированное декартово дерево так и останется декартовым деревом). Однако, с этим подходом появляется проблема: операции добавления и удаления элемента могут поменять нумерацию, и при наивной реализации на изменение всех ключей потребуется &amp;lt;tex&amp;gt;O(n)&amp;lt;/tex&amp;gt; времени, где &amp;lt;tex&amp;gt;n&amp;lt;/tex&amp;gt; {{---}} количество элементов в дереве.&lt;br /&gt;
&lt;br /&gt;
Решается эта проблема довольно просто. Основная идея заключается в том, что такой ключ &amp;lt;tex&amp;gt;X&amp;lt;/tex&amp;gt; сам по себе нигде не хранится. Вместо него будем хранить вспомогательную величину &amp;lt;tex&amp;gt;C&amp;lt;/tex&amp;gt;: '''количество вершин в поддереве нашей вершины''' (в поддерево включается и сама вершина). Обратим внимание, что все операции с обычным декартовым деревом делались сверху. Также заметим, что если по пути до некой вершины просуммировать все такие величины в левых поддеревьях, в которые мы не пошли, увеличенные на единицу, то придя в саму вершину и добавив к этой величине количество элементов в её левом поддереве, мы получим как раз ее ключ &amp;lt;tex&amp;gt;X&amp;lt;/tex&amp;gt;.&lt;br /&gt;
&lt;br /&gt;
==Операции, поддерживающие структуру декартова дерева==&lt;br /&gt;
Структура обычного декартова дерева поддерживается с помощью двух операций: &amp;lt;tex&amp;gt;split&amp;lt;/tex&amp;gt; {{---}} разбиение одного декартова дерева  на два таких, что в одном ключ &amp;lt;tex&amp;gt;X&amp;lt;/tex&amp;gt; меньше, чем заданное значение, а в другом {{---}} больше, и &amp;lt;tex&amp;gt;merge&amp;lt;/tex&amp;gt; {{---}} слияние двух деревьев, в одном из которых все ключи &amp;lt;tex&amp;gt;X&amp;lt;/tex&amp;gt; меньше, чем во втором. С учетом отличий декартова дерева по неявному ключу от обычного, операции теперь будут описываться так: &amp;lt;tex&amp;gt;split(root, t)&amp;lt;/tex&amp;gt; {{---}} разбиение дерева на два так, что в левом окажется ровно &amp;lt;tex&amp;gt;t&amp;lt;/tex&amp;gt; вершин, и &amp;lt;tex&amp;gt;merge(root1, root2)&amp;lt;/tex&amp;gt; {{---}} слияние двух любых деревьев, соответственно.&lt;br /&gt;
&lt;br /&gt;
===Split===&lt;br /&gt;
Пусть процедура &amp;lt;tex&amp;gt;split&amp;lt;/tex&amp;gt; запущена в корне дерева с требованием отрезать от дерева &amp;lt;tex&amp;gt;t&amp;lt;/tex&amp;gt; вершин. Также известно, что в левом поддереве вершины находится &amp;lt;tex&amp;gt;l&amp;lt;/tex&amp;gt; вершин, а в правом &amp;lt;tex&amp;gt;r&amp;lt;/tex&amp;gt;. Рассмотрим все возможные случаи: &lt;br /&gt;
* &amp;lt;tex&amp;gt;l = t&amp;lt;/tex&amp;gt;. В этом случае процедура &amp;lt;tex&amp;gt;split&amp;lt;/tex&amp;gt; должна просто пометить, что у корня больше нет левого сына, и вернуть его бывшего левого сына в качестве левой части ответа, а сам корень {{---}} в качестве правой. &lt;br /&gt;
* Случай (&amp;lt;tex&amp;gt;t = l + 1&amp;lt;/tex&amp;gt;) рассматривается аналогично предыдущему. &lt;br /&gt;
* &amp;lt;tex&amp;gt;t &amp;lt; l&amp;lt;/tex&amp;gt;. В этом случае нужно рекурсивно запустить процедуру &amp;lt;tex&amp;gt;split&amp;lt;/tex&amp;gt; от левого сына с тем же параметром &amp;lt;tex&amp;gt;t&amp;lt;/tex&amp;gt;, и левая часть сына станет левой частью ответа нашей процедуры, а правая часть сына станет левым сыном корня, после чего корень станет правой частью ответа. &lt;br /&gt;
* Случай &amp;lt;tex&amp;gt;t &amp;gt; l + 1&amp;lt;/tex&amp;gt; рассматривается аналогично предыдущему, с той лишь разницей, что от правого сына отрезается &amp;lt;tex&amp;gt;t - l - 1&amp;lt;/tex&amp;gt; вершин.&lt;br /&gt;
&lt;br /&gt;
===Merge===&lt;br /&gt;
Посмотрим любую из [[Декартово дерево#Операция merge|реализаций]] процедуры &amp;lt;tex&amp;gt;merge&amp;lt;/tex&amp;gt;. Заметим, что в ней программа ни разу не обращается к ключу &amp;lt;tex&amp;gt;X&amp;lt;/tex&amp;gt;. Поэтому реализация процедуры &amp;lt;tex&amp;gt;merge&amp;lt;/tex&amp;gt; для декартова дерева по неявному ключу вообще не будет отличаться от реализации той же процедуры в обычном декартовом дереве.&lt;br /&gt;
&lt;br /&gt;
===Поддержание корректности значений C===&lt;br /&gt;
Единственное действие, обеспечивающее корректность этих значений заключается в том, что после любого действия с детьми вершины нужно записать в ее поле &amp;lt;tex&amp;gt;C&amp;lt;/tex&amp;gt; сумму этих значений в ее новых детях, увеличенную на единицу.&lt;br /&gt;
&lt;br /&gt;
==Применение описанного дерева==&lt;br /&gt;
Таким образом, описана структура, от которой можно отрезать слева кусок произвольной длины и слить два любых куска в один в нужном порядке. Теперь мы имеем возможность:&lt;br /&gt;
* вставить элемент в любое место (отрежем нужное количество элементов слева, сольем левое дерево с деревом из одного добавленного элемента и результат {{---}} с правым деревом);&lt;br /&gt;
* переставить любой кусок массива куда угодно (сделаем нужные разрезы и слияния в правильном порядке);&lt;br /&gt;
* совершать групповые операции с элементами. Вспомним реализацию таких операций в дереве отрезков и поймем, что ничего не помешает нам сделать то же самое с описанным деревом. В групповые операции включается, естественно, и взятие функции от отрезка.&lt;br /&gt;
* сделав на одном исходном массиве два дерева из элементов разной четности, можно решить задачу про смену мест четных и нечетных на отрезке.&lt;br /&gt;
&lt;br /&gt;
==Ссылки==&lt;br /&gt;
* [http://habrahabr.ru/post/102364/ Habrahabr - Декартово дерево по неявному ключу]&lt;br /&gt;
* [http://e-maxx.ru/algo/treap#7 E-maxx - Неявные декартовы деревья]&lt;br /&gt;
&lt;br /&gt;
[[Категория: Деревья поиска]]&lt;br /&gt;
[[Категория: Дискретная математика и алгоритмы]]&lt;/div&gt;</summary>
		<author><name>80.79.247.164</name></author>	</entry>

	</feed>