Двоичные деревья поиска. Очередь с приоритетами

Содержание

Слайд 2

Цель лекции

Изучить теоретические основы двоичных деревьев поиска и очереди с приоритетами и

Цель лекции Изучить теоретические основы двоичных деревьев поиска и очереди с приоритетами
алгоритмы их обработки
Изучить методы реализации этих структур данных и алгоритмов на языке программирования Pascal (Delphi)

Слайд 3

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

Реализовать структуру данных, хранящую набор чисел и поддерживающую операции:
добавить число
найти число
удалить

Постановка задачи Реализовать структуру данных, хранящую набор чисел и поддерживающую операции: добавить
число
найти минимум
найти максимум
найти k-ое по порядку числу (k-ую порядковую статистику)

Слайд 4

Двоичное дерево

Дерево, каждая вершина которого имеет не более двух детей
На детях каждой

Двоичное дерево Дерево, каждая вершина которого имеет не более двух детей На
вершины задан порядок – есть «левый» ребенок и «правый» ребенок
Самая верхняя вершина – корень – не имеет родителя
Глубина вершины – расстояние от нее до корня
Высота дерева – наибольшая глубина его вершины

Слайд 5

Двоичное дерево поиска

Это двоичное дерево, в каждой вершине которого хранится число
При этом

Двоичное дерево поиска Это двоичное дерево, в каждой вершине которого хранится число
если в некоторой вершине A хранится число x, то:
все числа в левом поддереве A меньше чем x
все числа в правом поддереве A больше чем x

Слайд 6

Дерево поиска неоднозначно

Набор чисел: 1, 3, 10, 23, 50

Высота есть O(logn)

Высота есть

Дерево поиска неоднозначно Набор чисел: 1, 3, 10, 23, 50 Высота есть O(logn) Высота есть O(n)
O(n)

Слайд 7

Вершина дерева

node = record
data : integer;
left, right : integer; // при реализации

Вершина дерева node = record data : integer; left, right : integer;
// на собственном // менеджере памяти
end;

-1 соответствует тому, что соответствующего ребенка у вершины нет

Слайд 8

Поиск элемента в дереве

Если значение y в текущей вершине равно искомому x,

Поиск элемента в дереве Если значение y в текущей вершине равно искомому
то поиск завершен
Если x > y, то перейти к правому ребенку
Если x < y, то перейти к левому ребенку

Слайд 9

Программа

function find(x : integer) : boolean;
begin
cur := root; // root – корень

Программа function find(x : integer) : boolean; begin cur := root; //
дерева
result := false;
while (cur <> -1) do begin
if (memory[cur].data = x) then begin
result := true;
exit;
end;
if (x > memory[cur].data) then begin
cur := memory[cur].right;
end else begin
cur := memory[cur].left;
end;
end;
end;

Слайд 10

Поиск минимума и максимума

Минимум – самый левый элемент в дереве
Максимум – самый

Поиск минимума и максимума Минимум – самый левый элемент в дереве Максимум
правый элемент в дереве

Слайд 11

Добавить число

Необходимо:
Найти место, в котором должно находиться число
Создать новый узел дерева и

Добавить число Необходимо: Найти место, в котором должно находиться число Создать новый
поместить в него это число
Считаем, что все числа в дереве различны

Слайд 12

Программа (1)

function add(root, x : integer) : integer;
begin
if (root = -1) then

