Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Методичка.doc
Скачиваний:
275
Добавлен:
13.02.2015
Размер:
6.31 Mб
Скачать

Раздел 15. Некоторые классы графов Тема 15.1.Деревья

Определение:Деревомназывается связный, ориентированный граф без петель и кратных рёбер, не содержащий в себе циклов, удовлетворяющий следующим условиям:

  • имеется в точности один узел, называемый корнем, в который не входит ни одно ребро;

  • в каждый узел, кроме корня, входит ровно одно ребро;

  • из корня к каждому узлу идёт путь (который, как легко показать единственный).

Деревья являются простейшим видом связных графов. Любое дерево с вершинами содержитребро. Число различных деревьев, которые можно построить навершинах равно.

Определение:Дерево с одной выделенной вершиной называетсякорневым деревом.

Определение:Ориентированный граф, состоящий из нескольких деревьев, называетсялесом.

Определение: Пусть– дерево. Если дуга, тоназываетсяотцомузла, асыномузла.

Определение: Если есть путь изв, тоназываетсяпредкомузла, апотомкомузла.

Определение: Узел без потомков называетсялистом.

Определение: Узели его потомки вместе образуют поддерево леса, и узелназывается корнем этого поддерева.

Определение:Глубинаузлав дереве – это длина пути из корня в.

Определение:Высотаузла в дереве – это длина самого длинного пути из этого узла в какой-нибудь лист.

Определение: Высотой дерева называется высота его корня.

Пример 15.1:

Глубина узла , в данном примере, = 1, а его высота = 2. Высота дерева = 3.

Определение:Упорядоченным деревомназывается дерево, в котором множество сыновей каждого узла упорядоченно. При изображении упорядоченного дерева, как правило, считается, что множество сыновей каждого узла упорядоченно слева направо.

Определение:Бинарнымдеревомназывается такое упорядоченное дерево, что:

  • каждый сын произвольного узла идентифицируется либо как левый сын, либо как правый сын;

  • каждый узел имеет не более одного левого и не более одного правого сына.

Обратите внимание, что бинарное дерево не является частным случаем дерева, это совершенно иное, хотя и тесно связанное понятие.

Пример 15.2:

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

Определение: Бинарное дерево называетсяполным, если для некоторого целого числакаждый узел, глубины меньшейимеет как левого, так и правого сына, и каждый узел глубиныявляется листом.

Полное дерево глубины имеетузлов.

Очень часто используются алгоритмы, которые проходят дерево (посещают каждый его узел) в некотором порядке. Известно несколько способов сделать это. Мы рассмотрим три широко известных способа: прохождение дерева в прямом порядке, обратном порядке и внутреннем.

Будем считать, что – дерево с корнеми сыновьямипри. Приэто дерево состоит из единственного узла.

Прохождение дерева в прямом порядке:

  • посетить корень ;

  • посетить слева на право поддеревья с корнями .

Пример 15.3:

Прохождение дерева в обратном порядке:

  • посетить слева направо поддеревья с корнями ;

  • посетить корень .

Пример 15.4:

Прохождение дерева во внутреннем порядке:

  • посетить левое поддерево корня (если оно существует);

  • посетить корень;

  • посетить правое поддерево корня (если оно существует).

Пример 15.5:

Прежде чем дать описание одного из этих алгоритмов на некотором более формальном языке, поговорим о способах задания и хранения и бинарных деревьев. Бинарные деревья, как правило, хранятся посредством двух массивов ЛЕВЫЙСЫН и ПРАВЫЙСЫН, где номер элемента массива – это номер узла, а значение этого элемента – номер левого или правого узла – сына. Если элемент - сын отсутствует, то значение равно 0.

Теперь опишем алгоритм нумерации узлов двоичного дерева в соответствии с внутренним порядком. Для этого будем пользоваться неким подобием языка программирования, специально предназначенного для прозрачного и понятного описания алгоритмов.

Вход: Двоичное дерево, представленное массивами ЛЕВЫЙСЫН и ПРАВЫЙСЫН.

Выход: Массив, называемый НОМЕР, такой, что НОМЕР[i] – номер i - того узла во внутреннем порядке.

Метод: Будем использовать глобальную переменную СЧЕТ, значение которой – номер очередного узла в соответствии с внутренним порядком. Начальное значение переменной СЧЕТ = 1.

Программа выглядит так:

begin

СЧЕТ 1

ВНУТРПОРЯДОК(КОРЕНЬ)

End

Procedure ВНУТРПОРЯДОК(УЗЕЛ)

Begin

if ЛЕВЫЙСЫН[УЗЕЛ]0 then

ВНУТРПОРЯДОК(ЛЕВЫЙСЫН[УЗЕЛ]);

НОМЕР[УЗЕЛ] СЧЕТ;

СЧЕТ СЧЕТ+1

if ПРАВЫЙСЫН[УЗЕЛ]0 then

ВНУТРПОРЯДОК(ПРАВЫЙСЫН[УЗЕЛ]);

End

Такая процедура, которая явно или неявно вызывает сама себя, называется рекурсивной. Применение рекурсии часто даёт возможность давать более прозрачное и сжатое описание алгоритма, чем это же можно было бы сделать, не используя рекурсию. Если бы приведённый алгоритм не был записан рекурсивно, надо было бы строить явный механизм для прохождения дерева. Двигаться вниз по дереву нетрудно, но чтобы обеспечить возможность вернуться к предку, надо запомнить всех предков в стеке, а операторы работы со стеком усложнили бы алгоритм, лишив его наглядности.