![]() Главная страница Случайная страница КАТЕГОРИИ: АвтомобилиАстрономияБиологияГеографияДом и садДругие языкиДругоеИнформатикаИсторияКультураЛитератураЛогикаМатематикаМедицинаМеталлургияМеханикаОбразованиеОхрана трудаПедагогикаПолитикаПравоПсихологияРелигияРиторикаСоциологияСпортСтроительствоТехнологияТуризмФизикаФилософияФинансыХимияЧерчениеЭкологияЭкономикаЭлектроника |
Часть 5. Алгоритмы на графах. сматриваемом случае используется реализация DenseGRAPH графа в виде матрицы смежности из раздела 17.3
Указанные рисунки служат иллюстрацией динамики рекурсивного метода DFS и его соответствия с исследованием лабиринта методом Тремо. Во-первых, вектор, проиндексированный вершинами, соответствует освещению перекрестков: когда мы обнаруживаем ребро, ведущее к вершине, на которой мы только что побывали (т.е. " видим свет в конце коридора"), мы не производим рекурсивных вызовов, дабы следовать вдоль этого ребра (т.е. идти вдоль этого коридора). Во-вторых, механизм вызова-возврата функции этой программы является аналогом нити в лабиринте: когда мы обработаем все ребра, инцидентные некоторой вершине (исследуем все коридоры, исходящие от соответствующего перекрестка), мы возвращаемся (в обоих смыслах этого слова). Программа 18.1. Поиск в глубину связной компоненты_____________________________ Класс DFS соответствует исследованию Тремо. Конструктор помечает как посещенные все вершины той же связной компоненты, в которой содержится v, через вызов рекурсивной функции searchC, которая обходит все вершины, смежные с v, подвергая их всех проверке, и вызывая саму себя для каждого ребра, которое ведет из v в непомеченную вершину. Клиентские программы могут воспользоваться функцией count для вычисления числа посещенных вершин и перегруженным оператором [], чтобы узнать, в какой последовательности алгоритм поиска посещал вершины. #include < vector> template < class Graph> class cDFS { int cnt; const Graph & G; vector < int> ord; void searchC(int v) { ord[v] = cnt++; typename Graph:: adjIterator A(G, v); for (int t = A.beg();! A.end(); t = A.nxt()) if (ord[t] == -1) searchC(t); } public: cDFS (const Graph & G, int v = 0): G{G), cnt(0), ord(G.V(), -1) { searchC(v); } int count() const { return cnt; } int operator[](int v) const { return ord[v]; } };
Глава 18. Поиск на графе
РИСУНОК 18.6. ПОИСК В ГЛУБИНУ Данные диаграммы представляют собой графическое представление процесса, изображенного на рис. 18.5, на котором показано дерево рекурсивных вызовов поиска в глубину в его развитии. Ребра графа, выделенные толстыми жирными линиями, соответствуют ребрам в дереве DFS, показанном справа от каждой диаграммы графа. Заштрихованные ребра — это ребра, намеченные для включения в дерево на следующих шагах. На ранних стадиях (слева) выполнения алгоритма дерево растет вниз в виде прямой линии, что соответствует рекурсивным вызовам для вершин О, 2, 6 и 4 (слева). Затем мы выполняем рекурсивные вызовы для вершины 3, затем для вершины 5 (справа, две верхних диаграммы), после чего возвращаемся из этих вызовов, чтобы сделать очередной рекурсивный вызов для вершины 7 из вершины 4 (справа, вторая снизу) и для 1 из 7 (справа снизу). Часть 5. Алгоритмы на графах
вершину w из v в первый раз. Если бы мы закрывали эти двери во время входа и открывали во время выхода (обозначив соответствующий коридор за счет протягивания вдоль него нити), то в этом случае мы бы имели точное соответствие между поиском в глубину и исследованием Тремо. На рис. 18.6 показано дерево рекурсивных вызовов, соответствующее рис. 18.5, в динамике его развития. Это дерево рекурсивных вызовов, известное как дерево DFS, обеспечивает структурное описание процесса поиска. Как следует из раздела 18.4, дерево DFS, если оно наращивается должным образом, представляет собой полное описание динамики поиска в дополнение к только что описанной структуре вызовов.
Порядок обхода вершин зависит не только от графа, но и от его представления и реализации АТД. Например, на рис. 18.7 показана динамика поиска при использовании класса SparseMultiGRAPH из раздела 17.4 для реализации представления графа в виде списков смежных вершин. В случае представления графа в виде матрицы смежности мы проводим исследование ребер, инцидентных каждой вершине, в цифровой последовательности. Что же касается представления графа в виде списков смежных вершин, то мы проводим их исследование в том порядке, в каком они выступают в списках. Это различие приводит к совершенно другой динамике рекурсивных поисков, поскольку отличаются последовательности, в которых ребра появляются в списках (что имеет место, например, когда построение одного и того же графа производится путем вставки ребер в различном порядке). Обратите внимание на то обстоятельство, что наличие параллельных ребер не име-
|