Программа (1) function add(root, x : integer) : integer; begin if (root
begin
cur := free;
inc(free);
memory[cur] := data;
result := cur;
exit;
end;
if (memory[root].data = x) then begin
// Элемент уже есть в дереве
result := root;
exit;
end;

Слайд 13

Программа (2)

if (x < memory[root].data) then begin
cur := add(memory[root].left, x);
memory[root].left := cur;
result

Программа (2) if (x cur := add(memory[root].left, x); memory[root].left := cur; result
:= root;
exit;
end;
if (x > memory[root].data) then begin
cur := add(memory[root].right, x);
memory[root].right := cur;
result := root;
exit;
end;
end;

Слайд 14

Вызов процедуры

root := add(root, x);
Здесь root – это корень дерева.

Вызов процедуры root := add(root, x); Здесь root – это корень дерева.

Слайд 15

Удаление элемента

Необходимо найти элемент
Три случая:
вершина является листом – просто удаляем
у вершины один

Удаление элемента Необходимо найти элемент Три случая: вершина является листом – просто
ребенок – заменяем на ребенка
у вершины два ребенка

Слайд 16

Третий случай

Необходимо найти минимальную вершину в правом поддереве и переместить ее на

Третий случай Необходимо найти минимальную вершину в правом поддереве и переместить ее
место удаляемой
Проблем с перемещением не возникнет, так как у перемещаемой вершины не более одного ребенка

Слайд 17

Время работы операций

Время работы всех рассмотренных операций пропорционально высоте дерева – в

Время работы операций Время работы всех рассмотренных операций пропорционально высоте дерева –
худшем случае есть O(n)
Существуют методы выполнения этих операций так, чтобы высота все время была O(logn):
АВЛ-деревья
красно-черные деревья

Они будут рассмотрены в следующих лекциях

Слайд 18

Упражнение

Предложить алгоритм нахождения следующего и предыдущего элемента в двоичном дереве поиска со

Упражнение Предложить алгоритм нахождения следующего и предыдущего элемента в двоичном дереве поиска
временем работы пропорциональным высоте дерева

Слайд 19

Поиск k-ой порядковой статистики

В каждом узле дерева необходимо хранить дополнительную информацию –

Поиск k-ой порядковой статистики В каждом узле дерева необходимо хранить дополнительную информацию
число узлов в его поддереве
node = record
data : integer;
left, right : integer;
size : integer;
end;
memory[a].size = memory[memory[a].left].size + memory[memory[a].right].size + 1

Слайд 20

Поиск k-ого элемента

Дано:
Корень дерева
Число k
Если в левом поддереве не меньше, чем k

Поиск k-ого элемента Дано: Корень дерева Число k Если в левом поддереве
элементов, то переходим к нему
Если в левом поддереве ровно k-1 элемент, то k-ый элемент находится в корне
Иначе переходим к правому поддереву

Слайд 21

Программа

function getKth(root, k : integer) : integer;
begin
sizeLeft := 0;
if (memory[root].left <> -1)

Программа function getKth(root, k : integer) : integer; begin sizeLeft := 0;
then begin
sizeLeft := memory[memory[root].left].size;
end;
if (k <= sizeLeft) then begin
result := getKth(memory[root].left, k);
exit;
end;
if (sizeLeft = k – 1) then begin
result := memory[root].data;
exit;
end;
result := getKth(memory[root].right, k–sizeLeft–1);
end;

Слайд 22

Упражнение

Модифицировать операции добавления и удаления вершины, так чтобы они обновляли значения поля

Упражнение Модифицировать операции добавления и удаления вершины, так чтобы они обновляли значения
«размер поддерева». Время работы операций не должно измениться.

Слайд 23

Деревья выражений

(5+2)*3 – (1 + 2/3)

Деревья выражений (5+2)*3 – (1 + 2/3)

Слайд 24

Обходы двоичных деревьев

Основные варианты обхода двоичного дерева:
корень – левое поддерево – правое

Обходы двоичных деревьев Основные варианты обхода двоичного дерева: корень – левое поддерево
поддерево
левое поддерево – корень – правое поддерево
левое поддерево – правое поддерево – корень
Им соответствуют:
префиксная запись
инфиксная запись
постфиксная запись

Слайд 25

Примеры

- * + 5 2 3 + 1 / 2 3
5 +

Примеры - * + 5 2 3 + 1 / 2 3
2 * 3 – 1 + 2 / 3
5 2 + 3 * 1 2 3 / + -

Слайд 26

Упражнения

Предложить алгоритм вычисления выражений, заданных в постфиксной записи (указание: используйте стек)
Применение какого

Упражнения Предложить алгоритм вычисления выражений, заданных в постфиксной записи (указание: используйте стек)
из обходов позволяет вывести элементы дерева поиска в отсортированном порядке?

Слайд 27

Очередь с приоритетами

Структура данных с двумя операциями:
добавить элемент и назначить ему приоритет
извлечь

Очередь с приоритетами Структура данных с двумя операциями: добавить элемент и назначить
элемент с наименьшим приоритетом
Как с помощью такой структуры данных реализовать стек? Обычную очередь?

Слайд 28

Простая реализация

Храним массив записей
element = record
data, priority : integer;
end;
Добавление – в конец

Простая реализация Храним массив записей element = record data, priority : integer;
массива (O(1))
Извлечение – просматриваем весь массив и ищем минимум (O(n))

Слайд 29

Структура данных «куча»

Полное двоичное дерево
Хранится в массиве:
Левый ребенок – 2 * i
Правый

Структура данных «куча» Полное двоичное дерево Хранится в массиве: Левый ребенок –
ребенок – 2 * i + 1
Родитель – i div 2
Основное свойство: heap[i] > heap[i div 2]
Высота есть O(logn)
Можно рассматривать кучу для максимума, а для минимума
Иногда эту структуру данных называют «пирамида»

Слайд 30

Пример

Пример

Слайд 31

Добавление элемента (1)

Добавим в конец массива
После этого основное свойство кучи может быть

Добавление элемента (1) Добавим в конец массива После этого основное свойство кучи может быть нарушено
нарушено

Слайд 32

Добавление элемента (2)

Необходимо восстановить нарушенное основное свойство
Для этого выполним просеивание вверх –

Добавление элемента (2) Необходимо восстановить нарушенное основное свойство Для этого выполним просеивание
будем поднимать элемент вверх до тех пор, пока свойство не будет выполнено

Слайд 33

Добавление элемента (3)

Число обменов не превышает высоту дерева – то есть O(logn)

Добавление элемента (3) Число обменов не превышает высоту дерева – то есть O(logn)

Слайд 34

Добавление элемента (4)

procedure add(x : integer);
begin
inc(size);
heap[size] := x;
i := size;
while (i >

Добавление элемента (4) procedure add(x : integer); begin inc(size); heap[size] := x;
1) and do begin
if (heap[i div 2] > heap[i]) then begin
swap(heap[i], heap[i div 2]);
end else begin
break;
end;
end;
end;

Слайд 35

Извлечение элемента (1)

Минимальный элемент находится на вершине кучи
Поменяем его местами с последним

Извлечение элемента (1) Минимальный элемент находится на вершине кучи Поменяем его местами
и удалим
После этого может нарушиться основное свойство

Слайд 36

Извлечение элемента (2)

Было

Стало

Необходимо поместить верхний элемент на его место!

Извлечение элемента (2) Было Стало Необходимо поместить верхний элемент на его место!

Слайд 37

Извлечение элемента (3)

Извлечение элемента (3)

Слайд 38

Извлечение элемента (4)

Извлечение элемента (4)

Слайд 39

Извлечение элемента (5)

Извлечение элемента (5)

Слайд 40

Извлечение элемента (5)

function remove() : integer;
begin
result := heap[1];
heap[1] := heap[size];
dec(size);
i := 1;
while

Извлечение элемента (5) function remove() : integer; begin result := heap[1]; heap[1]
(2 * i <= size) and do begin
min := i;
if (heap[2 * i] < heap[min]) then begin
min := 2 * i;
end;
if (2 * i + 1 <= size) and
(heap[2 * i + 1] <= heap[min]) then begin
min := 2 * i + 1;
end;
if (min = i) then begin
break;
end;
swap(heap[i], heap[min]);
i := min;
end;
end;

Слайд 41

Сортировка с помощью кучи

Можно добавить все элементы массива в кучу, а потом

Сортировка с помощью кучи Можно добавить все элементы массива в кучу, а
по очереди извлечь
Требуется дополнительный массив – на хранение кучи
Чтобы сделать без дополнительной памяти надо:
Научиться преобразовывать массив в кучу
Научиться преобразовывать кучу в упорядоченный массив

Слайд 42

Построение кучи из массива

for i := n div 2 downto 1 do

Построение кучи из массива for i := n div 2 downto 1
begin
siftDown(i);
end;
procedure siftDown(i : integer);
begin
while (2 * i <= size) and do begin
min := i;
if (heap[2 * i] < heap[min]) then begin
min := 2 * i;
end;
if (2 * i + 1 <= size) and
(heap[2 * i + 1] <= heap[min]) then begin
min := 2 * i + 1;
end;
if (min = i) then begin
break;
end;
swap(heap[i], heap[min]);
i := min;
end;
end;

Слайд 43

Время работы

Грубая оценка – O(nlogn)
Более точная оценка – O(n)

Время работы Грубая оценка – O(nlogn) Более точная оценка – O(n)

Слайд 44

Построение упорядоченного массива из кучи

for j := 1 to n do begin
swap(heap[1],

Построение упорядоченного массива из кучи for j := 1 to n do
heap[size]);
dec(size);
i := 1;
while (2 * i <= size) and do begin
min := i;
if (heap[2 * i] < heap[min]) then begin
min := 2 * i;
end;
if (2 * i + 1 <= size) and
(heap[2 * i + 1] <= heap[min]) then begin
min := 2 * i + 1;
end;
if (min = i) then begin
break;
end;
swap(heap[i], heap[min]);
i := min;
end;
end;

Слайд 45

Время работы heapSort

Общее время работы heapSort есть O(n + nlogn) = O(nlogn)

Время работы heapSort Общее время работы heapSort есть O(n + nlogn) = O(nlogn)

Слайд 46

Выводы

Двоичные деревья поиска позволяют быстро выполнять словарные операции при условии, что у

Выводы Двоичные деревья поиска позволяют быстро выполнять словарные операции при условии, что
них небольшая высота
С помощью двоичных деревьев можно представлять математические выражения
Очередь с приоритетами реализуется на основе структуры данных «куча»
Имя файла: Двоичные-деревья-поиска.-Очередь-с-приоритетами.pptx
Количество просмотров: 289
Количество скачиваний: 2