Основы программирования. Лекции
Оглавление
Раздел 1. Введение в программирование 5
Тема 1.1. Основы алгоритмизации 5
Лекция№1 Алгоритмы. Свойства и способы описания линейных алгоритмов. 5
Лекция№2 Составные команды (следования, ветвления, цикла) 11
Лекция№3 Команда присваивания. Заголовок алгоритма 16
Лекция№4 Табличные величины, виды таблиц 21
Лекция№5 Вспомогательные алгоритмы Тестирование ПО 27
Тема 1.2. Языки программирования 33
Лекция№6 Языки программирования, их классификация. 33
Лекция»7 Принципы построения ПО. Трансляторы 40
Лекция№ 8 Стадии разработки программного продукта. Этапы решения задач на ПК. 47
Лекция №9 Величины. Типы данных. 52
Раздел 2. Основные конструкции языков программирования 57
Тема 2.1. Операторы языка программирования 57
Лекция№10 Синтаксис языка. Арифметические выражения 57
Лекция№ 11 Ввод и вывод данных. 63
Лекция№ 12 Условный оператор. Оператор выбора. 67
Лекция№13 Циклы с пост и предусловием Цикл с параметром Вложенные циклы 72
Раздел 3. Структурное и модульное программирование 78
Тема 3.1. Процедуры и функции 78
Лекция№14 Общие сведения о подпрограммах Определение и вызов подпрограмм 78
Тема 3.2. Структуризация в программировании 84
Лекция№15 Основы и методы структурного программирования 84
Тема 3.3. Модульное программирование 88
Лекция№16 Понятие и структура модуля. Компиляция и компоновка программы 88
Раздел 4. Структуры данных 92
Тема 4.1. Массивы 92
Лекция №17 Понятие массива. Особенности программирования массивов 92
Тема 4.2. Строки 97
Лекция№18 Символьный и строковый типы. Объявление типов. 97
Лекция№19 Поиск, удаление, замена и добавление символов в строке. 103
Лекция№20 Операции со строками. Функции и процедуры. Решение задач. 108
Тема 4.3. Множества 114
Лекция№21 Понятие и объявление множества. Операции над множествами. 114
Тема 4.4. Записи 121
Лекция№22 Определение типа записи. Правила работы с записями. 121
Тема 4.5. Файлы 125
Лекция№23 Типы файлов. Файлы последовательного доступа. 125
Тема 4.6. Указатели 131
Лекция№ 25. Указатели и применение динамически распределяемой памяти. 131
Тема 4.6 137
Лекция№26. Структуры данных на основе указателей. 137
Раздел 5. Объектно-ориентированное программирование 143
Тема 5.1 Основные принципы объектно-ориентированного программирования (ООП) 143
Лекция№27. Базовые понятия ООП. Основные принципы ООП. 143
Тема 5.1 147
Лекция№28. Классы объектов. Компоненты и их свойства. 147
Тема 5.2 Интегрированная среда разработчика 152
Лекция№29. Интерфейс среды разработчика: основные окна, интегрированной среде 152
Тема 5.3 Этапы разработки приложения 158
Лекция№30. Проектирование, тестирование и отладка приложения. 158
Тема 5.4 Иерархия классов 164
Лекция№31. Классы объектно-ориентированного языка программирования. 164
Лекция№32. Наследование. Перегрузка методов 170
Тема 5.5. Визуальное событийно-управляемое программирование 174
Лекция№33. Основные компоненты интегрированной среды разработки 174
Тема 5.5. 179
Лекция№34. События компонентов Процедуры, определенные пользователем. 179
Тема 5.6. Разработка оконного приложения 183
Лекция№35. Создание интерфейса оконного приложения. Компиляция и запуск приложения. 183
Раздел 1. Введение в программирование 188
Тема 1.1. Основы алгоритмизации 188
Лекция№1 Алгоритмы. Свойства и способы описания линейных алгоритмов. 188
Лекция№2 Составные команды (следования, ветвления, цикла) 193
Лекция№3 Команда присваивания. Заголовок алгоритма 198
Лекция№4 Табличные величины, виды таблиц 203
Лекция№5 Вспомогательные алгоритмы Тестирование ПО 209
Тема 1.2. Языки программирования 215
Лекция№6 Языки программирования, их классификация. 215
Лекция»7 Принципы построения ПО. Трансляторы 221
Лекция№ 8 Стадии разработкт программного продукта. Этапы решения задач на ПК. 227
Лекция №9 Величины. Типы данных. 232
Раздел 2. Основные конструкции языков программирования 238
Тема 2.1. Операторы языка программирования 238
Лекция№10 Синтаксис языка. Арифметические выражения 238
Лекция№ 11 Ввод и вывод данных. 244
Лекция№ 12 Условный оператор. Оператор выбора. 247
Лекция№13 Циклы с пост и предусловием Цикл с параметром Вложенные циклы 253
Раздел 3. Структурное и модульное программирование 259
Тема 3.1. Процедуры и функции 259
Лекция№14 Общие сведения о подпрограммах Определение и вызов подпрограмм 259
Тема 3.2. Структуризация в программировании 265
Лекция№15 Основы и методы структурного программирования 265
Тема 3.3. Модульное программирование 269
Лекция№16 Понятие и структура модуля. Компиляция и компоновка программы 269
Раздел 4. Структуры данных 273
Тема 4.1. Массивы 273
Лекция №17 Понятие массива. Особенности программирования массивов 273
Тема 4.2. Строки 278
Лекция№18 Символьный и строковый типы. Объявление типов. 278
Лекция№19 Поиск, удаление, замена и добавление символов в строке. 284
Лекция№20 Операции со строками. Функции и процедуры. Решение задач. 290
Тема 4.3. Множества 296
Лекция№21 Понятие и объявление множества. Операции над множествами. 296
Тема 4.4. Записи 303
Лекция№22 Определение типа записи. Правила работы с записями. 303
Тема 4.5. Файлы 308
Лекция№23 Типы файлов. Файлы последовательного доступа. 308
Тема 4.6. Указатели 316
Лекция№ 25. Указатели и применение динамически распределяемой памяти. 316
Тема 4.6 322
Лекция№26. Структуры данных на основе указателей. 322
Раздел 5. Объектно-ориентированное программирование 329
Тема 5.1 Основные принципы объектно-ориентированного программирования (ООП) 329
Лекция№27. Базовые понятия ООП. Основные принципы ООП. 329
Тема 5.1 333
Лекция№28. Классы объектов. Компоненты и их свойства. 333
Тема 5.2 Интегрированная среда разработчика 339
Лекция№29. Интерфейс среды разработчика: основные окна, интегрированной среде 339
Тема 5.3 Этапы разработки приложения 345
Лекция№30. Проектирование, тестирование и отладка приложения. 345
Тема 5.4 Иерархия классов 351
Лекция№31. Классы объектно-ориентированного языка программирования. 351
Лекция№32. Наследование. Перегрузка методов 357
Тема 5.5. Визуальное событийно-управляемое программирование 362
Лекция№33. Основные компоненты интегрированной среды разработки 362
Тема 5.5. 367
Лекция№34. События компонентов Процедуры, определенные пользователем. 367
Тема 5.6. Разработка оконного приложения 372
Лекция№35. Создание интерфейса оконного приложения. Компиляция и запуск приложения. 372
Литература 376
Раздел 1. Введение в программирование
Тема 1.1. Основы алгоритмизации
Лекция№1 Алгоритмы. Свойства и способы описания линейных
алгоритмов.
План лекции
1. Определение алгоритма Свойства
2. Типы алгоритмов
4. Способы описания алгоритмов
5. Примеры
Средства наглядности: презентация
Алгоритм — набор команд (инструкций), определяющих порядок действий исполнителя для решения поставленной задачи,
достижения некоторого результата.
Исполнитель алгоритма устройство, имеющее систему команд. Идеальными исполнителями являются машины, роботы,
компьютеры.
. Свойства алгоритмов:
1.Дискретность
— алгоритм представляет собой последовательность элементарных шагов (команд исполнителя).
2. Детерминированность
— при одних и тех же входных данных получается один и тот же результат, т.е. любое действие должно быть строго и
недвусмысленно определено в каждом случае;
3. Завершаемость (конечность) —
каждый алгоритм завершается за конечное число шагов при любом наборе исходных данных.
4.Результативность — после выполнения алгоритма известно, что считать результатом, алгоритм должен приводить к
правильному результату для всех допустимых входных значениях.
5. Массовость — применимость алгоритма ко множеству исходных данных.
Порядок выполнения алгоритма:
- Действия в алгоритме выполняются в порядке их записи
- Нельзя менять местами никакие два действия алгоритма
- Нельзя не закончив одного действия переходить к следующему
Типы алгоритмов:
1.Линейные (описание действий, которые выполняются однократно в заданном порядке; имеет линейную структуру).
2.Разветвляющиеся (алгоритм, в котором в зависимости от условия выполняется либо одна, либо другая
последовательность действий);
3.Циклические (описание действий, которые должны повторятся указанное число раз или пока не выполнено заданное
условие);
4,Вспомогательные (алгоритм, который можно использовать в других алгоритмах, указав только его
Способы описания алгоритмов
Способы записи алгоритмов определяются исполнителем. Команды, которые может выполнять исполнитель наз. СИСТЕМОЙ
КОМАНД ИСПОЛНИТЕЛЯ (СКИ). Способы записи бывают:
на естественном языке;
на специальном (формальном) языке (псевдокод);
с помощью формул, рисунков, таблиц;
с помощью стандартных графических объектов (геометрических фигур)
Псевдокод представляет собой систему обозначений и правил, предназначенную для единообразной записи алгоритмов. Он
занимает промежуточное место между естественным и формальным языком.
Правила оформления блок-схем можно посмотреть в ГОСТ 19.701-90.
ГОСТ 19.002-80. Схемы алгоритмов и программ. Правила выполнения.
ГОСТ 19.003-80. Схемы алгоритмов и программ. Обозначения условные
графические.
Каждому действию алгоритма соответствует геометрическая фигура
(блочный символ). Перечень наиболее часто употребляемых символов
приведен в таблице ниже.
Название |
Символ (рисунок) |
Выполняемая функция (пояснение) |
1. Блок вычислений |
Выполняет вычислительное действие или группу действий |
|
2. Логический блок |
Выбор направления выполнения алгоритма в зависимости от условия |
|
3. Блоки ввода/вывода |
|
Ввод или вывод данных вне зависимости от физического носителя |
Вывод данных на печатающее устройство |
||
4. Начало/конец (вход/выход) |
Начало или конец программы, вход или выход в подпрограмму |
|
5. Предопределенный процесс |
Вычисления по стандартной или пользовательской подпрограмме |
|
6. Блок модификации |
Выполнение действий, изменяющих пункты алгоритма |
|
7. Соединитель |
Указание связи между прерванными линиями в пределах одной страницы |
|
8. Межстраничный соединитель |
Указание связи между частями схемы, расположенной на разных страницах |
Правила построения блок-схем:
1. Блок-схема выстраивается в одном направлении либо сверху вниз, либо слева направо
2. Все повороты соединительных линий выполняются под углом 90 градусов
Пример алгоритма:
Дано: x, y, z.
Найти max
Алгоритм 1. (словесное описание)
Алгоритм 2. (псевдокод)
Пример. А.2., представленный блоксхемой
Контрольные вопросы
1. Дайте определение алгоритма. В каких сферах человеческой
деятельности применимы алгоритмы?
2. Какие свойства алгоритмов вам известны? Объясните на примере
разработанных вами алгоритмов суть этих принципов.
3. Какие существуют формы записи алгоритмов? Опишите их
достоинства и недостатки. В каких случаях они применяются?
4.Перечислите основные правила составления алгоритмов.
.
Лекция№2 Составные команды (следования, ветвления, цикла)
План лекции
1.Определение команды (оператора)
2.Команда следования
3.Команда ветвления
4.Команда цикла
Элементарной структурной единицей любого алгоритма является простая команда (оператор), обозначающая
один элементарный шаг переработки или отображения информации. Простая команда на языке схем алгоритма изображается в
виде функционального блока, имеющего один вход и один выход.
Из простых команд и проверки условий образуются составные команды, имеющие более
сложную структуру. Рассмотрим основные типы составных команд алгоритма.
Команда следования
Эта команда образуется из последовательности команд, следующих одна за другой. Под действием понимается либо простая,
либо составная команда. Эти команды могут записываться либо в строчку, либо в столбец — одна под одной.
Наличие скобок позволяет рассматривать команду следования как единое действие, распадающееся на последовательность
более простых действий.
Команда ветвления
б) неполное ветвление а) полное ветвление
С помощью команды ветвления (развилки) осуществляется выбор одного из двух возможных действий в
зависимости от условия.
Действия, указанные после служебных слов то и иначе, могут быть простыми или составными командами.
При исполнении команды ветвления выполняется только одно из действий: если условие соблюдено, то выполняется
действие 1, в противном случае — действие 2.
В том случае, когда условие соблюдено, продолжение исполнения алгоритма происходит по стрелке «+», в
противном случае — по стрелке «—».
Команда ветвления может использоваться в сокращенной форме (коррекция), когда в случае
несоблюдения условия никакое действие не выполняется.
Циклический алгоритм – Команда повторения (цикл) — описание действий, которые должны повторяться
указанное число раз или пока не выполнено заданное условие.
Большинство алгоритмов содержат серии многократно повторяемых команд. Если такие команды записывать в виде составной
команды следования, то каждую повторяемую команду пришлось бы выписать ровно столько раз, сколько раз она
повторяется.. Поэтому для обозначения многократно повторяемых действий используют специальную конструкцию,
называемую циклом.
Цикл — управляющая структура, организующая многократное выполнение указанного действия.
Составная команда цикла, называемая также командой повторения, содержит
условие. состоит из двух частей: условия цикла, которое используется для
определения количества повторений, и тела цикла(перечень повторяющихся
действий).
У любого цикла есть параметр. Параметр цикла – это
переменная, которая изменяется в теле цикла, а также участвует в условии его окончания.
Циклические алгоритмы бывают двух типов:
Циклы со счетчиком(параметром), в которых какие-то действия выполняются
определенное число раз; (безусловные циклы или арифметические циклы )
Циклы с условием, в которых тело цикла
выполняется, в зависимости от какого-либо условия. Различают циклы с предусловием и постусловием.
Схема цикла с предусловием
Под действием, как и прежде, понимается простая или составная команда. Исполнение такой команды повторения состоит в
том, что сначала проверяется условие (отсюда и название — цикл с предусловием), и если оно
соблюдено, то выполняется команда, записанная после служебного слова повторять. После этого снова
проверяется условие. Выполнение цикла завершается, когда условие перестает соблюдаться. Для этого необходимо, чтобы
команда, выполняемая в цикле, влияла на условие. (этот тип цикла называют также циклом «пока»). . Можно сказать что
условие цикла «пока» — это условие входа в цикл. В частном случае может оказаться что действие не
выполнялось ни разу. Условие цикла необходимо подобрать так, чтобы действия, выполняемые в цикле, не привели к
нарушению его истинности, иначе произойдет зацикливание. Зацикливание — бесконечное повторение
выполняемых действий.
Схема цикла с постусловием
Команда повторения с постусловием выполняется аналогично, только условие проверяется после
выполнения команды, а повторение выполнения команды происходит в том случае, когда условие не соблюдено, т. е.
повторение производится до соблюдения условия (поэтому этот тип цикла называют также циклом «до»). Таким образом,
тело цикла будет реализовано хотя бы один раз. Если условие не выполняется, то происходит возврат к выполнению
действий. Если условие истинно, то осуществляется выход из цикла. Для предотвращения зацикливания необходимо
предусмотреть действия, приводящие к истинности условия.
Циклы со счетчиком(параметром) используют, когда заранее известно, какое число повторений тела
цикла необходимо выполнить. Например, на уроке физкультуры вы должны пробежать некоторое количество кругов вокруг
стадиона.
В общем случае схема циклического алгоритма с условием будет выглядеть так:
Пока
— условие повторять — действие..Назовите правила с
Для создания циклов со счётчиком ( параметром) необходимо использовать правила:
Параметр цикла, его начальное и конечное значения и шаг должны быть одного типа
Запрещено изменять в теле цикла значения начальное, текущее и конечное для параметра
Запрещено входить в цикл минуя блок модификации
Если начальное значение больше конечного, то шаг — число отрицательное
После выхода из цикла значение переменной параметра неопределенно и не может использоваться в дальнейших
вычислениях
Из цикла можно выйти не закончив его, тогда переменная параметр сохраняет свое последнее значение
Контрольные вопросы
1 Дайте определение команды (оператора).
2. 3.Объясните роль условия в команде ветвления
3.Чтотакое «тело цикла» ?
4.Назовите правила создания цикла со счетчиком.
Лекция№3 Команда присваивания. Заголовок
алгоритма
План лекции
1.Формат команды присваивания
2.Свойства присваивания
3.Пример
4. Заголовок алгоритма.
Команда присваивания — команда исполнителя, в результате которой переменная получает новое значение. Формат
команды:
<имя переменной>:=<выражение>
Исполнение команды присваивания происходит в таком порядке: сначала вычисляется <выражение>, затем, полученное
значение присваивается переменной.
Пример.
Например, запись A:=B+5 читается так: «переменной A присвоить значение выражения B плюс 5».
Знаки присваивания «:=» и равенства «=» — разные знаки:
знак «=» означает равенство двух величин, записанных по обе стороны от этого знака;
знак «:=» предписывает выполнение операции присваивания.
Например, запись A:=A+1 выражает не равенство значений A и A+1, а указание увеличить значение переменной A на
единицу.
При выполнении команды присваивания сначала вычисляется значение выражения, стоящего справа от знака
«:=», затем результат присваивается переменной, стоящей слева от знака «:=». При
этом тип выражения должен быть совместим с типом соответствующей переменной.
Свойства присваивания:
1. пока переменной не присвоено значение, она остаётся неопределённой;
2. значение, присвоенное переменной, сохраняется в ней вплоть до выполнения следующего присваивания этой переменной
нового значения;
3. если мы присваиваем некоторой переменной очередное значение, то предыдущее её значение теряется безвозвратно.
В дальнейшем будет предполагаться, что исполнителем алгоритмов работы с величинами является
компьютер. Любой алгоритм может быть построен из команд присваивания,
ввода, вывода, ветвления и цикла.
Команда присваивания означает следующие действия, выполняемые компьютером: 1) вычисляется выражение; 2) полученное
значение присваивается переменной.
В блок-схемах команды присваивания изображаются в прямоугольниках. Такой блок называется вычислительным.
При описании алгоритмов не обязательно соблюдать строгие правила записи выражений, это можно делать в обычной
математической форме, так как это еще не язык программирования со строгим синтаксисом. В рассматриваемом алгоритме
имеется команда ввода: ввод a, b, c, d. В блок-схемах команда ввода записывается в параллелограмме — блоке
ввода-вывода. При выполнении этой команды процессор прерывает работу и ожидает действий пользователя. Пользователь
должен набрать на устройстве ввода (клавиатуре) значения вводимых переменных и нажать клавишу ввода. Значения
следует вводить в том же порядке, в каком эти переменные рас- положены в списке ввода. Обычно с помощью команды
ввода присваиваются значения исходных данных, а команда присваивания используется для получения промежуточных и
конечных величин. Полученные компьютером результаты решения задачи должны быть сообщены пользователю, для чего и
предназначена команда вывода: вывод m, n. С помощью этой команды результаты выводятся на экран или через устройство
печати на бумагу.
Пример
Составим алгоритм, в результате которого переменные A и B литерного типа обменяются своими значениями.
Решение вида A:=B B:=A неверно, так как после выполнения первой команды присваивания первоначальное значение
переменной A будет безвозвратно утеряно. Вторая команда присвоит переменной B текущее значение переменной A. В
результате обе переменные получат одно и то же значение.
Для решения исходной задачи введём промежуточную переменную M. . Тогда задачу обмена значениями можно решить
последовательным выполнением трех команд присваивания. Алгоритм обмена значениями переменных A и B запишем так:
Этой задаче аналогична следующая ситуация. Имеются два стакана: один — с молоком, другой — с водой. Требуется
произвести между ними обмен содержимым. Ясно, что в этом случае необходим третий стакан — пустой. Последовательность
действий при обмене будет такой: 1) перелить молоко из 1-го стакана в 3-й; 2) воду из 2-го стакана в 1-й; 3) молоко
из 3-го стакана во 2-й.
а) Имя алгоритма
При работе на ЭВМ принято давать имена, состоящие из не более 8 символов, без пробелов, на первом месте обязательно
буква, после имени состоящем из восьми символов через точку пишется расширитель имени, состоящем из не более трех
символов на первом месте обязательно буква. Например: KWUR.E, BASIC.COM,
LR4B.SC2 и т.д.
б) Список величин
(входные и выходные величины) с указанием их типа.
в) аргументы (входные данные) и результаты (выходные
данные).
Общий вид заголовка алгоритма работы с величинами таков:
алг имя алгоритма (список величин с
указанием типов)
~~
дано имена аргументов (имена входных величин)
~~~~
надо имена результатов (имена
выходных величин)
~~~~
нач
~~~
серия
кон
Как мы уже знаем, заголовок алгоритма описывает условие задачи, а тело алгоритма — ее решение. Чтобы записать
заголовок алгоритма, обычно достаточно внимательно изучить условие, не думая пока о решении.
При построении алгоритмов с аргументами важно точно определить количество аргументов и их типы. Для этого нужно
изучить условие задачи и выделить в нем ту информацию, которую необходимо задать, прежде чем приступать к решению.
Этой информации будут соответствовать аргументы алгоритма.
Например, в задаче «квадрат» такой дополнительной информацией была сторона квадрата, поэтому у алгоритма появился
один аргумент.
В общем случае переменным в условии задачи соответствуют аргументы в заголовке алгоритма.
Контрольные вопросы
1.Опишите порядок выполнения команды присваивания.
2.Назовите свойства присваивания.
3.Верно ли решение вида A:=B B:=A ? Почему?
4.Приведите пример заголовка алгоритма
Лекция№4 Табличные величины, виды таблиц
План лекции
1.Виды таблиц.
2. Примеры использования таблиц.
3. Табличные величины в алгоритмах.
Определение. Таблица – это упорядоченная последовательность величин одного типа, имеющая имя. Например;
Расписание звонков:
Таблица умножения Пифагора;
Поля игры “Морской бой”.
Из всего многообразия таблиц можно выделить простые и сложные.
Простые таблицы бывают линейные и прямоугольные. К типу линейных таблиц можно отнести расписание звонков. Таблица
умножения Пифагора относится к типу прямоугольных таблиц.
Сложные таблицы – это такие таблицы, которые состоят из простых.
Например, таблица “Расписание уроков на неделю”:
При решении задач человек очень часто пользуется таблицами. Таблицы бывают разными, но наиболее часто встречаются
линейные и прямоугольные. Эти два вида таблиц мы с вами и будем рассматривать.
И так, таблицы бывают линейные и прямоугольные. В линейной таблице только одна строка, в прямоугольной их несколько.
Каждой таблице дается свое название. Каждый элемент таблицы носит тоже название, что и сама таблица, различают же их
по номерам строк и столбцов, в которых они находятся.
Рассмотрим примеры использования таблиц в практической деятельности человека.
Пример 1. На метеостанции каждый час измеряется температуры воздуха и значения измерения записываются в таблицу:
Время измерения, ч |
… |
||||||
Температура, 0С |
15,5 |
… |
17,5 |
Эта линейная таблица содержит 24 элемента, занумерованные от 0 до 23. Второй элемент имеет значение 15,5, а нулевой
элемент – 17. Время измерения в таблице имеет значение номера столбца, в котором находятся показания
температуры.
Пример 2. На метеостанции вычисляют среднюю температуру воздуха каждые сутки и записывают в другую таблицу. Пусть нас
интересует средняя температура с 22 по 28 апреля:
Дата |
|||||||
Средняя температура, 0С |
15,5 |
17,5 |
Данная линейная таблица содержит семь элементов, занумерованных от 22 до 28.
Очевидно, что при хранении таблицы порядковые номера хранить нет необходимости: зная начало нумерации, можно путем
отсчета найти любой элемент таблицы. Кроме того, полезно знать и самый большой порядковый номер, так как это
позволяет определить заранее размер таблицы. Таким образом, чтобы указать, что некоторая величина является линейной
таблицей, нужно задать тип элементов таблицы, ее имя, начальный и конечный порядковые номера ее элементов.
В первом примере таблицу можно записать так: вещтаб температура [0:23 ], во втором – вещтаб средняя
температура [22:28].
Для удобства использования табличных величин в алгоритмах, их обычно обозначают одной буквой латинского алфавита.
Напримеры: вещтаб F[7:12], нат таб D[1:12], лит таб G[5:9].
Таблица F
-2,8 |
0,69 |
-23,87 |
F[7]:=-2,8; F[8]:=0; F[9]:=0,69; F[10]:=8; F[11]:=-23,87; F[12]:=11.
Таблица D
D[1]:=3; D[2]:=12; D[3]:=1; … ; D[12]:=100.
Таблица G
Петров |
Иванов |
Сидоров |
Волков |
Курочкин |
G[5]:= «Петров»; G[6]:= «Иванов»; … ; G[9]:= «Курочкин».
Рассмотрим примеры использования прямоугольных таблиц. Таблицы сложения и умножения однозначных чисел в различных
системах счисления. Эти таблицы имеют несколько строк, значит они прямоугольные.
Таблица S Таблица Р
+ |
× |
|||||||||
105 |
||||||||||
105 |
115 |
115 |
||||||||
105 |
115 |
125 |
115 |
145 |
225 |
|||||
105 |
115 |
125 |
135 |
135 |
225 |
315 |
наттаб S[1:4,1:4] наттаб Р[1:4,1:4]
При измерении температуры в течение месяца ежедневно каждый час данные можно вводить в таблицу:
время дата |
… |
||||||||
15,5 |
14,7 |
… |
17,5 |
||||||
16,5 |
… |
||||||||
15,5 |
14,5 |
14,3 |
… |
16,5 |
|||||
… |
.. |
… |
… |
… |
… |
… |
… |
… |
… |
15,5 |
14,7 |
… |
17,5 |
||||||
16,5 |
15,5 |
… |
вещтаб температура [1:31,0:23]
При указании нумерации в прямоугольных таблицах на первом месте пишутся строки, а на втором столбцы. Каждый элемент
таблицы носит тоже имя, что и сама таблица, различают же его по номеру строки и столбца, в которых он находится.
Например: S[2,4]:=115; P[3,4]:=225; температура[30,5]:=14,7.
Табличные величины удобно использовать в алгоритмах. Рассмотрим примеры таких алгоритмов.
Пример 1. |
алг таблица сложения (наттаб S[1:4,1:4]) рез S начнатi,j |
Пример 2 |
алг таблица умножения (наттаб Р[1:4,1:4]) рез Рначнат i,j |
Пример 3 |
алг таблицы сложения и умножения (наттаб S[1:4,1:4],наттаб Р[1:4,1:4 ) рез Р,S начнат i,j |
При сложении или умножении номера строки и столбца мы получаем число в десятичной системе счисления. 5 = 10, значит
необходимо определить количество пятерок в полученной сумме или произведении. Это количество определяется в
алгоритме «таблица умножения» с помощью команды повторения. В алгоритме «таблица сложения» учитывается тот факт, что
4+4=8, т.е. больше одной пятерки при сложении в сумме содержаться не может, поэтому достаточно использовать команду
ветвления. Можно в обоих алгоритмах использовать команду повторения. Более того, оба эти алгоритма можно объединить
в один .
Составим алгоритм для вычисления средней суточной температуры воздуха в течении месяца (Пример 4). Для удобства
записи назовем таблицу «температура» одной буквой «Т». Для сохранения значений средней температуры в течении месяца
создадим линейную таблицу на 31 элемент и назовем ее «С». Каждый элемент первой таблицы будет различаться по номеру
строки и столбца – Т[i,j], а каждый элемент второй таблицы будет различаться по номеру столбца – С[i].
Среднесуточную температуру будем вычислять по правилу вычисления среднего арифметического, и сохранять результат в
созданной нами линейной таблице.
Пример 4 |
алг среднесуточная температура (вещтаб Т[1:31,0:23], вещтаб С[1:31]) арг Т рез С начнат i,j |
Контрольные вопросы
1.Виды таблиц.
2. Примеры использования таблиц.
3. Табличные величины в алгоритмах
Лекция№5 Вспомогательные алгоритмы Тестирование ПО
План лекции
1.Вспомогательный алгоритм
2. Процедуры
3. Функции
4.Тестирование
5.Рекомендации по тестированию
5. Виды тестирования
Вспомогательный алгоритм представляет собой модуль, к которому можно многократно обращаться из
основного алгоритма. Использование вспомогательных алгоритмов может существенно уменьшить размер алгоритма и
упростить его разработку. Вспомогательный алгоритм многократно используется в основном алгоритме с различными
значениями некоторых входящих величин, называемых параметрами.
Для реализации вспомогательных алгоритмов служат подпрограммы или процедуры. Подпрограмма —
самостоятельный фрагмент программы, оформленный в виде, допускающем многократное обращение к нему из разных точек
программы. Обращение к подпрограмме — переход к выполнению подпрограммы с заданием информации,
необходимой для ее выполнения и возврата.
Существует два вида подпрограмм: процедуры и функции. Разница между ними состоит в том, что функция через свое имя
возвращает одно значение определенного типа и может, использоваться в выражениях наряду со встроенными функциями
.
Процедура
Алгоритмический язык |
Паскаль |
алг <имя процедуры> (<список параметров>) <операторы> кон |
procedure <имя процедуры> (<список параметров>); <описание> begin <операторы> end |
Процедура оформляется следующим образом: |
Вызов процедуры из основной программы производится оператором вызова процедуры:
<имя процедуры>(<список значеиий>).
В процедуру могут передаваться параметры, то есть некоторые переменные, которые могут использоваться внутри
процедуры.
Для того чтобы передать параметр по ссылке, в Паскале в описании формальных параметров в теле процедуры используется
ключевое слово var:
procedure SubTest(a,b:integer; var c:real, var d:integer);
здесь параметры а и b передаются по значению, а параметры с и d — по ссылке.
Функции
Функции по своей сути похожи на процедуры, но возвращают одно значение через свое имя.
Бейсик |
Паскаль |
FUNCTION <имя>(<параметры>) <операторы> END FUNCTION |
function <имя>(<параметры>):<тип результата>; <описания> begin <операторы> end |
Описание функции: |
Для того чтобы вернуть значение из функции, необходимо внутри тела функции переменной, имя которой совпадает с именем
функции, присвоить необходимое значение. Эту переменную не надо объявлять в области описания.
Вызов функции производится в выражениях и операторах подобно стандартным функциям языка:
<Переменная> := <Функция> (<Параметры>)
Реализация алгоритмов с помощью подпрограмм — процедур и функций — называется процедурным
программированием. В последние годы все большую популярность приобретают методы объектного и событийного
программирования.
Тестирование ПО
«Тестирование программ может использоваться для демонстрации наличия ошибок,
но оно никогда не покажет их отсутствие»Эдсгер Вибе Дейкстра
Необходимо различать понятия тестирование и отладка.
Тестирование направлено на выявление ошибок в программе, которая считается работающей. В то время как, отладкой называют процесс локализации и исправления
обнаруженных ошибок.
Программа подвергается постоянному тестированию на протяжении всего процесса работы над ней.
Основным условием для тестирования является наличие известных наборов исходных данных и соответствующих им
результатов выходных данных. Укажем некоторые рекомендации к тестированию программ
1) Все результаты тестирования необходимо строго фиксировать на бумаге, так чтобы их можно было воспроизвести после
того как будут внесены изменения в программу.
2) Необходимо тестировать граничные условия. Например, необходимо постоянно проверять, что тело цикла повторяется
нужное число раз, а условное выражение корректно разветвляет вычисления.
3) Необходимо тестировать пред- и постусловия. Другими словами, прежде чем использовать данные в вычислениях,
необходимо удостовериться в их корректности. Например, при вычислении частного двух чисел делитель не должен быть
равен 0; индекс массива не должен превышать допустимый диапазон.
4) Следует проверять коды возвратов функций. Например, если по каким-либо причинам вызов библиотечной функции
fopen(), осуществляющей открытие файла, завершилось неудачей, то дальнейшая работа с данным файлом должна быть
прервана.
Альфа-тестирование – тестирование готового продукта на специально созданных задачах.
Бета-тестирование – опробование бесплатной тестовой версии программного продукта на реальных задачах.
Если «альфа-» и «бета-тестирование» относятся к стадиям до выпуска продукта (а также, неявно, к объёму тестирующего
сообщества и ограничениям на методы тестирования), тестирование «белого ящика» и «чёрного ящика» имеет отношение к
способам, которыми тестировщик достигает цели..
При тестировании с использованием стратегии «белого ящика» (иногда
можно встретить термин «прозрачный ящик») полагают, что структура программного обеспечения известна, этим
и объясняется название стратегии. В данном случае тесты подбирают таким образом, чтобы пройти хотя бы один раз по
каждой ветви алгоритма.
При тестировании с использованием стратегии «черного ящика»
полагают, что структура программного обеспечения неизвестна, то есть программа рассматривается как «черный ящик»
при этом известны только наборы входных и выходных данных. В данном случае, тестовые данные формируют только на
основе спецификации программы, без учета информации о ее структуре. Регрессионное тестирование. После внесения
изменений в очередную версию программы, регрессионные тесты подтверждают, что сделанные изменения не повлияли на
работоспособность остальной функциональности приложения.
При тестировании сложных программных систем стратегия «белого ящика» не позволяет выявить пропущенный
маршрут, поэтому, возможно, что некоторые ошибки в программе останутся не обнаруженными. Также эта стратегия не
позволяет выявить ошибки, связанные с обрабатываемыми входными данными. В свою очередь, при использовании стратегии
«черного ящика» невозможно подобрать такие тестовые наборы, чтобы выполнить проверку всех возможных
комбинаций исходных данных. В этой связи, на практике, как правило, используют комбинации двух стратегий.
В заключение отметим, что даже после внедрения в эксплуатацию процесс тестирования не прекращается.
Контрольные вопросы
.Вспомогательный алгоритм
2. Процедуры
3. Функции
4.Тестирование
5.Рекомендации по тестированию
5. Виды тестирования
Тема 1.2. Языки программирования
Лекция№6 Языки программирования, их классификация.
План лекции
1. ЯП: определение, классификация.
2. История ЯП. Презентация
3. Достоинства/ недостатки ЯП.
4. Эволюция ЯП BASIC, Pascal
Средства наглядности: презентации
Язык программирования — это формализованный язык, который представляет собой совокупность алфавита,
правил написания конструкций (синтаксис) и правил толкования конструкций (семантика).
В настоящее время насчитывается несколько сотен языков программирования, рассчитанных на разные сферы применения ЭВМ,
т. е. на разные классы решаемых с помощью ЭВМ задач. Эти языки классифицируют по разным уровням, учитывая степень
зависимости языка от конкретной ЭВМ.
Общепринятой и строгой классификации языков программирования не существует. Поэтому в курсе представлена
классификация наиболее распространенных языков, сложившаяся исторически:
Языки программирования |
||
Низкого уровня |
Высокого уровня |
|
Машинный |
Машинно-зависимые |
Машинно-независимые |
Ассемблер |
Универсальные |
|
Автокод |
Проблемно-ориентированные |
|
Объектно-ориентированные |
||
Командные языки баз данных |
На самом нижнем уровне классификации находится машинный язык, т. е. внутренний язык ЭВМ, на котором в конечном итоге
представляется и исполняется программа.
Универсальные языки высокого уровня обеспечивают создание различных программ (задач), например
Алгол, Си, ПЛ/1 и т.д..
Проблемно-ориентированные языки создавались под какие-то конкретные классы задач, например, Фортран
— научные расчеты, Кобол — экономические расчеты, Лисп и Пролог — искусственный интеллект и т.д.
Объектно-ориентированные языки четвертого поколения (4GL — forth-generation
language) и программирование основаны на создании модели системы, как совокупности объектов и использует
следующие базовые понятия: класс, объект, событие, свойства объекта, метод обработки. Первым языком программирования
этой группы был — Симула-67. В настоящее время к этим языкам относятся — С++, Visual Basic, Java Script и
динамический HTML и другие современные языки программирования.
Командные языки баз данныхпредназначены длярасширения возможностей среды управления базами данных,
для создания собственных функций интерфейса — взаимодействия с пользователем.
История языков программирования.
Неструктурные языки (широко использовались 40-е годы)
Преимущества.
Оптимизация программы под аппаратную архитектуру.
Как следствие, обеспечение высокой эффективности вычислений.
Недостатки
Для каждого типа вычислительной машины должен был быть написан свой вариант исходного кода.
Применение: быстрые численные расчеты, создание драйверов устройств.
Примеры языков: Ассемблеры.
Директивные (структурные) языки (появились в 50-е годы)
Преимущества
Повторное использование ранее написанных блоков кода.
Высокая степень независимости программы от типа вычислительной машины.
Повышение эффективности труда разработчиков, в том числе и за счет абстрагирования от конкретных деталей аппаратного
обеспечения.
Недостатки
Некоторая потеря в скорости вычислений.
Применение: создание операционных систем и системных программ, разработка небольших пользовательских приложений,
научные расчеты.
Примеры языков: FORTRAN, C, Pascal.
Декларативные (функциональные и логические) языки (зародились в 60-е годы)
Программный код на декларативном языке программирования представляет собой описание действий, которые можно
осуществлять, а не последовательный набор команд.
Преимущества
Легче формализуется математическими средствами.
Как следствие, программы проще тестировать, т.е. проверять на наличие ошибок.
Высокая степень абстракции.
Недостатки
Снижение скорости работы программы.
Применение: доказательство теорем, возможность обработки разнородных данных.
Функциональные языки
Программу на функциональном языке можно представить как функцию с одним или несколькими аргументами.
Преимущества
Автоматическое динамическое распределение памяти компьютера для хранения данных.
Программист получает возможность абстрагироваться от представления данных и других рутинных операциях и
сосредоточиться на предметной области.
Недостатки
Нелинейная структура программы, следовательно, такое программирование сложно для понимания.
Относительно невысокая эффективность вычислений.
Применение: обработка рекурсивных структур данных, обработка символьной информации.
Примеры языков: Haskell.
Логические языки
Программа представляет собой совокупность правил или логических высказываний. Преимущества
Возможность откатов, т.е. возвращения к предыдущей подцели при отрицательном результате одного из вариантов в
процессе поиска решения. Это избавляет от необходимости поиска решения путем полного перебора вариантов.
Недостатки
Узкий класс решаемых задач.
Применение: эмуляция искусственного интеллекта, разработка экспертных систем.
Примеры языков: Prolog.
Объектно-ориентированные языки
Программа представляет собой описание объектов, их свойств (или атрибутов), классов и отношений между ними, способов
взаимодействия.
Преимущества
Смысловая близость к предметной области любой структуры и назначения. Механизм наследования свойств и методов
позволяет строить производные понятия на основе базовых, создавая тем самым модели предметной области.
Использование ранее созданных библиотек классов позволяет сэкономить время.
Полиморфизм обеспечивает гибкость и универсальность программного обеспечения.
Удобство разработки ПО группой лиц.
Недостатки
Сложность полной формализации реального мира создает в дальнейшем трудности тестирования созданного ПО.
Применение: разработка больших пользовательских приложений.
Примеры языков (большинство современных языков программирования поддерживают концепцию объектно-ориентированного
программирования): C++, Python.
Языки сценариев
Программа представляет собой совокупность возможных сценариев обработки данных. Выбор конкретного сценария зависит от
наступления того или иного события.
Преимущества
Основные достоинства данного класса языков программирования унаследованы от объектно-ориентированных языков.
Легкость использования с инструментальными средствами автоматизированного проектирования и быстрого создания ПО.
Недостатки
Сложность тестирования.
Большое количество вариантов, которые требуется предусмотреть.
Большая вероятность побочных эффектов.
Применение: интернет технологии. Примеры языков: JavaScript, Python, PHP
Эволюция ЯП BASIC, Pascal
Контрольные вопросы
1.ЯП: определение, классификация.
2. История ЯП. Презентация
3. Достоинства/ недостатки ЯП.
4. Эволюция ЯП BASIC, Pascal . Презентация
Лекция»7 Принципы построения ПО. Трансляторы
План лекции
1.Примеры ПО
2. Состав ПО
3. Разработка ПО
4.Трансляторы
5. Трансляция
Средства наглядности: презентация
Программное обеспечение (ПО)
– (Семакин И.Г., Хеннер Е.К.) это совокупность программ, хранящихся в долговременной памяти компьютера.
ПРИМЕРЫ ПРОГРАММНОГО ОБЕСПЕЧЕНИЯ
Системное ПО
Операционные системы: Windows, Linux и др.
Файловые менеджеры: Total Commander, FAR.
Антивирусные программы:DrWeb, Антивирус Касперского, Avast
Архиваторы:Winrar, Winzip, 7-zip
Программы обслуживания дисков: Defrag, Norton Disk Doctor
Инструментальное ПО
Системы программирования: Borland Delphi ,Borland C++ Builder, Microsoft Visual Basic , Microsoft Visual C++
Среды программирования, облегчающие разработчику создание программы: Microsoft Visual Studio.
В программном обеспечении компьютера есть необходимая часть, без которой на нем просто ничего не сделать. Она
называется системным ПО.
Кроме системного ПО в состав программного обеспечения компьютера входят еще прикладные
программы и системы программирования.
Состав прикладного программного обеспечения
Программы, с помощью которых пользователь может решать свои информационные задачи, не прибегая к программированию,
называются прикладными программами.
Программы общего назначения: текстовые и графические редакторы, системы управления базами данных (СУБД), табличные
процессоры, коммуникационные (сетевые) программы.
. Операционная система — это набор программ, управляющих оперативной памятью, процессором, внешними устройствами и
файлами, ведущих диалог с пользователем .
Системное ПО – это комплекс программ, обеспечивающих выполнение общих для всех программ технических задач,
взаимодействие с аппаратурой, диалог с пользователем.
Операционные оболочки — программы, выполняющие роль посредника между пользователем и ПО компьютера
Сервисные — Множество специальных программ обслуживающего (сервисного) характера.
Разрабо́тка ПО (англ. software development) — это род
деятельности) и процесс, направленный на создание и поддержание работоспособности, качества и надежности программного обеспечения, используя
технологии, методологию и практики из информатики, управления проектами, математики, инженерии и других областей знания[
Разработка ПО может быть разделена на несколько разделов. Это:
Требования к
ПО (извлечение, анализ, спецификация и ратификация )
Проектирование
ПО
Проектирование ПО средствами Автоматизированной
Разработки Программного Обеспечения (CASE) и стандарты формата описаний, такие как Унифицированный Язык
Моделирования (UML), используя различные подходы: проблемно-ориентированное
проектирование и т. д.
Программирование (создание
ПО с помощью языков программирования)
Тестирование программного
обеспечения( поиск и исправление ошибок в программе)
Сопровождение программного
обеспечения (программные системы часто имеют проблемы совместимости и переносимости, а
также нуждаются в последующих модификациях)
Управление конфигурацией
ПО (стандартизированный и структурированный методы)
Управление
разработкой ПО
Процесс разработки
ПО (основными парадигмами считаются agile или waterfall)
Инструменты разработки ПО, см. CASE: методика оценки сложности
системы, выбора средств разработки и применения программной системы.
Качество программного
обеспечения: методика оценки критериев качества программного продукта и требований к
надёжности.
Локализация программного обеспечения
— ветвь языковой
промышленности.
Опыт управления разработкой программ отражается в соответствующих руководствах, обычаях и стандартах. Если при
разработке используется несколько стандартов и нормативных документов, то имеет смысл составить профиль.
Участники процесса разработки ПО: пользователь,
заказчик, разработчик,
руководитель проекта, аналитик, тестировщик, поставщик.
Трансляторы
Чтобы вычислительная машина могла выполнить программу, написанную на каком-либо языке программирования, в её
программном обеспечении должна быть программа-транслятор для этого языка.
Интерпретатор переводит каждую команду программы с одновременным её выполнением и, если обнаруживает
ошибку, сообщает о ней и прекращает выполнение программы.
Компилятор переводит всю программу целиком и в конце работы выдаёт список ошибок, если они
обнаружены.
Трансляция — процесс перевода программы, написанной на алгоритмическом языке, на
машинный язык (в коды компьютера).
Транслятор — программа-переводчик.
Компиляция: процесс трансляции и выполнения программы четко разделены во времени.
Интерпретация: последовательно чередуются перевод группы инструкций языка в коды и их выполнение
Компиляция выгодней по времени выполнения и памяти.
Интерпретация удобней для организации диалоговых программ.
По типу выходных данных различают два основных вида трансляторов:
компилирующие окончательный выполнимый код;
компилирующие интерпретируемый код, для выполнения которого требуется дополнительное программное обеспечение.
Окончательным выполнимым кодом являются приложения, реализованные как EXE-файлы, DLL-библиотеки, COM-компоненты. К
интерпретируемому коду можно отнести байт-код JAVA-программ, выполняемый посредством виртуальной машины JVM.
Языки, формирующие окончательный выполнимый код, называются компилируемыми языками (С, C++, FORTRAN, Pascal). Языки,
реализующие интерпретируемый код, называются интерпретируемыми языками(Java, LISP, Perl, Prolog).
В процессе трансляции выполняется анализ исходной программы, а затем синтез выполнимой
формы данной программы. В зависимости от числа просмотров исходной программы, выполняемых компилятором, трансляторы
разделяются на однопроходные, двухпроходные и трансляторы, использующие более двух проходов.
К достоинствам однопроходного компилятора можно отнести высокую скорость компиляции, а к недостаткам — получение, как
правило, не самого эффективного кода.
Широкое распространение получили двухпроходные компиляторы. Они позволяют при первом проходе выполнить анализ
программы и построить информационные таблицы, используемые при втором проходе для формирования объектного кода.
Рис. 2.1. Основные этапы трансляции программы
На этапе лексического анализа выполняется выделение основных составляющих программы – лексем ( ключевые слова,
идентификаторы, символы операций, комментарии, пробелы и разделители). Составляется таблица символов, в которой
каждому идентификатору сопоставлен свой адрес, чтобы вместо конкретного значения (строки символов) использовать его
адрес в таблице символов. Требует применения сложных контекстно-зависимых алгоритмов.
На этапе синтаксического анализа выполняется разбор полученных лексем с целью получения семантически понятных
синтаксических единиц (выражения, объявление, оператор языка программирования, вызов функции), которые затем
обрабатываются семантическим анализатором. На этапе семантического анализа выполняется обработка синтаксических
единиц и создание промежуточного кода.
К наиболее общим задачам, решаемым семантическим анализатором, относятся:
обнаружение ошибок времени компиляции;
заполнение таблицы символов конкретными значениями;
замена макросов их определениями;
выполнение директив времени компиляции.
Макросом называется некоторый предварительно определенный код, который на этапе компиляции
вставляется в программу во всех местах указания вызова данного макроса.
На фазе синтеза программы производится: генерация кода; редактирование связей.
Процесс генерации кода состоит из преобразования промежуточного кода (или оптимизированного кода) в
объектный код. При этом в зависимости от языка программирования получаемый объектный код может быть представлен в
выполнимой форме или как объектный модуль, подлежащий дальнейшей обработке редактором связей.
Так, процесс генерации кода.
Редактор связей.
Контрольные вопросы
Примеры ПО
2. Состав ПО
3. Разработка ПО
4.Трансляторы
5. Трансляция Презентация
Тема1.2 ЯП
Лекция№ 8 Стадии разработки программного продукта. Этапы решения
задач на ПК.
План лекции
- Постановка задачи
- Анализ
- Проектирование
- Программирование
- Оформление документации
- Испытания Эксплуатация
- Этапы решения задачи на ЭВМ.
При разработке программного продукта можно выделить следующие стадии:
Стадия
предпроектных исследований и технического задания (постановка задачи)— определение
требований к программному продукту и осуществление формальной постановки задачи.
Часто её называют стадией постановки задачи., чтобы позволить программисту или аналитику однозначно определить, что
будет делать создаваемая программа. Постановка решаемой на компьютере задачи должна включать список ее входных
данных, список требуемых результатов и любые инструкции (правила), которых нужно следовать при решении задачи. В результате согласования между заказчиком и исполнителем всех перечисленных вопросов
составляют техническое задание (ТЗ) в соответствии с ГОСТ 19.201-78, которое служит основанием для дальнейшей
работы.
Стадия
технического предложения (анализ)— определение методов решения задачи.
Выполняется анализ задачи – это определение и детализация логического порядка действий, которые нужно выполнить над
исходными данными, чтобы получить требуемое решение.
Стадия эскизного
проектирования(проектирование) — разработка структуры программного продукта, выбор структур для
хранения данных, построение и оценка алгоритмов подпрограмм и определение особенностей взаимодействия программы с
вычислительной средой (другими программами, операционной системой и техническими средствами).
На этой стадии при использовании процедурного подхода сложные задачи разбиваются на подзадачи, для которых может
строиться своя модель и выбираться свой метод решения. При этом результаты одной подзадачи могут использоваться как
исходные данные в другой.
Целесообразно проверить правильность выбранных моделей и методов, выполнив их вручную для некоторых значений исходных
данных.
Одновременно с написанием алгоритма, необходимо точно определить тип и структуру обработанных данных. В одних случаях
данными могут быть несколько обычных чисел, в других организация данных будет более сложной.
При определении структуры данных с каждым объектом данных должно быть связано осмысленное имя или идентификатор. При
разработке программы идентификаторы будут связаны с расположением данных в памяти.
На данной стадии разрабатываются и оцениваются алгоритмы подпрограмм.
Разработка алгоритма состоит в пошаговом описании предлагаемого решения задачи. Каждый шаг должен быть описан в виде
кратких и точных операторов с использованием структурированного языка или псевдокода.
Каждой процедуре должно быть дано содержательное имя или идентификатор, для того, чтобы ее можно было вызвать по
имени как модуль из другой процедуры:
Пример 1
find_area (Найти площадь),
calc_balance (Вычислить_баланс)
prepare_fit (Подготовить_декларацию)
Разработка алгоритма завершается, когда каждый из операторов может быть записан непосредственно на языке
программирования.
Стадия
технического проектирования (программирование)— составление программы на выбранном языке программирования, ее отладка и
тестирование.
Разработанные алгоритмы реализуют, составляя по ним текст программы с использованием конкретного языка
программирования. Далее программу необходимо перевести в последовательность машинных команд (машинный код). Для
этого используется специальная программа — компилятор.
Если исходный текст программы не содержит ошибок, то компилятор создает исполнимый код программы. Далее программа
выполняется. При этом необходимо выявить ошибки выполнения..
Процесс локализации и исправления ошибок получил название отладки программы. При отладке
программы часто используют специальные программы – отладчики, которые позволяют выполнить любой фрагмент программы в
пошаговом режиме и проверить содержимое интересующих нас переменных.
Отлаженная программа подвергается тестированию. Тестирование — это процесс выполнения
программы при различных наборах данных с целью обнаружения логических ошибок. Для поиска логических ошибок также
можно использовать отладчик: по шагам отследить процесс получения результатов.
Стадия рабочего
проектирования — оформление документации.
Документация должна включать: описание постановки задачи; описание анализа задачи; описание определения структуры
данных;
описание алгоритма; текст программы с комментариями;
тесты с входными и выходными данными; записи о процедурах внедрения;
руководство пользователя; записи о всех модификациях.
Стадии испытаний
и внедрения в эксплуатацию — всестороннее тестирование программы и сопровождение при
внедрении в эксплуатацию.
Эргономические цели и показатели качества программного продукта
Эргономика включается в процессы разработки и тестирования программного продукта как часть системы качества.
Разработка пользовательского интерфейса (ПИ) ведется параллельно дизайну программного продукта в целом и в основном
предшествует его осуществлению.
Графический интерфейс пользователя ( англ. graphical user interface, GUI –
система средств для взаимодействия пользователя с компьютером.
С точки зрения эргономики, самое важное в
программе — создать такой пользовательский интерфейс, который сделает работу эффективной и производительной, а также
обеспечит удовлетворенность пользователя от работы с программой.
Этапы решения задачи на ЭВМ.
Работа по решению любой задачи с использованием компьютера включает в себя следующие шесть этапов: 1.постановка
задачи, 2.формализация задачи, 3.построение алгоритма, 4.составление программы на языке программирования, 5.отладка
и 6.тестирование программы.
6. Проведение расчетов и анализ полученных результатов.
Часто эту последовательность называют технологической цепочкой решения задачи на ЭВМ (непосредственно к
программированию из этого списка относятся п. 3 … 5).
На этапе постановки задачи следует четко определить, что дано и что требуется найти. Третий этап — это построение
алгоритма. Опытные программисты часто сразу пишут программы на определенном языке, не прибегая к каким-либо
специальным средствам описания алгоритмов (блок-схемам, псевдокодам), однако в учебных целях полезно сначала
использовать эти средства, а затем переводить полученный алгоритм на язык программирования. Первые три этапа — это
работа без компьютера. Последующие два этапа — это собственно программирование на определенном языке в определенной
системе программирования.
На последнем — шестом — этапе разработанная программа уже используется в практических целях.
Таким образом, программист должен уметь строить алгоритмы, знать языки программирования, уметь работать в
соответствующей системе программирования. Основой профессиональной грамотности программиста является развитое
алгоритмическое мышление.
Контрольные вопросы
- Постановка задачи
- Анализ
- Проектирование
- Программирование
- Оформление документации
- Испытания Экксплуатация
- Этапы решения задачи на ЭВМ.
Лекция №9 Величины. Типы данных.
План лекции
- Константы и переменные
- Простые и структурированные типы данных
3. Основные типы данных
Совокупность величин, с которыми работает компьютер, принято называть данными.
Всякая величина занимает свое определенное место в памяти ЭВМ ( ячейку памяти). тип. В алгоритмах и языках
программирования величины подразделяются на констан-ты и переменные. Константа — неизменная величина, и в алгоритме
она представляется собственным значением, например: 15, 34.7, k, True и др. Переменные величины могут изменять свои
значения в ходе выполнения программы и представляются в алгоритме символическими именами — идентификаторами,
например: X, S2, cod15 и др. Любые константы и переменные занимают ячейку памяти, а значения этих величин
определяются двоичным кодом в этой ячейке.
Типы величин: целые, вещественные, логические и символьные Это понятие является фундаментальным в
программировании.
Рис. 1.1. Уровни данных относительно программы
Типы величин характеризуются множеством допустимых значений, множеством допустимых операций, формой внутреннего
представления (табл. 1.1).
Типы констант определяются по контексту (т.е. по форме записи в тексте), а типы переменных устанавливаются в
описаниях переменных. По структуре данные подразделяются на простые и структурированные. Для простых величин,
называемых также скалярными, справедливо утверждение одна величина — одно значение, а для структурированных — одна
величина — множество значений.
таблица 1.1. Основные типы данных
К структурированным величинам относятся массивы, строки, множества и др.
Любые данные ТП характеризуются своими типами. Тип определяет:
- Формат представления данных в памяти компьютера
- Множество допустимых значений, принимаемое переменной или константой, принадлежащей к выбранному типу
- Множество допустимых операций применимых к этому типу
Тип переменной определяется при ее декларации. Одна из базовых концепций Паскаля заключается в жесткой проверке
соответствия типов в операциях присваивания.
Типы данных в языке ТП делятся на 5 основных классов:
- Простые типы
- Структурированные типы
- Ссылочные типы
- Процедурные типы
- Объектные типы
К простым типам относятся: целочисленные типы, логический тип, символьный тип, перечисляемый тип, интервальный тип,
вещественные типы.
Среди этих видов выделяют подмножества типов, отличных от вещественного, называемых
порядковым типом.
В TP имеется 5 предопределенных, целочисленных типов. Каждый тип обозначает определенное подмножество целых
чисел:
Тип |
Диапазон |
Формат |
Короткое целое shortint |
-128..127 |
8 бит со знаком |
Целое integer |
-32768..32767 |
16 бит со знаком |
Длинное целое longint |
-2147483648..2147483647 |
32 бита со знаком |
Длиной в байт byte |
0..255 |
8 бит без знака |
Длиной в слово word |
0..65535 |
16 бит без знака |
Верхнее граничное значение и нижнее граничное значение целочисленных типов задаются как константы и имеют
соответствующее имя.
В тексте программы данные целочисленных типов записываются в десятичном или
шестнадцатеричном формате и не должны содержать десятичные точки.
Над целочисленными данными возможно
выполнение операций сложения, вычитания и умножения, а также операций сравнения.
К логическим типам относятся данные типов Boolean, ByteBool, WordBool, LongBool.
Значением каждого данного
логического типа могут являться 2 значения: TRUE (1) и FALSE (0).
Для данных логического типа применимы только
две операции сравнения: равно и не равно.
Переменные типа Boolean и ByteBool занимают один байт;
переменная WordBool — 2 байта.
Интервальный тип данных определяется посредством задания подмножества значений
одного из ранее определенных типов. Можно использовать все простые типы, за исключением вещественного. При задании
диапазона указывается наименьшее и наибольшее значения, разделенные двумя точками. При этом оба значения обязательно
одного типа.
К вещественному типу относится подмножество вещественных чисел, представленных в формате с
плавающей точкой и фиксированным числом цифр.
В ТП имеется 5 видов вещественных типов:
Тип |
Диапазон |
Точность |
Формат |
Real (вещественное) |
2.9*10-39..1.7*1038 |
11-12 знаков |
6 байт |
Single (с одинарной точностью) |
1.5*10-45..3.4*1038 |
7-8 знаков |
4 байта |
Double (с двойной точностью) |
5.0*10-324..1.7*10308 |
15-16 знаков |
8 байт |
Extended (с повышенной точностью) |
3.4*10-4932..1.1*104932 |
19-20 знаков |
10 байт |
Comp (сложное) |
-9.2*1018..9.2*1018 |
19-20 знаков |
8 байт |
Действия над типами с одинарной, двойной, повышенной точностью и сложным типом могут выполняться только при
наличии числового сопроцессора. Поэтому считается что постоянно доступным является только тип Real.
Контрольные вопросы
- Константы и переменные
- Простые и структурированные типы данных
3. Основные типы данных
Раздел 2. Основные конструкции языков программирования
Тема 2.1. Операторы языка программирования
Лекция№10 Синтаксис языка. Арифметические выражения
План лекции
- Описание синтаксиса языка
- Синтаксические определения
- Арифметические выражения
Язык программирования — это формализованный язык, который представляет собой совокупность алфавита,
правил написания конструкций (синтаксис) и правил толкования конструкций (семантика).
Описание синтаксиса языка включает определение алфавита и
правил построения различных конструкций языка из символов алфавита и более простых конструкций. Для этого обычно
используют форму Бэкуса-Наура (БНФ) или синтаксические диаграммы.
СИМВОЛЫ языка-это основные неделимые знаки, в терминах которых пишутся все тексты на языке.
ЭЛЕМЕНТАРНЫЕ КОНСТРУКЦИИ -это минимальные единицы языка, имеющие
самостоятельный смысл. Они образуются из основных символов языка.
ВЫРАЖЕНИЕ в алгоритмическом языке состоит из элементарных конструкций и символов, оно задает правило вычисления
некоторого значения.
ОПЕРАТОР задает полное описание некоторого действия, которое необходимо выполнить. Для описания сложного действия
может потребоваться
группа операторов. В этом случае операторы объединяются в СОСТАВНОЙ
ОПЕРАТОР или БЛОК.
Действия, заданные операторами, выполняются над ДАННЫМИ. Предложения алгоритмического языка, в которых даются
сведения о типах данных,
называются ОПИСАНИЯМИ или неисполняемыми операторами.
Объединенная единым алгоритмом совокупность описаний и операторов образует ПРОГРАММУ на алгоритмическом языке.
В процессе изучения алгоритмического языка необходимо отличать алгоритмический язык от того языка, с помощью
которого осуществляется
описание изучаемого алгоритмического языка. Обычно изучаемый язык называют просто языком, а язык, в терминах которого
дается описание
изучаемого языка — МЕТАЯЗЫКОМ.
Синтаксические определения могут быть заданы формальными или неформальным способами. Существуют три формальных
способа:
-металингвистическая символика, называемая Бэкуса-Наура формулами;
-синтаксические диаграммы;
-скобочные конструкции.
бом.
О С Н О В Н Ы Е С И М В О Л Ы
Основные символы языка-буквы, цифры и специальные символы-состав-
ляют его алфавит. ТУРБО ПАСКАЛЬ включает следующий набор основных
символов:
1) 26 латинских строчных и 26 латинских прописных букв:
A B C D E F G H I J K L M N O P Q R S T U V W X Y Z
a b c d e f g h i j k l m n o p q r s t u v w x y z
2) _ подчеркивание
3) 10 цифр:
0 1 2 3 4 5 6 7 8 9
4) знаки операций:
+ — * / = <> < > <= >= := @
5) ограничители:
. , ‘ ( ) [ ] (. .) { } (* *) .. : ;
6) спецификаторы:
^ # $
7) служебные (зарезервированные) слова:
ABSOLUTE EXPORTS LIBRARY SET
ASSEMBLER EXTERNAL MOD SHL
AND FAR NAME SHR…
Кроме перечисленных, в набор основных символов входит пробел. Про-
белы нельзя использовать внутри сдвоенных символов и зарезервирован-
ных слов.
Э Л Е М Е Н Т А Р Н Ы Е К О Н С Т Р У К Ц И И
Элементарные конструкции языка ПАСКАЛЬ включают в себя имена, чис-
ла и строки.
Примеры.
0, 1, … 9
называют терминалами (лексемами) — это «конечные символы», т.е. по умолчанию известные в ЯП.
<цифра> так называемый нетерминал (нетерминальный символ).
Он определяется через терминалы, другие нетерминалы и самого себя. Причем в последнем случае правило задания
нетерминала называется рекурсивным (как определение нетерминала <идентификатор>) РБНФ (Расширенные БНФ)
[] — 0 или 1 повторение.
{} — 0 и более повторений
Пример.
Грамматика языка — совокупность всех синтаксических правил данного ЯП, обычно заданных в форме БНФ.
Грамматика не учитывает все виды ошибок, в ЯП формулируются дополнительные семантические правила.
Лексемы Паскаля
спецсимволы: := += *
ключевые слова (begin, end, if, for)
идентификаторы (a, b1)
константы (2, ‘ABC’, #5)
комментарии (3 вида)
{…}
(*…*) //…
Переменные и их описание
Основные сведения
Переменная — это ячейка памяти компьютера, имеющая имя и тип.
Тип определяет размер переменной и множество принимаемых ею значений.
В языке Pascal любая переменная перед использованием должна быть описана. Обычно переменные описываются в разделе
описаний.
Синтаксис в виде РБНФ
Пример секции описания переменных.
Арифметические выражения
Основные сведения
Каждое выражение имеет тип. Выражение называется арифметическим, если его тип — числовой. Выражение строится
посредством операций (унарных или бинарных) и операндов.
В арифметических выражениях если a и b — одного типа, то и a op b принадлежит к тому
же типу. Исключением является операция «/»:
a / b — вещественное.
Если a и b принадлежат к различным типам, то выражение принадлежит к «старшему» типу.
Например:
Стандартные функции
В арифметические выражения могут входить стандартные функции:
Порядок выполнения операций в арифметических выражениях
Операции с большим приоритетом выполняются первыми
Функции вычисляются до операций
Выражение в скобках вычисляется раньше
Операции с одинаковым приоритетом выполняются слева направо, если идут подряд. Операции div и mod для целых
x div y = x / y, округленное до ближайшего целого по направлению к нулю. Это результат от целочисленного деления. x
mod y = x ‐ (x div y) * y. Это остаток от целочисленного деления.
Пример использования
Целочисленные операции часто применяются для определения четности числа:
Контрольные вопросы
- Описание синтаксиса языка
- Синтаксические определения
- Арифметические выражения
Раздел 2 Тема 2.1
Лекция№ 11 Ввод и вывод данных.
План лекции
1.Синтаксис и семантика оператора ввода
2. Обработка ошибок ввода
3. Синтаксис и семантика оператора вывода
4. Форматы вывода
Ввод данных — это передача информации от внешних устройств в оперативную память. Вводятся, как правило, исходные
данные решаемой задачи. Вывод — обратный процесс, когда данные передаются из оперативной памяти на внешние
носители
(принтер, дисплей, магнитные устройства и т.д.). Результаты решения всякой задачи должны быть выведены на один из
этих носителей.
Оператор ввода
Синтаксис
Семантика
Происходит считывание данных с клавиатуры и запись их в переменные из <списка переменных> где <список
переменных>— это последовательность имен переменных, разделенных запятыми. Слово read переводится как
читать(Точнее говоря, Read — это оператор обращения к стандартной процедуре ввода.)
. Вводить данные нужно либо через пробел, либо по нажатию <Enter>, при этом программа не перейдет к
выполнению следующего оператора, пока не будут считаны все данные.
Другой вариант оператора ввода с клавиатуры имеет вид:
Здесь слово ReadLn означает read line — читать строку. Этот оператор отличается от Read только тем, что после
считывания последнего в списке значения для одного оператора ReadLn данные для следующего оператора будут
считываться с начала но
вой строки.
Имеются также стандартные функции ReadInteger, ReadReal, ReadlnInteger, ReadlnReal:
С процедурой ввода связан ряд ошибок времени выполнения (например, если переменная используется в качестве делителя,
и вводится 0, или, если должно быть получено целое число, а вводится ‘ABC’). Эти ошибки нужно уметь
обрабатывать.
Оператор try/except и обработка ошибок ввода
Операторы, которые могут получать ошибку, заключаются специальный охранный блок оператор try.
Синтаксис
Семантика
Если внутри блока try происходит ошибка выполнения, то все последующие операторы в блоке игнорируются, и
выполнение программы переходит к блоку except. По выходе из except программа продолжает
работу.
Если ошибки не происходит, то выполняются все операторы в блоке try, блок except не
выполняется, и программа продолжает работу.
Оператор вывода
Синтаксис
Семантика
Выражения в списке вычисляются, и их значения выводятся на экран. В случае writeln после вывода
осуществляется переход на новую строку.
Форматы вывода
Вывод с помощью write[ln]Format
Пример вывода с использованием форматной строки.
Будет выведено:
В форматной строке тоже можно использовать формат вывода.
{0, 10}: 10 — это ширина поля вывода
{0, 10:f3}: 3 — это количество знаков в дробной части для вещественного числа (показывает это спецификатор
f).
{0, 10:e3} — экспоненциальный формат.
Контрольные вопросы
1.Синтаксис и семантика оператора ввода
2. Обработка ошибок ввода
3. Синтаксис и семантика оператора вывода
4. Форматы вывода
Раздел 2 Тема 2.2 Условный оператор
Лекция№ 12 Условный оператор. Оператор выбора.
План лекции
1.Составной оператор
2 Синтаксис, семантика оператора IF
3.. Правила записи
4. Синтаксис, семантика оператора case
ОПЕРАТОР задает полное описание некоторого действия, которое необходимо выполнить. Для описания сложного действия
может потребоваться группа операторов. В этом случае операторы объединяются в СОСТАВНОЙ ОПЕРАТОР или БЛОК.
Пример 3. Упорядочить по возрастанию значения в двух переменных а, Ь:
В данном примере использован составной оператор — последовательность операторов, заключенная в фигурные скобки. В Си
фигурные скобки выполняют роль операторных скобок по аналогии с Begin, End в Паскале.
Обратите внимание на то, что перед закрывающей фигурной скобкой точку с запятой надо ставить обязательно, а после
скобки точка с запятой не ставится.
Условный
оператор IF
IF a=28 THEN WriteLn (f) ELSE k:=44
Переводится он так:
ЕСЛИ a=28 ТО печатай f ИНАЧЕ присвой переменной k значение 44.
Оператор if можно записывать и без части else. Например, if s<t then w:=a+1. Это означает, что если s<t, то
нужно выполнить оператор w:=a+1, в противном случае ничего не делать, а просто перейти к следующему оператору.
Правила
записи оператора IF
Вспомним правило расстановки точек с запятыми. Они применяются для того, чтобы отделять друг от друга операторы,
выполняющиеся друг за другом. Поэтому и после оператора if мы тоже ставили точку с запятой, если после него шел
какой-нибудь оператор. Перед end точку с запятой ставить не возбраняется, а
перед ELSE точку с запятой ставить запрещено.
IF условие THEN оператор [ ELSE оператор ]
Квадратные скобки здесь означают, что их содержимое можно писать, а можно и не
Примеры работы оператора if:
ФРАГМЕНТ ПРОГРАММЫ |
ЧТО НА ЭКРАНЕ |
a:=10; if a>2 then WriteLn (‘!!!’) else |
!!! |
a:=4; if a>5 then a:=a+10 else a:=a-1; WriteLn (a) |
3 |
s:=6; if s-8<0 then s:=s+10; WriteLn (s) |
16 |
s:=6; if s<0 then s:=s+10; s:=s+1; WriteLn (s) |
7 |
Пояснение: Обратите внимание, что в последнем примере оператор if кончается оператором s:=s+10, а не s:=s+1. Поэтому
оператор s:=s+1 будет выполняться всегда, независимо от величины s.
В условии оператора if сравниваемые строки должны совпадать полностью. Научившись выполнять операции над строками, вы
научитесь избегать таких ситуаций.
Синтаксис
Семантика
Примеры использования для решения задач
Пример 1. Нахождение минимума
Дано: x, y
Найти: min
Оператор case выбора варианта
Синтакстис
Семантика
Вначале вычисляется выражение<переключатель>, после чего его значение ищется в одном из
<списков выбора>.
Если значение попадает в какойто <список выбора>, то выполняется соответствующий ему оператор, иначе, если есть
ветвь else, то выполняется оператор по ветке else.
Ограничения выражениепереключатель должно иметь так называемый порядковый тип:
целый символьный перечислимый
НО НЕ строковый или вещественный.
значения в <списках выбора> не должны пересекаться.
Примеры использования оператора выбора
Пример 1. День недели
Пример 2. Цифра или буква
Контрольные вопросы
1.Составной оператор
2. Синтаксис, семантика оператора IF
3. Правила записи
4. Синтаксис, семантика оператора case
Тема 2.3. Операторы цикла
Лекция№13 Циклы с пост и предусловием Цикл с параметром
Вложенные циклы
План лекции
1.Оператор цикла while
2.Оператор цикла repeat (do while)
3.Вложенные циклы
4.Оператор цикла с параметром (for)
Циклы
Цикл — разновидность управляющей конструкции в высокоуровневых языках программирования, предназначенная для
организации многократного исполнения набора инструкций. Основная цель циклов – сократить размер текста
программы.
В языке Си операторы while, do while и for , в языке Pascal операторы while, repeat
Оператор цикла while называется циклом с предусловием и имеет следующий формат: while (выражение)
оператор
В качестве выражения допускается использовать любое выражение языка Си, а в качестве тела любой оператор, в том числе
пустой или составной. Схема выполнения оператора while следующая:
1. Вычисляется выражение.
2. Если выражение ложно, то выполнение оператора while заканчивается и выполняется следующий по порядку оператор.
Если выражение истинно, то выполняется тело оператора while.
3. Процесс повторяется с пункта 1.
В операторе while вначале происходит проверка условия продолжения цикла, которая предваряет непосредственно
циклические вычисления, поэтому оператор while удобно использовать в ситуациях, когда тело оператора не всегда нужно
выполнять, а также когда заранее неизвестно количество необходимых для выполнения шагов цикла.
Синтаксис цикла while Семантика цикла while
В качестве примера использования оператора цикла рассмотрим программу вычисления факториала целого положительного
числа NL Сопоставим программу решения этой задачи, написанную на Паскале, с программой на Си. Пример…см.учебник
СемакинаИ.Г. стр65/стр.202, метуказ. Лясина Д.В. ОП лаб2
Синтаксис цикла repeat (ЯП Паскаль), цикла do whil (ЯП С++)
Циклы с постусловием (repeat) — сначала делается повторение цикла, а потом проверяется условие и если оно
выполняется, то происходит следующее повторение цикла.
Исполнение цикла повторяется до того момента, когда станет равным true.
Семантика цикла repeat
Оператор цикла do while называется оператором цикла с постусловием и используется в тех случаях, когда необходимо
выполнить тело цикла хотя бы один раз. Формат оператора имеет следующий вид:
do {тело} while (выражение);
Схема выполнения оператора do while (рис.2):
1. Выполняется тело цикла (которое может быть составным оператором).
2. Вычисляется выражение.
3. Если выражение ложно, то выполнение оператора do while заканчивается и выполняется следующий по порядку оператор.
Если выражение истинно, то выполнение оператора продолжается с пункта 1.
Зацикливание происходит, если:
условие цикла с предусловием всегда истинно
условие цикла с постусловием всегда ложно, иными словами равно нулю.
Пример
Использование в качестве выражения константы 1 приводит к тому, что условие повторения цикла все время остается
истинным и работа цикла никогда не заканчивается. Тело в этом цикле представляет собой пустой оператор. При
исполнении такого оператора программа будет «топтаться на месте». см. учебник Семакина стр.202
Итерация — однократное повторение тела икла. Отличия между циклами while и repeat while тело может не выполниться ни
разу ,repeat тело выполнится хотя бы один раз.
Пример 1. Сумма нечетных двузначных чисел
С использованием while С использованием repeat
Моделирование repeat с помощью while Моделирование while с помощью repeat
Вложенные циклы
Операторы while и do while могут быть вложенными.
Пример:
int i,j,k;
…
i=0; j=0; k=0;
do { i++;
j—;
while (a[k] < i) k++;
}
while (i<30 && j<-30);
Следующий фрагмент программы на Си++ содержит два вложенных цикла for. В нем запрограммировано получение на экране
таблицы умножения.
Вывод. При наличии нескольких вложенных циклов, в первую очередь, нужно оптимизировать самый внутренний.
Оператор цикла с параметром (for)
Формат оператора цикла с параметром:
for (выражение_1; выражение_2; выражение_3) оператор;
Выражение 1 выполняется только один раз в начале цикла. Обычно оно определяет начальное значение параметра цикла
(инициализирует параметр цикла). Выражение 2 — это условие выполнения цикла. Выражение 3 обычно определяет изменение
параметра цикла, оператор — тело цикла, которое может быть простым или составным. В последнем случае используются
фигурные скобки.
Синтаксис
Семантика
Ограничения:
выражения 1 и 2 должны быть совместимы по присваиванию с переменной
переменная должна иметь порядковый тип (такой же, как и в case — целый, символьный или перечислимый) переменная цикла
for не должна меняться внутри цикла for переменная цикла for должна быть описана в той же п/п, где используется
цикл
:
Контрольные вопросы
- Оператор цикла while
- Оператор цикла repeat (do while)
- Вложенные циклы
- Оператор цикла с параметром (for)
Раздел 3. Структурное и модульное программирование
Тема 3.1. Процедуры и функции
Лекция№14 Общие сведения о подпрограммах Определение и вызов
подпрограмм
План лекции
- Вспомогательные алгоритмы
- Процедуры Описание и вызов процедуры
- Функции Описание и вызов функции
- Функции обратного вызова (callback)
.
Вспомогательные алгоритмы
Вспомогательный алгоритм — это алгоритм, который используется для реализации другого алгоритма. Вспомогательные
алгоритмы имеют: имя, список параметров. Некоторые параметры являются входными, а некоторые — выходными
Если один из выходных параметров возвращается особым образом, так что алгоритм можно использовать в выражении, то
такой алгоритм называется алгоритмомфункцией.
В программировании вспомогательные алгоритмы называются подпрограммами и делятся на: процедуры и функции.
Подпрограммы позволяют избежать дублирования кода при решении однотипных задач. Алгоритм один раз описывается, а
затем может быть многократно вызван с различным набором параметров из другого алгоритма.
Процедуры
Пример. Даны 3 пары положительных чисел.
Найти их среднее арифметическое и среднее геометрическое.
Очевидно, удобно сделать подпрограмму, которой бы на вход подавались два числа, а выходными параметрами являлись их
среднее арифметическое и среднее геометрическое соответственно.
Строка называется заголовком процедуры.
Теперь можем многократно использовать описанную процедуру в основной программе:
Оператор вызова процедуры
Синтаксис описания процедуры
Замечание. Pascal допускает внутренне описание подпрограмм (вложенность подпрограмм)
Синтаксис вызова процедуры
Семантика вызова процедуры
имя должно быть именем процедуры, описанной ранее (или именем текущей процедуры) количество фактических параметров
должно совпадать с количеством формальных параметров фактические параметры переменные должны быть именами
переменных, типы которых совпадают с типами соответствующих формальных параметров
фактические параметры значения должны быть выражениями, типы которых совместимы по присваиванию с типами
соответствующих формальных параметров
Входно выходные параметры описываются с ключевым словом var, как и выходные.
Пример 1. Увеличение значения переменной на заданное число.
Функции
Функции являются другой разновидностью подпрограмм. Это подпрограмма, возвращающая одно значение особым образом так,
что ее вызов можно использовать в выражении. Точнее, — вызов процедуры является оператором, а вызов функции —
выражением. Пример. Функция sign(x) — знак числа
называется заголовком функции, а integer — это тип возвращаемого значения.
В каждой функции неявно определена переменная Result, хранящая возвращаемое значение функции и имеющая тот
же тип. Присваивание значения Result в теле функции обязательно по всем ветвям алгоритма
Функции очень похожи на процедуры. Но функция в отличие от процедуры обладает некоторыми свойствами переменной
величины и поэтому описание функции отличается от описания процедуры следующими двумя вещами:
- В заголовке функции после скобок с формальными параметрами должен быть указан тип функции (у нас это Integer).
- Внутри описания функции между BEGIN и END ей хотя бы раз должно быть присвоено какое-нибудь значение (у нас это
perimetr:=2*(dlina+shirina)).
Параметры по умолчанию
Предварительное объявление подпрограмм
Forward объявление делается для подпрограмм, которые обязательно будут описаны ниже. Если программа не будет описана
в этом же файле, то, в конце компиляции этого файла, мы получим ошибку компилятора о forward, который был
объявлен, но не описан.
Процедурные переменные
Процедурный тип и переменные
Переменная называется процедурной если ей можно присваивать процедуры или функции указанного типа. После присваивания
процедурной переменной имени процедуры, эта процедура может быть вызвана через переменную:
Такой способ вызова процедуры является чрезвычайно гибким, поскольку позволяет менять вызываемое действие в процессе
работы программы.
Замечание. До первого присваивания процедурная переменная имеет тип nil. В этот момент вызывать процедуру
через переменную нельзя — будет ошибка выполнения.
Функции обратного вызова (callback)
Процедурные переменные могут являться параметрами других процедур.
При вызове этих процедур им на вход в качестве параметров передаются имена процедур, которые будут вызваны позже
внутри основной процедуры. Другими словами, мы передаем в процедуру действие, которое должно быть вызвано (будет
вызвано) в определенный момент в этой процедуре. (обратный вызов ( callback)).
Примечание. Прямой вызов — это передача процедуры в качестве параметра.
Контрольные вопросы
- Вспомогательные алгоритмы
- Процедуры Описание и вызов процедуры
- Функции Описание и вызов функции
- Функции обратного вызова (callback)
История создания структурного программирования (реферат)
Тема 3.2. Структуризация в программировании
Лекция№15 Основы и методы структурного программирования
План лекции
1. Основные положения структурного программирования
2. Элементы структурного программирования
3. Метод разработки «сверху вниз»
4. Метод разработки «снизу вверх»
Технология структурного программирования базируется
на процедурной декомпозиции, при которой программа представляется в виде иерархической структуры блоков. Структурный
подход к программированию был предложен в 70-ых годах ХХ века Э. Дейкстрой, разработан и дополнен Н. Виртом , Х.Д.
Милсом, Д. Е. Кнутом и др. Цель структурного программирования — повышение качества и надежности разрабатываемых
программ, сокращение сроков разработки. см. учебник Семакина И.Г. стр.268
Основные положения структурного программирования
- Программа должна содержать только основные структуры алгоритмов: базовые (следование, ветвление, цикл-пока) и дополнительные к базовым (выбр, цикл-до,счетный цикл). Эти конструкции могут быть вложены
друг в друга, но никакие другие средства управления последовательностью выполнения операций не должны
использоваться (например, оператор
безусловного перехода). - Повторяющиеся фрагменты программы или фрагменты, представляющие
из себя логически целостные вычислительные блоки, оформляются как подпрограммы (функции). Тогда в тексте основной программы
вместо самих фрагментов будут присутствовать вызовы соответствующих функций. При вызове функции управление передается
на выполнение этой функции. После чего управление возвращается на оператор, следующий за вызовом функции. - Разработка программы ведется пошагово, методом «сверху-вниз».
Сначала разрабатывается структура основной программы, которая должна состоять в основном из вызовов функций, каждая
из которых выполняет определенное действие. Вместо самих функций, в программу вставляются «заглушки».
«Заглушка»- это функция, имеющая «пустое» тело, то есть «заглушка» ничего не делает.
Полученная программа отлаживается. После того, как программист убедился, что функции вызываются в правильном
порядке, то есть структура основной программы верна, последовательно разрабатываются функции-«заглушки».
При этом разработка каждой функции ведется также, как и основной программы. Разработка программы заканчивается
тогда, когда не останется ни одной «заглушки». Такая последовательность разработки программы позволяет
программисту отлаживать небольшой логически законченный фрагмент программы. При этом ошибки локализуются именно в
отлаживаемой функции. К разработке следующей функции можно переходить только тогда, когда данная функция будет
работать безошибочно.
Кроме этого программы должны быть «самодокументированы». Это предполагает широкое и грамотное
использование комментариев и идентификаторов.
При подготовке текста программы необходимо использовать отступы, которые должны отражать вложенность операторов. Это
позволяет визуально отслеживать структуру программы.
Структурное программирование позволяет быстро и качественно не только разрабатывать программы, но и модифицировать их
в процессе эксплуатации.
.
Элементы структурного программирования
Структуризованная программа (или подпрограмма) — это программа, составленная из фиксированного множества базовых
конструкций. Рассмотрим основные определения и способы образования этих конструкций в схемах алгоритмов.
{}
Из операций, развилок и слияний строятся базовые конструкции: следование, ветвление, цикл. Применяя только эти три
конструкции, можно реализовать алгоритм решения любой задачи.
Конструкция, представляющая собой последовательное выполнение двух или более операций, называется следованием.
Конструкция, состоящая из развилки, двух операций и слияния, называется ветвлением. Одна из операций может
отсутствовать.
Конструкция, имеющая линии управления, ведущие к предыдущим операциям или развилкам, называется циклом.
Конструкции следование, ветвление и цикл можно представить как операции, так как они имеют единственный вход и
единственный выход.
Произвольную последовательность операций можно представить как одну операцию.
Операция может быть реализована любым оператором языка ПАСКАЛЬ (простым или составным), либо группой операторов, за
исключением оператора перехода GOTO.
В языке ПАСКАЛЬ количество базовых конструкций увеличено до шести, это:
-следование, -ветвление; -цикл с предусловием; -цикл с постусловием;
-цикл с параметром; -вариант.
Методы разработки подпрограмм
1.Метод разработки «сверху вниз»
Сначала строится основной алгоритм, затем вспомогательные алгоритмы.
Этот метод используется, когда: основной алгоритм понят, и его можно сразу записать; задача четко поставлена; имеется
ровно одна задача, которую необходимо решить.
Замечание. В коде главной программы имеются вызовы других подпрограмм, которые реализуются позже (возможно, другими
разработчиками);
На период, пока они не написаны, вместо них могут использоваться «заглушки» (подпрограммы с тривиальными телами).
Код каждой разрабатываемой подпрограммы также может разрабатываться методом сверху вниз.
2.Метод разработки «снизу вверх»
Вначале составляется «нижняя» подпрограмма, она всесторонне тестируется, и потом пишется основная программа или
подпрграмма.
Метод используется, когда: на момент начала решения задача нечетко поставлена; когда задача — большая; нет ясно
выраженного главного алгоритма, а есть множество взаимосвязанных задач (примером может служить программа,
реализующая деятельность университета).
Первый подход еще называют методом последовательной детализации, второй — сборочным методом.
Вспомогательные алгоритмы самого нижнего уровня состоят только из простых команд.
Метод последовательной детализации применяется в любом конструировании сложных объектов. Методика последовательной
детализации позволяет организовать работу коллектива программистов над сложным проектом.
Если между главным алгоритмом и «нижними» подпрограммами имеется несколько уровней, то имеет смысл сочетать оба
метода разработки.
Контрольные вопросы
1.Основные положения структурного программирования
2. Элементы структурного программирования
3. Метод разработки «сверху вниз»
4. Метод разработки «снизу вверх»
Тема 3.3. Модульное программирование
Лекция№16 Понятие и структура модуля. Компиляция и компоновка
программы
План лекции
- Модуль Примеры
- Синтаксис модуля Семантические ограничения
- Схема компиляции программы с модулями
- Компоновка
Модуль — это совокупность взаимосвязанных процедур, функций, типов, переменных и констант, предназначенных для
решения ряда однотипных задач, и помещенных в специальным образом оформленный файл.
Модули разбивают большой проект на относительно независимые части, при этом, каждая часть «живет своей жизнью»:
модуль, написанный для одного программного проекта, может быть использован в другом программном проекте.
Различают модули в виде исходных текстов и откомпилированные модули.
Откомпилированные модули уменьшают суммарное время компиляции и позволяют скрывать программный код от
модификации.
Приме: Модуль MyLib
Модуль в языке Object Pascal состоит из двух разделов:
раздел интерфейса, раздел реализации
В разделе интерфейса описываются все переменные константы, типы, заголовки подпрограмм, которые можно будет
использовать в других модулях, подключающих данный.
В разделе реализации содержится
реализация подпрограмм, заголовки которых приведены в разделе интерфейса описание вспомогательных констант,
переменных, типов и подпрограмм, которые нужны для реализации подпрограмм из раздела интерфейса и не видны из других
модулей.
Синтаксис модуля
Семантические ограничения
тем не менее, если A ссылается на B в разделе интерфейса, а B ссылается на A в разделе реализации, то это
допустимо:
потому что модули компилируются в два этапа: интерфейс, реализация.
1. Компилируется файл основной программы. 2. Если в нем встречена секция uses, то компилируются модули из этой секции
слева направо. 3. Каждый модуль компилируется так же, как и основная программа по пунктам 12:
Если в модуле подключаются другие модули, то компилируются они,
и так происходит до тех пор, пока компилятор не дойдет до модулей, не содержащих подключения других модулей.
По окончании компиляции модуля его откомпилированный вариант (.pcu в PascalABC.NET) записывается на диск.
После записи откомпилированного модуля на диск компилятор возвращается к основному модулю (вызывающему) или
программе, и докомпилирует его до конца. Основная программа после компиляции хранится в оперативной памяти.
Первый этап компиляции закончен. Начинается заключительный этап компиляции — линковка (компоновка). Специальная
программа — линковщик — собирает из откомпилированных модулей единый исполняемый файл (.exe в Windows).
Стандартные откомпилированные модули хранятся в специальной папке, например, в PascalABC.NET — в папке \LIB. Если
модуль претендует на звание стандартного, его можно туда поместить.
Контрольные вопросы
- . Модуль Примеры
- Синтаксис модуля Семантические ограничения
- Схема компиляции программы с модулями Компоновка
Раздел 4. Структуры данных
Тема 4.1. Массивы
Лекция №17 Понятие массива. Особенности программирования
массивов
План лекции
- Определение массива
- Виды массивы
- Динамические массивы
- Сортировка массивов
Массив — упорядоченная структура, предназначенная для хранения однотипных данных.
Упорядочение элементов в массиве происходит по их индексам.
Индекс — порядковый номер элемента. Индексов может быть несколько. Такие массивы называются многомерными (с одним
индексом — одномерными соответственно).
Массив задается именем (заглавные латинские буквы), типом данных и размерностью.
Размерность — максимально возможное количество элементов в массиве. В один момент времени можно обратиться только к
одному элементу массива. Для этого указывается имя массива и в скобках индекс элемента.
Массивы делятся на одномерные (линейные) и двумерные.
Прообразом в математике для одномерного массива является вектор. Для двумерного – матрица.
Массивом можно назвать ряд ячеек памяти, отведенных для хранения значений индексированной
переменной.
Какие бывают массивы
Массивы могут быть одномерные, двумерные, трехмерные, четырехмерные и т.д.:
array [1..10] of Integer -одномерный массив 10 ячеек
array [1..10, 1..5] of Integer -двумерный массив 50 ячеек
array [1..10, 1..5, 1..2] of Integer -трехмерный массив 100 ячеек
array [1..10, 1..5, 1..2, 1..3] of Integer -четырехмерный массив 300 ячеек
Массивы бывают не только числовые, но и символьные, строковые и прочие. Подходит любой известный нам тип.
Например:
array [1..50] of Char
Это означает, что в каждой из 50 ячеек должно находиться не число, а произвольный символ.
Пример.
Как и диапазонный тип, индексы могут иметь типы: целый, символьный, перечислимый.
Кроме того, в качестве индекса может выступать порядковый тип, например:
Обращение к элементам массивов
Чтобы обратиться к элементу массива, нужно использовать конструкцию
В некоторых компиляторах можно специальными директивами отключить проверку выхода за границы диапазона, это
увеличивает скорость выполнения.
Динамические массивы
Динамическим называется массив, память под который выделяется в процессе работы программы. В pascalABC.NET имеются
встроенные динамические массивы, которые описываются следующим образом:
Динамические массивы индексируются с нуля, тип индекса только целое.
Выделение памяти
Для выделения памяти динамическому массиву имеется два варианта. Объектный стиль:
Процедурный стиль:
Здесь n может быть не только константой, но и переменной.
В обоих случаях элементы массивов заполняются нулями.
Сортировки массивов
Сортировкой называется процесс расположения элементов массива в порядке убывания (возрастания) из
значений.
Пример :
Алгоритм
выполнения сортировки называется методом сортировки. К наиболее распространенным методам относятся:
- Простым выбором
- Простой перестановкой
- Пузырьковый метод
- На каждом шаге находится минимальный (максимальный) неотсортированной части. Он меняется с первым элементом в
неотсортированной части, после чего отсортированная часть увеличивается на один элемент. На первом шаге весь
массив считается неотсортированным. Сортировка заканчивается за (n-1) шаг. Это самый компактный алгоритм
сортировки. Его можно оптимизировать, проверяя случай
«холостого» прохода по элементам массива. Т.е. если за проход ни один элемент не изменил позицию, значит
массив уже отсортирован и проверять дальше нет смысла
1. Пример: 241795
1 шаг: 1 | 42795 |
Контрольные вопросы
- Определение массива
- Виды массивы
- Динамические массивы
- Сортировка массивов
Тема 4.2. Строки
Лекция№18 Символьный и строковый типы. Объявление типов.
План лекции
1. Объекты символьного типа
2. Кодирование управляющих символов
3. Инициализация переменных
4. Функция gets Терминатор
Символьная информация очень часто является предметом обработки в программах на языках высокого уровня. Примером могут
служить как простой вывод текстовой информации на экран для информирования пользователя о ходе вычислительного
процесса или результатах его работы, так, например, обработка информации из текстового файла, требующая выполнения
специфических операций слияния строк, удаления подстрок из текста, изменения порядка следования слов или фраз и т.п.
Для решения задач подобного класса в языке Си поддерживается работа со строками – объектами, хранящими символьную
информацию.
Фактически, строки рассматриваются как массивы, элементами которых являются объекты символьного типа (char или
unsigned char). Однако, смысловая связь элементов строки, объединяющая буквы в слова, а слова в предложения,
обособляет строки от массивов других типов, поэтому для них предусмотрены как специальные способы обозначения, так и
алгоритмы обработки в виде специализированных функций.
Строковые константы в языках Си и Си++ помещаются в кавычки:
“Это строка на языке Си”
Этот объект имеет в программе тип char * и указывает на то место в памяти программы, где расположена
данная строка. При этом размер выделенного под строку участка памяти (при условии кодировки информации в стандарте
ANSI) равен количеству символов строки и еще один символ (терминатор строки ‘\0’).
Символьные константы заключаются в апострофы:
‘A’ – символ А, ‘ ‘ – символ пробел.
Эти объекты имеют тип char и их значение – код соответствующего символа в кодировке ASCII. Если
известен ASCII-код символа, то его можно представить в программе в виде ‘\код8’ или ‘\xкод16′,
где код8 – код символа в восьмеричной системе счисления, а код16 – в
шестнадцатеричной. Например, следующий код:
cout<<“\x48\x45\x4c\x4c\x4f”;
выведет на экран строку HELLO, поскольку ‘\x48’-код символа ‘H’,
‘\x45’ – символа ‘E’ и т.д.
Для кодирования управляющих символов в языке Си используются также специализированные Esc-последовательности:‘\n’
–перевод строки
‘\t’ – горизонтальная табуляция |
‘\a’ – сигнал-звонок |
‘\r’ –возврат курсора к началу строки |
‘\b’ – возврат на одну позицию |
‘\\’ – обратный слеш |
‘\f’ – перевод страницы |
‘\» – апостроф |
‘\v’ – вертикальная табуляция |
‘\”’ – кавычка |
Если строковые данные изменяются в процессе выполнения программы, то их необходимо хранить в переменных. Строковая
переменная определяется в языке Си как массив:
char str[20];
Как всякую переменную, строку можно инициализировать при определении с помощью константной строки:
char str[20]=”Привет”;
Инициализацию можно осуществлять и поэлементно, но такая форма записи неудобна:
char str[20]={‘П’, ’р’, ‘и’, ‘в’, ‘е’ ‘т’, ‘\0’};
Обращает на себя внимание необходимость в таком случае обязательно указывать терминатор ‘\0’ (символ, определящий
конец полезной информации в строке) в конце строки. В константных строках терминатор добавляется в конец строки
автоматически.
После такого определения можно работать с каждым символом строки как с обычным элементом массива:
str[0]=’п’; //теперь строка начинается со строчной буквы
cout<<str[2]; //выводим на экран букву и
Несмотря на схожесть строк с определением обычных массивов, они имеют ряд уникальных свойств. Так, например,
ввод-вывод элементов целочисленного массива всегда осуществляется поэлементно:
int mas[20];
for (i=0; i<20; i++)
cin >> mas[i]; //вводим очередной элемент массива
Однако, поэлементный (побуквенный) ввод текстовой информации, когда каждый символ необходимо подтверждать клавишей
Enter был бы неудобен. В этой связи функции ввода вывода языков Си и Си++ поддерживают работу со
строками, позволяя вводить или выводить содержимое строки целиком:
Ввод строки с клавиатуры char str[20]; cin >> str;//или scanf(”%s”, str) |
Вывод содержимого строки на экран cout << str; //или printf(”%s”, str); |
Необходимо сразу отметить, что ввод текстовой информации с использованием функции scanf или объекта
cin не всегда допустим, поскольку эти объекты считают пробел за разделитель вводимых объектов и
поэтому позволяют вводить строку до первого пробела. Если необходим ввод данных, содержащих пробелы, лучше
воспользоваться специальной функцией gets:
char *gets(char *s);
В отличие от стандартных средств ввода данных, gets допускает ввод пробельных символов (пробел,
символ табуляции).
char str[20];
gets( str); // можно вводить строку с пробелом
Использование функции gets небезопасно: пользователь может ввести символов больше, чем
зарезервировано под строку, и это приведет к порче областей данных, смежных с занимаемой строкой. Безопасной версией
функции является gets_s:
char *gets(char *s, size_t sizeInCharacters);
где параметр sizeInCharacters задает максимально возможное количество считываемых из входного потока
символов:
char str[20];
gets_s( str, 20); //безопасно от переполнения буфера
На практике можно использовать также функцию fgets в форме:
fgets(str, sizeof(str), stdin); /* считать из входного потока stdin(c клавиатуры) не более sizeof(str) символов в
строку str */
Вывод содержимого строки с использованием стандартных средств (cout, printf) не вызывает проблем,
но существует и специализированная функция для вывода строк puts:
int puts(const char *s);
char str[20];
…
puts( str ); //выводим содержимое строки на экран
Важную роль в обработке данных в строке играет так называемый терминатор — символ с нулевым кодом,
завершающий информационную часть строковой переменной. Рассмотрим фрагмент программы:
char str[20]=”Hello”, str1[30];
puts( str );
gets_s(str1, 30);
fputs(str1);
Что будет выведено на экран в первом и втором случае? Все 20 символов строки str или только строка “Hello”,
занимающая ее первые несколько символов? Во втором случае на экран выведется 30 символов строки str1 или
только та строка, которую перед этим введет пользователь? Очевидно, что на экран должна выводиться лишь полезная,
информационная часть строки, а ее хвост, заполненный компьютерным «мусором», должен игнорироваться при выводе и
выполнении других функций обработки строки.
Для отделения информационной строки от неинициализированной ее части используется терминатор ‘\0’. Все
функции для работы со строками языка Си учитывают наличие терминатора. Когда строка вводится с клавиатуры, после
всех введенных символов в строку добавляется терминатор. Запись константной строки ”…” предполагает
наличие терминатора на месте закрывающейся кавычки. Функции обработки строк (объединения, поиска, сравнения и т.д.)
просматривают содержимое строки до терминатора. Поэтому непредсказуемо закончится действие такого фрагмента:
char str[6]=”Hello”; //5 символов строки+ терминатор
str[5]=’!’; //изменяем символ с индексом 5 – это был терминатор
fputs(str); /* строка выводится, пока не встретится терминатор, но мы его удаили из строки */
Приведенный пример будет выводить информацию, начиная с адреса начала строки в памяти, до тех пока не встретит байт с
нулевым значением. Данный пример подчеркивает необходимость осторожного использования операций посимвольного
изменения данных в строке.
Контрольные вопросы
1. В какой форме можно представить строковую информацию в языке Си?
2. Что такое терминатор строки и какую роль он играет при обработке данных в строке?
3. Почему при вводе информации в строку предпочтительнее использовать функцию fgets?
Лекция№19 Поиск, удаление, замена и добавление символов в
строке.
План лекции
- Получение длины строки
- Копирование строки
- Конкатенация
- Лексикографическое сопоставление
Переменная типа строка предназначена для обработки цепочек символов. Каждый символ является элементом типа char.
Строки могут вводиться с помощью стандартных операторов read/readln и выводиться стандартными
операторами write/writeln.
Объявляются переменные типа строка в разделе var. При объявлении указываются
идентификатор переменной, зарезервированное слово string и, в квадратных скобках, целое число — максимально
возможная длина строки. Наибольшая длина строки составляет 256 символов. Если переменная имеет значение с
максимальной длиной строки, то при объявлении переменной ограничиваются зарезервированным словом.
Пример:
var
identificator_1:
string;
identificator_2: string[20];
identificator_3: string[255];
Значение строкового типа
также как и значение типа char при записи внутри программы заключаются в апострофы.
Пример:
identificator_1:=’это
— компьютер’;
Для обработки текстовой информации типичными являются операции вставки и замены подстрок, их
поиска и удаления, слияния и сравнения строк. Для реализации этих операций удобно использовать специализированные
функции из библиотек string.h, stdlib.h, stdio.h. Например, для получения длины строки (ее информационной части до
терминатора) используется функция strlen библиотеки string.h:
int strlen(char * string);
Рассмотрим задачу копирования одной строки в другую. Простое копирование с использованием операции присваивания
недопустимо.
Гораздо удобнее воспользоваться стандартной функцией копирования строк strcpy:
char *strcpy(char *dest, const char *src);
Эта функция копирует содержимое строки src в строку dest, как результат возвращает адрес строки приемника dest.
char str[100]=“One, two, three”, temp[50];
strcpy(temp, str); //копируем содержимое строки str в temp
puts(temp);
strcpy(temp, “four, five”);
puts(temp);
Использование функции strcpy в программе небезопасно, если источник копируемой строки ненадежен
если пользователь введет больше 49 символов, то копирование строки str приведет к переполнению строки temp,
объем копируемой информации превысит объем зарезервированной под строку памяти и это приведет к изменению смежных с
занимаемыми temp областями памяти.
Безопасной версией функции копирования строк является функция
errno_t strcpy_s(char *strDestination, size_t numberOfElements,
const char *strSource );
Функция strcpy_s копирует содержимое в адресе strSource, включая конечный
символ-терминатор строки, в строку-приемник, указанную параметром strDestination. Строка назначения должна
быть достаточно велика для хранения строки источника и его конечного нуль-символа.
Еще одной часто используемой операцией со строками является их объединение (конкатенация). Операцию можно реализовать
с использованием функции strcat
char *strcat(char *dest, const char *src);
Объединяет исходную строку src и результирующую строку dest, присоединяя первую к последней.
Возвращает адрес приемника dest.
char str[100]=“One, two, three”;
strcat(str, “, four”);
puts(str); //на экран выведет One, two, three, four
Использование функции strcat также может быть небезопасным, поскольку отсутствует контроль над
количеством копируемых символов и результирующая строка способна переполнить приемник.
Еще одной функцией, позволяющей контролировать количества добавляемых к строке символов, язвлеятся
strncat:
char *strncat(char *dest, const char *src, int maxlen);
Эта функция объединяет результирующую строку dest и не более maxlen символов строки src.
Таким образом, должно соблюдаться неравенство:
strlen ( dest ) + maxlen + 1 ≤ sizeof ( dest )
Важной и очень часто используемой операцией в программе является процедура сравнения строк. Под сравнением здесь
понимается лексикографическое сопоставление строк, и меньшей при таком сравнении будет та строка, которая в условном
словаре будет стоять выше (раньше). Такое сравнение должно выполняться попарным последовательным сравнением кодов
символов строк слева направо до первого несовпадения или до конца одной из строк.
Выполнить сравнение строк может функция strcmp:
int strcmp(char *s1, char *s2);
Эта функция сравнивает две строки и возвращает отрицательное значение, если s1<s2; нуль, если s1=s2; положительное
значение, если s1>s2. Отношение s1>s2носит лексикографический характер, по которому, например:
“abc”>”aaa” “bcd”>“abc”
“aaa”>”aa” “abc”>”Abc”
Используя данную функцию, можно, например, защитить свою программу паролем. На практике часто можно встретить вариант
сравнения, не использующий операции отношения:
if (!strcmp(str, “мой пароль”))
Функция сравнения имеет ряд модификаций:
int stricmp(char *s1, char *s2);
Операцию поиска подстроки в строке можно реализовать с использованием функции strstr:
char *strstr(char *s1, const char *s2);
Функция ищет в строке s1 строку s2. Возвращает адрес первого символа вхождения строки s2.
Если строка отсутствует — возвращает нуль.
char str1[20]=“One, two, three, two”;
char str2[10]=“two”;
cout<<strstr(str1,str2); //на экран выведет two, three, two
В приведенном примере функция strstr найдет первое вхождение строки “two” в str1 и вернет
адрес первого символа ‘t’. Далее выводится содержимое строки str с этой позиции.
В состав стандартных библиотек языка Си не входит функция удаления подстроки. Но ее можно реализовать с помощью
других функций. Удалить подстроку длинною k символов, начиная с позиции i, можно вызовом strcpy:
strcpy(str+i, str+i+k);
Контрольные вопросы
- Почему использование функций strcpy и strcat может быть потенциально опасным для программы?
- . Что такое лексикографический порядок следования строк? Как реализуется лексикографическое сравнение строк в
языке Cи?
Лекция№20 Операции со строками. Функции и процедуры. Решение
задач.
План лекции
1.Таблица функций языка Си для строк
2. Функции языка Си для преобразования
3. Функциия strtok
В таблице 1 приведены некоторые дополнительные функции, которые могут быть использованы для обработки информации в
строках
Таблица 1. Функции языка Си для работы со строками
Прототип функции |
Выполняемое действие |
char *strchr(const char *s, int c); |
Ищет в строке s первое вхождение символа c, начиная с начала строки. В |
char *strrchr(const char *s, int c); |
Аналогично предыдущему, только поиск осуществляется с конца строки. |
int strcspn(const char *s1, const char *s2); |
Возвращает длину максимальной начальной подстроки строки s1, не содержащей символов из |
char *strdup(const char *s); |
Копирует строку во вновь выделенный блок памяти, самостоятельно выделяя из кучи |
char *strlwr(char *s); |
Преобразует все прописные (большие) буквы в строчные (малые) в строке s. |
char *strupr(char *s); |
Преобразует все строчные (малые) буквы в прописные (большие) в строке s. |
char *strnset(char *s, int c, int n); |
Заполняет строку s символами c. Параметр n задает |
char *strpbrk(const char *s1, const char *s2); |
Ищет в строке s1 первое вхождение любого символа из строки s2. |
char *strrev(char *s); |
Изменяет порядок следования символов в строке на обратный (кроме завершающего нулевого символа). Функция |
char *strset(char *s, int c); |
Заменяет все символы строки s заданным символом c. |
int strspn(const char *s1, const char *s2); |
Вычисляет длину максимальной начальной подстроки строки s1, содержащей только символы из |
Еще один тип часто использующихся при работе со строками операций — преобразование числовых данных в строковые и
обратно. Функции, реализующие эти операции, представлены в таблице 2.
Таблица 2. Функции языка Си для преобразования числовых данных в строковые и обратно
Прототип функции |
Выполняемое действие |
double atof(const char *s); |
Преобразует строку s в число с плавающей точкой типа double |
int atoi(const char *s); |
Преобразует строку s в число типа int. Возвращает значение или нуль, |
char *itoa(int value, char *s, int radix); |
Преобразует значение целого типа value в строку s. Возвращает указатель |
char *ecvt(double value, int ndig, int *dec, int *sign); |
Преобразует значение value типа double в завершающуюся нулем строку. |
Для преобразования данных из числового типа в строку можно использовать также функцию sprintf, а для
обратного – sscanf:
char str[10], str1[10];
int x=-30;
double y=123.45678;
sprintf(str, “%d”, x);
puts(str); //выведет -30
sprintf(str1, “%5.2lf”, y);
puts(str1); //выведет 123.46
При работе со строками зачастую встает задача разбиения строки на лексемы – подстроки с использованием одного или
нескольких разделителей. Так, например, может быть поставлена задача разбить строку на предложения – тогда
разделителями будут служить знаки пунктуации, завершающие предложения: ‘.’, ‘!’, ‘?’. Если же
встанет задача разбить предложения на слова, то в качестве разделителей придется использовать символы ‘ ’,
‘,’, ‘:’, ‘-’, ‘»’, ‘”’, ‘;’. Разбиение строки на лексемы можно выполнить
алгоритмически, отыскивая в строке символы-разделители и занося подстроки, заключенные между ними в отдельные
элементы массива строк-лексем. Но гораздо удобнее будет воспользоваться специализированной функцией
strtok, которая выполнит подобное разбиение автоматически. Эта функция имеет следующий прототип:
char * strtok( char * string, const char * delims );
Как уже упоминалось, эта функция ищет в строке string лексемы, представляющие собой последовательность
символов, разделенных знаками-разделителями. Для поиска последовательности лексем функцию strtok необходимо
вызвать несколько раз, каждый вызов будет возвращать адрес очередной найденной лексемы.
При первом вызове функции необходимо передать адрес строки для поиска в качестве первого аргумента (string).
Функция начинает поиск с первого символа переданной строки. При последующих вызовах функции передается нулевой
указатель, что заставляет ее искать следующую лексему начиная с позиции окончания последней найденной лексемы.
Позиция окончания лексемы определяется по совпадению очередного символа строки string с одним из символов
строки разделителей delims. Этот конечный маркер лексемы заменяется нулевым символом, и адрес лексемы
возвращается функцией. Следующий вызов функции strtok начинаются нулевого символа — маркера конца
предыдущей лексемы. Если функция strtok при очередном вызове не найдет ни одной лексемы – она
вернет нулевой указатель. Из приведенного описания понятно, что исходная строка string при обработке
функцией strtok претерпевает изменения (в нее добавляются множество символов-терминаторов для
обозначения конца лексемы), поэтому, возможно, имеет смысл сначала сделать копию строки для использования в функции
strtok. Рассмотрим пример разбиения строки на отдельные слова с использованием функции strtok:
char str[51];
char seps[]=» ,.!?-\n»;
puts(«Введите строку»);
gets_s(str, 50);
char *pWord;
//находим первую лексему
pWord = strtok(str, seps);
//пока находятся новые лексем (слова в строке)
while (pWord) {
//выводим лексему на экран
puts(pWord);
//ищем следующую лексему
pWord = strtok(NULL, seps);
}
//Выводм на экран исходную строку
puts(str);
Если запустить этот код и ввести с клавиатуры строку
Делу время – потехе час!
На экране получим список слов этой строки:
Делу
время
потехе
час
Делу
Необходимо обратить внимание, что при попытке вывести исходную строку после ее обработки функцией
strtok, получаем только первое слово. Это связано с тем, что функция заменила вхождения в строку
разделителей из строки seps на символы конца строки.
Приведенные выше определения и примеры ориентированы на работу с кодировкой символов ASCII, где каждый символ
кодируется одним байтом. Однако, эта кодировка постепенно перестает быть общеупотребимой, уступая более
универсальным, многобайтовым. Примером здесь может служить кодировка Unicode. Стандарт Unicode был предложен
некоммерческой организацией Unicode Consortium, образованной в 1991 г. Для представления каждого символа в этом
стандарте используются два байта, что позволяет закодировать очень большое число символов из разных письменностей: в
документах Unicode могут соседствовать русские, латинские, греческие буквы, китайские иероглифы и математические
символы.
Для хранения символов Unicode необходимо два байта, из-за чего их называют широкими символами (wide characters). В
языке Си им соответствует тип wchar_t.
wchar_t wstr=L”Hello”;
Объявленная строка wstr в общем случае может иметь разный размер в различных компиляторах, при программировании под
Win32 в средах C++ Builder и Microsoft Visual Studio она будет иметь размер 12 байт (2 байта на каждый символ и
двухбайтовый терминатор). Лексемные широкие строки предваряются символом L : L ”содержимое строки ”.
Для работы с широкими символами в библиотеке string.h имеется набор специальных функций, схожих по названию и
выполняемым действиям рассмотренным выше однобайтным вариантам: wcscat (конкатенация строк),
_wcspcpy (копирования строк), wcscmp (сравнение строк) и т.п. Полный перечень
функций для работы с многобайтовыми строками можно посмотреть в документации по библиотеке string.h.
Контрольные вопросы
1 Преобразования числовых данных в строковые и обратно
2. Разбиения строки на лексемы
Тема 4.3. Множества
Лекция№21 Понятие и объявление множества. Операции над
множествами.
План лекции
- Свойства множеств
- Операции объединения, пересечения, разности
- Множестваконстанты. Пустое множество.
Множеством в Паскале называется набор значений какого-нибудь порядкового типа, подчиняющийся специфическим правилам,
о которых мы поговорим дальше. В программе множество записывается в виде списка этих значений в квадратных скобках.
Например, [7,5,0,4] или [‘п’ , ’ж’ , ’л’]. Множество не должно состоять более, чем из 256 элементов и не должно
содержать элементов с порядковыми номерами меньше 0 и больше 255.
Если в множестве элемент повторяется, то считается, что он входит туда только один раз. Например, множества [2,5,2] и
[2,5] эквивалентны.
Порядок элементов в множестве не играет роли. Множества [2,5] и [5,2] эквивалентны.
В описании тип множества задается словами set of. Например, конструкция
VAR a : set of Byte
говорит о том, что задана переменная, значением которой может быть любое множество из любого числа элементов типа
Byte. Так, в некоторый момент процесса выполнения программы значением a может быть множество [210, 3, 92], а через
пару секунд — [8, 5, 3, 26, 17].
Конструкция VAR c: set of (april, may, june) говорит о том, что переменная c может иметь значением
любое множество из имен april, may, june. Например, [april, june].
Конструкция VAR d: set of 10..18 говорит о том, что переменная d может иметь значением любое
множество целых чисел из диапазона от 10 до 18.
Над множествами определено несколько операций. Рассмотрим три из них: объединение (+), пересечение
(*) и разность (-).
Операция |
Результат |
Пояснение |
[1,4,4,5] + [1,2,3,4] |
[1,2,3,4,5] |
В результирующее множество входят элементы, имеющиеся хотя бы в одном из исходных множеств |
[1,4,4,5] * [1,2,3,4] |
[1,4] |
В результирующее множество входят только те элементы, которые имеются в каждом из исходных множеств |
[1,2,3,4] — [1,3,5] |
[2,4] |
В результирующее множество входят те элементы “уменьшаемого”, которые не встречаются в “вычитаемом” |
Операция [1,2]*[3,4] будет иметь результатом [ ], то есть пустое множество.
Вот операции сравнения множеств:
if a = b then … |
Если множества a и b состоят из одинаковых элементов … |
if a <> b then … |
Если множества a и b отличаются хотя бы одним элементом … |
if a <= b then … |
Если a является подмножеством b, то есть все элементы a являются элементами b … |
if a >= b then … |
Если b является подмножеством a, то есть все элементы b являются элементами a … |
Операция проверки вхождения элемента E в множество a:
if E in a then …
Например, a:= [1,2,4]; if 2 in a then … {Если 2 входит в множество a ….}
К сожалению, Паскаль не желает выводить множества на печать, точно так же, как он не желает печатать перечислимые
типы. Поэтому просто так узнать, из каких элементов состоит множество, не удастся. Вот один из обходных путей:
Пусть задано множество a, описанное, как set of Byte. Будем пробовать уменьшать его на все элементы подряд, от 1 до
255, и каждый раз, когда это удается, распечатывать соответствующее число. Вот подходящий фрагмент, в котором мне
понадобится “для транзита” еще одно множество b:
for i:=1 to 255 do begin
b:=a-[i]; if a<>b
then begin WriteLn(i); a:=b end
end {for}
Вот гораздо более короткий и естественный путь:
for i:=0 to 255 do if i in a
then WriteLn(i)
Я думаю, что работа с множествами Паскаля — любопытное и полезное занятие. Например, она нужна математикам, чтобы
проверять свои теоремы. Я проиллюстрирую работу с множествами на простеньком примере:
Медиум загадывает шестерку чисел, каждое в диапазоне от 0 до 10 (числа могут и совпадать). Экстрасенс отгадывает их,
называя свою шестерку. Есть ли между шестерками совпадающие числа? Если есть, то распечатать их.
Сначала решим задачу традиционными методами, а именно с применением массивов, а не множеств:
CONST razmer = 10; kol = 6;
VAR Medium, Extrasens :array[1..kol] of 0..razmer;
i, j, k
:Integer;
BEGIN
{Формируем случайным образом две шестерки:}
Randomize;
for i:= 1 to kol do begin
Medium[i]
:=Random(razmer+1); Extrasens[i] :=Random(razmer+1)
end {for};
{Проверяем две шестерки на совпадение:}
k:=0; {Нам придется
подсчитывать количество совпадений. k — счетчик}
for i:= 1 to kol
do
for j:= 1 to kol do
if Medium[i] = Extrasens[j] then begin
k:=k+1;
WriteLn(Medium[i]) {Распечатываем совпадения}
end {if};
if k=0 then
WriteLn(‘Не угадал ни разу‘)
END.
У данной программы есть недостатки. Пусть медиум загадал числа 2 4 1 5 4 8, а экстрасенс назвал 1 4 9 6 1 4.
Программа распечатает числа 4 4 1 1 4 4, а достаточно было бы только 1 4. К тому же пришлось организовывать счетчик
совпадающих чисел, чтобы иметь возможность ответить, угадано ли хоть одно число.
А теперь применяем множества:
CONST razmer = 10; kol = 6;
VAR Medium, Extrasens, a :set of 0..razmer;
i
:Integer;
BEGIN
{Формируем случайным образом две шестерки:}
Randomize;
Medium:=[ ]; Extrasens:=[ ]; {Начинаем формировать “с нуля”, то есть с пустых множеств}
for i:= 1 to kol do begin
Medium := Medium
+ [Random(razmer+1)]; {Наращиваем по одному элементу в множестве медиума}
Extrasens := Extrasens +
[Random(razmer+1)] {Наращиваем по одному элементу в множестве экстрасенса}
end {for}
a:= Medium * Extrasens; {Множество a содержит совпадающие числа. Вот так – одним махом.}
if
a=[ ] then WriteLn(‘Не угадал ни разу‘)
else begin
WriteLn(‘Есть совпадения, вот они: ‘);
{Распечатываем элементы множества a:}
for i:=0 to razmer do if i in a
then WriteLn(i);
end {else}
END.
Описание множеств. Множестваконстанты. Пустое множество.
Операции над множествами:
Вывод множеств.
Цикл foreach по множеству.
Пример использования множеств: решето Эратосфена. Алгоритм.
Лекция 25
Алгоритм Эратосфена
Код алгоритма Эратосфена
Контрольные вопросы
- Свойства множеств
- Операции объединения, пересечения, разности
- Множестваконстанты. Пустое множество.
Тема 4.4. Записи
Лекция№22 Определение типа записи. Правила работы с записями.
План лекции
1.Объявление записи
2. Обращение к записи
3.Поля записи
Пример:
Для реализации объединения
данных разного типа в языке Pascal существует специальная структура — запись. Объявление записи
начинается с зарезервированного словаrecord, за которым перечисляются имена и типы всех составляющих записей ее
полей. Заканчивается объявление скобкой end.
Пример:
type
karta = record
family:
string[20];
name: string[15];
age: integer;
end;
При обращении к записи в
программе указывается имя записи и через точку имя поля.
Пример:
karta.family:=’Иванов’;
karta.name:=’Иван’;
karta.age:=20;
Для
упрощения обращения к записи может быть использован оператор работы со структурой with.
Пример:
with karta
do
begin
family:=’Иванов’;
name:=’Иван’;
age:=20;
end;
Полями
записи наряду с простыми типами могут быть и данные структурированных типов, например, массивы или записи.
Пример
1:
var z: record
pole1: string;
pole2: array [1..10] of
byte;
end;
Begin
for i:=1 to 10 do
read (z.pole2[i]);
End.
Пример 2:
объявите запись, содержащую сведения о фамилии, дате рождения и адресе студента.
var student: record
fam:
string[15];
data: record
day: 1..31;
mes: 1..12;
year:
integer;
end;
adres: record
street: string[15];
dom:
byte;
kvart: byte;
end;
end;
Begin
with student do
begin
fam:=
‘Иванов’;
with data do
begin
day:= 30;
mes:=
4;
year:= 1987;
end;
with adres do
begin
street:=
‘Туполева’;
dom:= 22;
kvart:=
154;
end;
end;
End.
Для использования в программе набора с одинаковыми полями
используются массивы записей.
Пример: объявить массив из десяти записей.
1 вариант решения:
var A:
array [1..10] of record
fam: string;
name: string;
end;
2 вариант решения:
type
student = record
fam: string;
name: string;
end;
var A: array [1..10] of student;
Контрольные вопросы
1. Объявление записи
2. Обращение к записи
3.Поля записи
Тема 4.5. Файлы
Лекция№23 Типы файлов. Файлы последовательного доступа.
План лекции
- Определение
- Классификация
- Понятие файловой переменной, файлового указателя
- 2 способа открытия файла
- Буферизация в файлах
Файлы
Файл — именованная область на диске, содержащая некоторую информацию.
Преимущества:
- Хранят данные в промежутках между запусками программ.
- Размер данных в файле может существенно превышать оперативную память компьютера.
Классификация файлов
Файлы обычно классифицируют по двум признакам: 1. По типу компонент:
текстовые;
двоичные:
типизированные; бестиповые.
2. По способу доступа:
последовательный; произвольный.
Текстовые файлы
Тип text. Состоят из строк переменной длины, в конце каждой из которых находится символ перехода на новую
строку (#13#10 в Windows, #10 в Linux). В
PascalABC.NET это константа NewLine.
Двоичные файлы: информация хранится в виде двоичного кода.
Типизированные файлы
Тип file of <type>. Содержат данные фиксированного типа <type>.
Бестиповые файлы
Тип file. Могут хранить данные различных типов.
В двоичных файлах информация хранится в том виде, как она хранится в оперативной памяти, а в текстовых числовая
информация преобразуется к строковому виду и обратно, что занимает больше времени.
Файл называется
файлом произвольного доступа, если можно перейти к его iму элементу за время, не зависящее от размеров файла («за
константное время»), файлом последовательного доступа, если переход к iму элементу требует количество операций,
пропорциональное i («требует линейного времени»).
Все файлы, содержащие элементы разного размера могут иметь только последовательный доступ к элементам. Таковыми
являются:
текстовые, бестиповые файлы.
Типизированные файлы имеют произвольный доступ.
Понятие файловой переменной, файлового указателя
Перед тем, как работать с информацией в файле, его надо открыть. После этого можно выполнять операции чтения из файла
и записи в файл. По окончании работы его нужно закрыть.
Закрытый файл можно:
переименовывать перемещать копировать удалять
С каждым открытым файлом связан так называемый файловый указатель, который указывает на текущую позицию в файле.
Файловый указатель создается при открытии файла, и, как правило, устанавливается на 1й элемент файла.
После каждой операции чтения или записи файловый указатель продвигается вперед на размер считанных элементов.
Паскальпрограмма
2 способа открытия файла
- Reset(f) — открытие текстового файла на чтение, а двоичного — на чтение и запись; файл должен
существовать; файловый указатель — на начало файла; - Rewrite(f) — создание нового файла (если такого файла не существовало) или обнуление существующего;
файловый указатель — в начало;
текстовые файлы при этом открываются только на запись, а двоичные — на чтение и запись;
Функция Eof(f) [расшифровывается как End Of File] возвращает true, если файловый указатель находился за
концом файла.
После работы с данными в файле его необходимо закрыть с помощью Close(f).
Если, не закрывая, выполнить Reset(f), то файловый указатель просто перейдет к началу.
Буферизация в файлах
С каждым файлом связан некий буфер памяти, в который информация из файла частично считывается, и, из которого
записывается в нужную часть файла.
Наличие буфера ускоряет операции чтения и записи, поскольку они выполняются преимущественно не с внешним устройством
(файлом), а с участком оперативной памяти.
Если забыть закрыть файл, открытый на запись, то можно потерять лишь данные, сохраненные в буфере.
Подпрограммы для работы с закрытыми файлами
Типичные ошибки вводавывода при работе с файлами
- Файл открыли, но забыли выполнить Assign.
- Открыли, но файла нет на диске (или нет прав доступа на чтение).
- Попытка считывания за концом файла.
Все эти ошибки нужно обрабатывать с помощью исключений.
Пример 1. Файл не существует.
Оператор try..finally
Этот оператор отличается тем, что не обрабатывает исключение, а лишь выполняет некоторое завершающее действие,
которое должно быть совершено в любом случае. Для обработки нужен внешний блок try..except.
Пример 2. Попытка считывания за концом файла.
Контрольные вопросы
- Определение
- Классификация
- Понятие файловой переменной, файлового указателя
- 2 способа открытия файла
- Буферизация в файлах
Лекция№24 Файлы произвольного доступа. Создание структуры записи.
План лекции
Файлы произвольного доступа
Структура описания файловой переменной
Чтение элементов файла
Последовательность действий для создания и заполнения файла
Файл называется
файлом произвольного доступа, если можно перейти к его iму элементу за время, не зависящее от размеров файла («за
константное время»), файлом последовательного доступа, если переход к iму элементу требует количество операций,
пропорциональное i («требует линейного времени»).
Все файлы, содержащие элементы разного размера могут иметь только последовательный доступ к элементам
Прямой доступ к записям файла. В стандарте языка Паскаль до
пустим только последовательный доступ к элементам файла. Од
ной из дополнительных возможностей, реализованных в Турбо
Паскале, является прямой доступ к записям файла.
Как уже отмечалось, элементы файла пронумерованы в порядке
их занесения в файл, начиная с нуля. Задав номер элемента файла,
можно непосредственно установить на него указатель. После этого
можно читать или перезаписывать данный элемент. Установка ука
зателя на нужный элемент файла производится процедурой
Seek(FV,n)
Здесь F V — имя файловой переменной, п — порядковый номер
элемента. В следующем примере эта процедура будет использована. Пример 2. Имеется файл, сформированный программой из
пре
дыдущего примера. Пусть некоторые студенты пересдали экзамен
и получили новые оценки. Составить программу внесения резуль
татов переэкзаменовки в файл. Программа будет запрашивать но
мер студента в ведомости и его новую оценку. Работа заканчивает
ся, если вводится несуществующий номер (9999).
Пример требует некоторых пояснений. Список студентов в ве
домости пронумерован, начиная от 1, а записи в файле нумеру
ются от 0. Поэтому, если п — это номер в ведомости, то номер
соответствующей записи в файле равен п-1 . После прочтения
записи «номер п-1» указатель смещается к следующей п-й запи
си. Для повторного занесения на то же место исправленной запи
си повторяется установка указателя.
Файловый тип переменной — это структурированный тип, представляющий собой совокупность однотипных элементов,
количество которых заранее (до исполнения программы) не определено.
Структура описания файловой переменной:
Var <имя переменной>: Fil e Of <тип элемента>;
где <тип элемента> может быть любым, кроме файлового.
Например:
Var Fi: File Of Integer;
Fr: File Of Real;
Fc: File Of Char;
Файл можно представить как последовательную цепочку элементов (эл.), пронумерованных от 0, заканчивающуюся
специальным кодом, называемым маркером конца (<м. к.>):
Количество элементов, хранящихся в данный момент в файле, называется его текущей длиной. Существует специальная
ячейка памяти, которая хранит адрес элемента файла, предназначенного для текущей обработки (записи или чтения). Этот
адрес называется указателем или окном файла.
Для того чтобы начать запись в файл, его следует открыть для записи. Это обеспечивает процедура Rewrite (FV) ; где FV
— имя файловой переменной. При этом указатель устанавливается на начало файла. Если в файле есть информация, то она
исчезает.
Схематически выполнение процедуры Rewrite можно представить так:
Стрелка внизу отмечает позицию указателя.
Запись в файл осуществляется процедурой Write (FV, V); где v — переменная того же типа, что и файл FV. Запись
происходит туда, где установлено окно (указатель). Сначала записывается значение, затем указатель смещается в
следующую позицию. Если новый элемент вносится в конец файла, то сдвигается маркер конца.
Схема выполнения оператора:
Пример 1. В файловую переменную Fx занести 20 вещественных
чисел, последовательно вводимых с клавиатуры.
Для чтения элементов файла с его начала следует открыть файл для чтения. Это делает процедура Reset (FV). В
результате указатель устанавливается на начало файла. При этом вся информация в файле сохраняется. Схема выполнения
процедуры:
Чтение из файла осуществляется процедурой Read (FV, v) ; гдеv — переменная того же типа, что и файл FV. Значение
текущегоэлемента файла записывается в переменную v; указатель смещается к следующему элементу.
Пример 1. В файловую переменную Fx занести 20 вещественных
чисел, последовательно вводимых с клавиатуры.
Для чтения элементов файла с его начала следует открыть файл для чтения. Это делает процедура Reset (FV). В
результате указатель устанавливается на начало файла. При этом вся информация в файле сохраняется. Подведем итог сказанному. Для создания и заполнения файла
требуется следующая последовательность действий:
1. Описать файловую переменную.
2. Описать переменную того же типа, что и файл.
3. Произвести назначение (Assign).
4. Открыть файл для записи (Rewrite).
5. Записать в файл данные (Write).
6. Закрыть файл (Close).
Ее результат — целое число, равное текущей длине файла.
Замечание: согласно стандарту Паскаля в файл, открытый опе
ратором Rewrite, можно только записывать информацию, а файл,
открытый оператором Reset , можно использовать только для чте
ния. В Турбо Паскале допускается запись (Write) в файл, откры
тый для чтения (Reset). Это создает определенные удобства для
модификации файлов.
Контрольные вопросы
Тема 4.6. Указатели
Лекция№ 25. Указатели и применение динамически распределяемой
памяти.
План лекции
- Определение указателя
- Типы указателей
- Бестиповые указатели
- Динамическая память
Оперативная память состоит из последовательный ячеек. Каждая ячейка имеет номер, называемый адресом. В 32битных
системах можно адресовать 232 байт ( 4Гб) памяти, в 64битных — 2 64
соответственно.
Переменная (или константа), хранящая адрес, называется указателем.
Для чего нужны указатели
Указатели повышают гибкость доступа к данным:
Вместо самих данных можно хранить указатель на них. Это позволяет хранить данные в одном экземпляре и множество
указателей на эти данные.
Через разные указатели эти данные можно обновлять (пример — корпоративная БД).
Указателю можно присвоить адрес другого объекта (вместо старого появился новый телефонный справочник).
С помощью указателей можно создавать сложные структуры данных.
Типы указателей
Указатели делятся на:
1.Типизированные (указывают на объект некоторого типа)
Имеют тип: ^<тип> Пример. ^integer — указатель на integer
2.Бестиповые (хранят адрес ячейки памяти неизвестного типа)
Преимущество: могут хранить что угодно Имеют тип: pointer Пример кода.
@ — унарная операция взятия адреса
Операция разадресации (разыменования)
^ — операция разыменования pi^ — то, на что указывает pi, т.е. другое имя i или ссылка на i.
Тут надо вспомнить определение ссылки: Ссылка — другое имя объекта.
Нулевой указатель
Все глобальные неинициализированные указатели хранят специальное значение nil, что говорит о том, что они
никуда не указывают. Указатель, хранящий значение nil называется нулевым.
Попытка разыменовать нулевой указатель приводит к ошибке времени выполнения.
Бестиповые указатели
Бестиповому указателю можно присвоить адрес переменной любого типа, т.е. бестиповой указатель совместим по
присваиванию с любым типовым указателем.
Попытка разыменовать бестиповой указатель приводит к ошибке компиляции. Т.е. он может только хранить адреса.
Оказывается, любой типизированный указатель совместим по присваиванию с бестиповым, т.е. следующий код верен:
Вопрос. Нельзя ли интерпретировать память, на которую указывает p, как принадлежащую к определенному типу?
Ответ — да, можно. Вот как это сделать:
Доступ к памяти, имеющей другое внутреннее представление
Замечание. Важно, что типы real и Rec имеют один размер.
Переменная типа динамический массив является указателем на данные массива, хранящиеся в динамической памяти.
Динамическая память
Особенности динамической памяти
Память, принадлежащая программе, делится на:
1.Статическую (память, занимаемая глобальными переменными и константами)
2. Автоматическую (память, занимаемая локальными данными, т.е. стек программы)
3.Динамическую (память, выделяемая программе по специальному запросу)
В дополнение к статической и автоматической памяти, которые фиксированы после запуска программы, программа может
получать нефиксированное количество динамической памяти. Ограничения на объём выделяемой динамической памяти связаны
лишь с настройками операционной системы и объемом оперативной памяти компьютера.
Основная проблема — явно выделенную динамическую память необходимо возвращать, иначе не хватит памяти другим
программам.
Для явного выделения и освобождения динамической памяти используются процедуры:
По окончании работы программы, вся затребованная программой динамическая память возвращается ОС.
Но лучше освобождать динамическую память явно! Иначе в процессе работы программы она может занимать большие объёмы
(ещё не освобождённой) памяти, что вредит общей производительности системы. Ошибки при работе с динамической
памятью
1.
Ошибка разыменования нулевого указателя (попытка использовать невыделенную динамическую память).
2.
Утечка памяти (память, которая выделилась в результате первого вызова New(p), принадлежит программе, но не
контролируется никаким указателем.
2a.
Утечка памяти в подпрограмме: обычно если динамическая память выделяется в подпрограмме, то она должна в этой же
подпрограмме возвращаться. Исключение составляют т.н. «создающие» п/п:
Ответственность за удаление памяти, выделенной в подпрограмме, лежит на программисте, вызвавшем эту подпрограмму.
Out of Memory (очень большие утечки памяти, в результате которых динамическая память может «исчерпаться»).
Контрольные вопросы
- Определение указателя
- Типы указателей
- Бестиповые указатели
- Динамическая память
Тема 4.6
Лекция№26. Структуры данных на основе указателей.
План лекции
- Определение ДСД
- Виды ДСД
- Стеки
- Очереди
- Линейные списки
Динамические структуры данных – связные структуры данных, память под которые выделяется (и освобождается при
необходимости) динамически в процессе работы программы. В качестве динамически объединяемых элементов выступают
всегда объекты структурного типа данных, поскольку основной принцип формирования динамических структур — наличие в
объекте полей двух типов: один для хранения полезной информации (информационный блок), другой для связывания
структур между собой (адресный блок). На языке Си структурный тип для объектов, объединяемых в динамическую
структуру, определяется следующим образом:
struct Node
{ … //информационный блок
Node *field1, *field2, *mas[10];//адресный блок
};
Здесь информационный блок включает в себя полезную информацию, хранящуюся в элементе, а адресный блок может включать
в себя набор указателей, позволяющих связать отдельные элементы между собой в единый список, дерево, очередь и
другие типы динамических структур. Количество указателей и способ их интерпретации могут существенно отличаться для
различных типов динамических структур. К динамическим структурам относятся:
- однонаправленные (односвязные) списки;
- двунаправленные (двусвязные) списки;
- многосвязные списки;
- стеки;
- деки;
- очереди;
- циклические списки (кольца);
- бинарные деревья;
- графы.
Рассмотрим основные правила работы с динамическими структурами данных типа стек, очередь и список, базируясь на
приведенное описание компоненты.
СТЕКИ
Стеком называется динамическая структура данных, добавление компоненты в которую и исключение компоненты из которой
производится из одного конца, называемого вершиной стека. Стек работает по принципу
LIFO (Last-In, First-Out) —
поступивший последним, обслуживается первым.
Обычно над стеками выполняется три операции:
-начальное формирование стека (запись первой компоненты);
-добавление компоненты в стек;
-выборка компоненты (удаление).
Для формирования стека и работы с ним необходимо иметь две переменные типа указатель, первая из которых определяет
вершину стека, а вторая — вспомогательная. Пусть описание этих переменных имеет вид:
var pTop, pAux: Pointer;
где pTop — указатель вершины стека;
pAux — вспомогательный указатель.
ОЧЕРЕДИ
Очередью называется динамическая структура данных, добавление компоненты в которую производится в один конец, а
выборка осуществляется с другого конца. Очередь работает по принципу:
FIFO (First-In, First-Out) —
поступивший первым, обслуживается первым.
Для формирования очереди и работы с ней необходимо иметь три переменные типа указатель, первая из которых определяет
начало очереди, вторая — конец очереди, третья — вспомогательная.
Описание компоненты очереди и переменных типа указатель дадим сле-
дующим образом:
type
PComp=^Comp;
Comp=record
D:T;
pNext:PComp
end;
var
pBegin, pEnd, pAux: PComp;
где pBegin — указатель начала очереди, pEnd — указатель конца очереди, pAux — вспомогательный указатель.
Тип Т определяет тип данных компоненты очереди.
ЛИНЕЙНЫЕ СПИСКИ
Виды списков
Линейный односвязный список
Циклический односвязный список
Двусвязный линейный список
Циклический двусвязный список
В стеки или очереди компоненты можно добавлять только в какой — либо один конец структуры данных, это относится и к
извлечению компонент.
Связный (линейный) список является структурой данных, в произвольно выбранное место которого могут включаться
данные, а также изыматься оттуда.
Каждая компонента списка определяется ключом. Обычно ключ — либо число, либо строка символов. Ключ располагается в
поле данных компоненты, он может занимать как отдельное поле записи, так и быть частью поля записи.
Основные отличия связного списка от стека и очереди следующие:
-для чтения доступна любая компонента списка;
-новые компоненты можно добавлять в любое место списка;
-при чтении компонента не удаляется из списка.
Для формирования списка и работы с ним необходимо иметь пять пере-
менных типа указатель, первая из которых определяет начало списка,
вторая — конец списка, остальные- вспомогательные.
Описание компоненты списка и переменных типа указатель дадим сле-
дующим образом:
type
PComp= ^Comp;
Comp= record
D:T;
pNext:PComp
end;
var
pBegin, pEnd, pCKey, pPreComp, pAux: PComp;
где pBegin — указатель начала списка, pEnd — указатель конца списка,
pCKey, pPreComp, pAux — вспомогательные указатели.
Контрольные вопросы
- Определение ДСД
- Виды ДСД
- Стеки
- Очереди
- Линейные списки
Раздел 5. Объектно-ориентированное программирование
Тема 5.1 Основные принципы объектно-ориентированного
программирования (ООП)
Лекция№27. Базовые понятия ООП. Основные принципы ООП.
План лекции
- Концепция ООП
- Принципы ООП
- Элементы ООП
- История
Объе́ктно-ориенти́рованное программи́рование (ООП) — методология программирования, основанная на
представлении программы в виде совокупности объектов, каждый из которых является экземпляром определенного
класса[1], а классы образуют иерархию наследования
Необходимо обратить внимание на следующие важные части этого определения: 1) объектно-ориентированное
программирование использует в качестве основных логических конструктивных элементов объекты, а не
алгоритмы; 2) каждый объект является экземпляром определенного класса; 3) классы
образуют иерархии. Программа считается объектно-ориентированной, только если выполнены все три указанных требования.
В частности, программирование, не использующее наследование, называется не объектно-ориентированным, а
программированием с помощью абстрактных типов данных
Концепция объектно-ориентированного программирования
Концепцию ООП характеризует следующее:
- В качестве строительных блоков разрабатываемых приложений используются объекты.
- Каждому классу соответствует некоторый объектный тип, представляющий собой совокупность элементов данных и
методов (для операций над данными), скомпонованных вместе для удобства использования. - Каждый объект – переменная, являющаяся представителем (экземпляром) определённого класса.
- Классы связаны друг с другом соотношениями, с помощью которых объекты могут расширяться; при этом описания
существующих объектов могут многократно использоваться при описании новых объектов. - Представителями класса могут быть представители, как непосредственного класса, так и любого класса предка.
Концепция ООП базируется на трёх основных принципах:
инкапсуляция;
полиморфизм;
наследование.
Инкапсуляция
Инкапсуляция – объединение данных и действий над ними в одном объектном типе.
Наследование
Наследование – это способность одного класса использовать характеристики (описание) другого. Наследование
устанавливает между двумя классами отношение «предок – потомок». Предок – это класс, предоставляющий свои
возможности и характеристики (описание) другим классам через механизм наследования. Класс, который использует
характеристики класса посредством наследования, называется потомком. Непосредственный предок, от которого происходит
данный класс, называется родителем.
В Object Pascal используется модель простого наследования, т.е. класс-потомок может иметь только одного родителя,
класс-предок может иметь несколько потомков.
Полиморфизм
Полиморфизм – это возможность определения единого по имени действия (метода в виде процедуры или функции),
применимого ко всем объектам иерархии наследования, т.е. возможность иметь несколько методов с одним и тем же именем
для различных объектов одной иерархии. Это средство для развития объектов в потомках. Оно реализуется тем, что
объект-потомок может добавлять и переопределять методы, т.е. заменять методы предка на новые с теми же именами.
Класс
Класс является описываемой на языке терминологии исходного кода моделью ещё не существующей сущности (объекта).
Фактически он описывает устройство объекта, являясь своего рода чертежом. Говорят, что объект — это экземпляр класса.
При этом в некоторых исполняющих системах класс также может представляться некоторым объектом при выполнении
программы посредством динамической
идентификации типа данных. Обычно классы разрабатывают таким образом, чтобы их объекты соответствовали
объектам предметной области.
Объект
Сущность в адресном пространстве вычислительной системы,
появляющаяся при создании экземпляра класса (например, после запуска результатов компиляции исвязывания исходного кода на выполнение)
История
ООП возникло в результате развития идеологии процедурного
программирования, где данные и подпрограммы (процедуры, функции) их обработки формально не связаны. Для
дальнейшего развития объектно-ориентированного программирования часто большое значение имеют понятия события (так
называемое событийно-ориентированное
программирование) и компонента (компонентное
программирование, КОП).
Взаимодействие объектов происходит посредством сообщений. Результатом дальнейшего развития ООП, по-видимому, будет агентно-ориентированое программирование,
где агенты — независимые части кода на уровне выполнения. Взаимодействие агентов происходит посредством
изменения среды, в которой они находятся.
Языковые конструкции, конструктивно не относящиеся непосредственно к объектам, но сопутствующие им для их безопасной
(исключительные ситуации, проверки) и эффективной
работы, инкапсулируются от них в аспекты (в аспектно-ориентированном
программировании). Субъектно-ориентированное
программирование расширяет понятие объекта посредством обеспечения более унифицированного и независимого
взаимодействия объектов. Может являться переходной стадией между ООП и агентным программированием в части
самостоятельного их взаимодействия.
Первым языком программирования, в котором были предложены основные понятия, впоследствии сложившиеся в парадигму,
была Симула, но термин «объектная ориентированность» не
использовался в контексте использования этого языка. Взгляд на программирование «под новым углом» (отличным от
процедурного) предложили Алан Кэй и Дэн Ингаллс в
языке Smalltalk. Здесь понятие класса стало основообразующей
идеей для всех остальных конструкций языка (то есть класс в Смолтоке является примитивом, посредством которого
описаны более сложные конструкции).
Наиболее распространённые в промышленности языки (С++, Delphi, C#, Java и др.) воплощают объектную модель Симулы.
Примерами языков, опирающихся на модель Смолтока, являются Python, Ruby.
Контрольные вопросы
- Концепция ООП
- Принципы ООП
- Элементы ООП
4. История
Тема 5.1
Лекция№28. Классы объектов. Компоненты и их свойства.
План лекции
- Определение класса
- Объявление класса
- Поля
- Свойства
- Палитра компонентов
Класс – это структура языка, включающая, помимо описания данных, описание процедур и функций, которые могут
быть выполнены над представителем класса – объектом.
Переменные в зависимости от предназначения именуются полями или свойствами. Процедуры и функции класса – методами.
Соответствующий классу тип называется объектным типом.
Пример объявления простого класса:
type
TPerson = class (TObject)
private
fname: string[15]; faddress: string[35];
public
procedure Show;
end;
TPerson – это имя класса, fname и faddress – имена полей, show – имя метода.
Согласно принятому в Delphi соглашению, имена полей должны начинаться с буквы f (от слова field – поле).
Описание класса помещают в программе в раздел описания типов (type).
Тексты на Object Pascal представляются в виде модулей (pas – файлов). Классы могут быть объявлены в секции интерфейса
(interface) или в секции реализации (implementation) модуля. Определение классов внутри подпрограмм и других блоков
не допускается.
Объект или экземпляр объекта – это конкретный экземпляр, созданный в соответствии с его объявлением с помощью
класса.
Объект может содержать другой объект. Например, объект формы может содержать объект кнопки.
Объект может быть ассоциирован с другим объектом, т.е. содержать ссылку на другой объект. Он может вызывать
его методы и использовать его поля.
Переменная типа класс – это переменная объектного типа и называется экземпляром класса или объектом.
Объекты как представители класса объявляются в программе в разделе var, например:
Var student: TPerson; professor: TPerson;
Поля
Поля – это данные, уникальные для каждого экземпляра класса. Они предназначены для хранения данных во время
работы экземпляра класса (объекта). Поле объявляется как обычная переменная и может быть любого типа. В описании
класса поля должны предшествовать методам и свойствам.
При создании новых классов на базе ранее созданных класс-потомок наследует все поля класса-родителя. Удаление и
переопределение полей невозможно. Допускается добавление новых полей.
Поля и методы у разных объектов одного типа одни и те же. Методы – это процедуры и функции, определённые внутри
класса и предназначенные для операций над полями и свойствами.
В Delphi у всех стандартных классов все поля недоступны и заменены базирующимися на них свойствами. Свойства
компонентов Delphi определяют их внешний вид и поведение.
Свойства
Свойства – это высокоуровневые атрибуты компонентов класса.
В объектах Delphi пользователь полностью отгорожен от полей объекта с помощью свойств. Внешне свойства напоминают
поля, но внутри содержат методы, обеспечивающие доступ к свойствам. При каждом обращении к такому методу выполняются
соответствующие действия, т.е. к свойствам класса доступ возможен только через методы.
Каждому свойству соответствует поле, содержащее значение свойства, и два метода, обеспечивающих доступ к значению
поля.
Методы
Метод – это подпрограмма (процедура или функция), которая определена как элемент класса. Описание метода
аналогично описанию обычной подпрограммы модуля. Внутри одного класса можно объявить столько методов, сколько надо.
Компоненты и палитра компонентов
Палитра компонентов – это каталог, состоящий из визуальных и невизуальных компонентов. Компонент –
это структурная единица Delphi. Основу ООП в Delphi составляет набор компонентов, который позволяет Delphi с помощью
компонентов использовать множество возможностей, присущих Windows.
Окно формы – это окно Windows.. Сама форма также является компонентом. Новая форма, которая создаётся при загрузке
Delphi или при создании нового проекта, является главной формой приложения.
Палитра компонентов расположена в правой части главного окна и имеет вид многостраничного блокнота, где на каждой
странице размещён набор пиктограмм её компонентов. Активизировать группу компонентов требуемой страницы надо щелчком
мыши на её закладке.
Компоненты Delphi
Библиотека визуальных компонентов (Visual Component Library — VCL) Delphi содержит множество предопределенных типов
компонентов, из которых пользователь может строить свою прикладную программу. Палитра компонентов расположена справа
в полосе инструментальных панелей интегрированной среды разработки Delphi.
Поскольку число страниц в палитре велико и не все закладки видны на экране одновременно, в правой части палитры
компонентов имеются две кнопки со стрелками, направленными влево и вправо. Эти кнопки позволяют перемещать
отображаемую на экране часть палитры.
Имена компонентов, соответствующих той или иной пиктограмме, можно узнать из всплывающей подсказки, появляющейся,
если задержать над этой пиктограммой курсор мыши. Если выбрать в палитре компонент и нажать клавишу F1, то
отобразится справка по типу данного компонента.
Имена на ярлычках выглядят, например, так: MainMenu, Button и т.д. Однако, в
Delphi все имена классов в действительности начинаются с символа «Т», например, TMainMenu, TButton.
Все компоненты системы Delphi можно разделить на:
визуальные, т.е. те, которые будут представлены на форме во время выполнения приложения в том же виде, что и
при разработке приложения (например, кнопки, метки и пр.);
не визуальные, т.е. те, вид которых во время выполнения приложения не совпадает с тем, что представлено во
время разработки (например, меню, окна диалога).
Контрольные вопросы
- Определение класса
- Объявление класса
- Поля
- Свойства
- Палитра компонентов
Тема 5.2 Интегрированная среда разработчика
Лекция№29. Интерфейс среды разработчика: основные окна,
интегрированной среде
План лекции
- Определение
- Преимущества интегрированной среды
3. Окна интегрированной среды
Интерфейс (interface) – это средства взаимодействия, средства связи, сопряжения, согласования. Этим
термином в информатике обозначают довольно широкий круг понятий:
- физический (аппаратный) интерфейс (на уровне электронных компонентов),
- интерфейс программиста (комплекс правил и соглашений о стыковке программных модулей),
- интерфейс пользователя как набор средств диалога, взаимодействия программы (машины) с человеком.
Интегрированная среда разработки — это совокупность программных средств, поддерживающая все этапы разработки
программного обеспечения от написания исходного текста программы до ее компиляции и отладки, и обеспечивающая
простое и быстрое взаимодействие с другими инструментальными средствами (программным
отладчиком-симулятором, внутрисхемным
эмулятором, эмулятором
ПЗУ и программатором).
.
При традиционном подходе, начальный этап написания программы строится следующим образом:
Исходный текст набирается при помощи какого-либо текстового редактора. По завершении набора, работа с текстовым
редактором прекращается и запускается кросс компилятор. Как правило, вновь написанная программа содержит
синтаксические ошибки, и компилятор сообщает о них на консоль оператора.
Вновь запускается текстовый редактор, и оператор должен найти и устранить выявленные ошибки, при этом сообщения о
характере ошибок выведенные компилятором уже не видны, так как экран занят текстовым редактором.
И этот цикл может повторяться не один раз. Если программа имеет большой объем, собирается из различных частей, и
подвергается длительному редактированию или модернизации, то даже этот начальный этап может потребовать много сил и
времени. После этого наступает этап отладки программы и к редактору с компилятором добавляется эмулятор или
симулятор, за работой которого хотелось бы следить прямо по тексту программы в текстовом редакторе.
Избежать большого объема однообразных действий и тем самым существенно повысить эффективность процесса разработки и
отладки позволяют т.н. интегрированные среды (оболочки) разработки (Integrated Development Environment, IDE).
Работа в интегрированной среде дает программисту:
Возможность использования встроенного многофайлового текстового редактора, специально ориентированного на работу с
исходными текстами программ;
Диагностика выявленных при компиляции ошибок, и исходный текст программы, доступный редактированию, выводятся
одновременно в многооконном режиме;
Возможность организации и ведения параллельной работы над несколькими проектами. Менеджер проектов позволяет
использовать любой проект в качестве шаблона для вновь создаваемого проекта;
Перекомпиляции подвергаются только редактировавшиеся модули;
Возможность загрузки отлаживаемой программы в имеющиеся средства отладки, и работы с ними без выхода из оболочки;
Возможность подключения к оболочке практически любых программных средств.
В последнее время, функции интегрированных сред разработки становятся стандартной принадлежностью программных
интерфейсов эмуляторов и отладчиков-симуляторов.
Окно Конструктора формы первоначально находится в центре экрана и имеет заголовок В нем выполняется проектирование
формы, для чего на форму из Палитры компонентов помещаются необходимые компоненты. При этом проектирование
заключается в визуальном конструировании формы, а работа разработчика похожа на работу в среде простого графического
редактора. Сам Конструктор формы во время ее проектирования остается «за кадром», и разработчик имеет дело
с самой формой, поэтому часто окно Конструктора также называют окном формы или просто формой.
Окно Редактора кода (заголовок Unitl.pas) после запуска системы программирования находится под окном Конструктора
формы и почти полностью перекрывается им. Редактор кода (Редактор) представляет собой обычный текстовый редактор, с
помощью которого можно редактировать текст модуля и другие текстовые файлы приложения, например, файл проекта.
Каждый редактируемый файл находится в окне Редактора на отдельной странице, доступ к которой осуществляется щелчком
на соответствующем ярлычке. Первоначально в окне Редактора кода на странице Code содержится одна закладка исходного
кода модуля формы разрабатываемого приложения.
Переключение между окнами Конструктора формы и Редактора кода удобно выполнять с помощью клавиши <F12>.
Окно Проводника кода (Exploring Unitl.pas) пристыковано слева от окна Редактора кода. В нем в виде дерева
отображаются все объекты модуля формы, например, переменные и процедуры (рис. 1.2). В окне Проводника кода можно
удобно просматривать объекты приложения и быстро переходить к нужным объектам, что особенно важно для больших
модулей. Вызов окна Проводника кода выполняется по команде Code Explorer (Проводник кода) меню View (Вид).
Окно Инспектора объектов находится в левой части экрана и отображает свойства и события объектов для текущей формы
Formi. Его можно вызвать на экран командой View | Object Inspector (Просмотр | Инспектор объектов) или нажатием
клавиши <F11>.
Окно Инспектора объектов имеет две страницы: Properties (Свойства) и Events (События).
Страница Properties отображает информацию о текущем (выбранном) компоненте в окне Конструктора формы и при
проектировании формы позволяет удобно изменять многие свойства компонентов.
Страница Events определяет процедуру, которую компонент должен выполнить при возникновении указанного события. Если
для какого- либо события существует процедура, то в процессе выполнения приложения при возникновении этого события
процедура вызывается автоматически. Такие процедуры служат для обработки соответствующих событий, поэтому их
называют процедурами-обработчиками или обработчиками. Отметим, что события также являются свойствами, которые
указывают на свои обработчики.
В конкретный момент времени Инспектор объектов отображает свойства и события текущего (выбранного) компонента, имя и
тип которого отображаются в списке под заголовком окна Инспектор объектов. Компонент, расположенный на форме, можно
выбрать щелчком мыши на нем или выбором в списке Инспектора объектов. Каждый компонент имеет свой набор свойств и
событий, определяющих его особенности.
Начиная с 4-й версии, Delphi поддерживает технологию Dock-окон, которые могут стыковаться (соединяться) друг с другом
с помощью мыши. Стыкующимися окнами являются инструментальные (не диалоговые) окна интегрированной среды разработки,
в том числе окна Инспектора объектов и Проводника кода. Со стыкованными окнами удобнее выполнять такие операции, как
перемещение по экрану или изменение размеров.
Для соединения двух окон следует с помощью мыши поместить одно из них на другое, и после изменения вида рамки
перемещаемого окна отпустить его, после чего это окно автоматически пристыкуется сбоку от второго окна. Разделение
окон выполняется перемещением пристыкованного окна за двойную линию, размещенную под общим заголовком. После
соединения окна представляют собой одно общее окно, разделенное на несколько частей. При стыковке/расстыковке окно
изменяет свое название. Так, окно Проводника кода, состыкованное с окном Редактором кода, имеет общее с ним
название, например, Unitl.pas, в то время как при отстыковке название изменяется на Exploring Unitl.pas. Окна
Инспектора объектов и Обозревателя дерева объектов при стыковке объединяют свои названия (через запятую указываются
названия каждого
Скрытое окно вызывается на экран командой пункта View (Просмотр) главного меню. Например, окно Проводника кода
выводится на экран командой View | Code Explorer (Просмотр | Проводник кода).
Контрольные вопросы
- Определение
- Преимущества интегрированной среды
3. Окна интегрированной среды
Тема 5.3 Этапы разработки приложения
Лекция№30. Проектирование, тестирование и отладка приложения.
План лекции
1.Определение
2. Создание приложения в среде Delphi
3. Создание приложения в среде Visual Basic
Процесс разработки новых приложений состоит из 4-х основных этапов:
1. Проектирование. Определяются цели и
задачи, способы их решения, а также определяется структура данных и язык программирования, на котором будет написано
приложение.
2. Создание интерфейса. В программную среду разработки вводятся необходимые управляющие элементы:
кнопки, текстовые поля, флажки, переключатели и другие элементы.
3. Отладка. Все управляющие элементы
связываются программным кодом и путем ввода конкретных значений происходит проверка работоспособности кода и
отлавливание возможных ошибок. Логические ошибки самые коварные в этом плане. Этот этап по времени самый длительный.
4.
Заключительный этап. Идет компиляция кода и создание дистрибутива. Компиляция — процесс перевода программного кода в
машинный язык, понятный каждому компьютеру. Здесь же идет подключение необходимых программных библиотек для полной
работоспособности приложения. На выходе получаем законченный продукт — файл с расширением «.ехе».
Создание приложения в среде Delphi можно условно разделить на несколько этапов:
1. Создание графического интерфейса будущего приложения
С помощью Панели инструментов на форму помещаются управляющие элементы, которые должны обеспечить взаимодействие
приложения с пользователем.
2. Задание значений свойств объектов графического интерфейса
С помощью окна «Свойства объекта» задаются значения свойств управляющих элементов, помещенных ранее на форму.
3. Создание и редактирование программного кода
Для создания заготовки событийной процедуры необходимо осуществить двойной щелчок мышью по управляющему элементу. В
окне «Редактор кода» появится заготовка событийной процедуры, имя которой состоит из двух частей: имени формы,
содержащий управляющий элемент, и имени объекта и имени события (например,TForm1.Button1Click). Затем в окне
«Редактор кода» производится ввод и редактирование программного кода процедуры.
4. Сохранение проекта
Т.к. проект включает в себя несколько файлов, рекомендуется для каждого проекта создать отдельную папку на диске.
Сохранение проекта производится с помощью меню File:
– Сначала необходимо сохранить форму и связанный с ней программный модуль (файл с расширением pas) с помощью команды
Save As.… По умолчанию для файла формы предлагается имя Unit1.pas.
– Далее необходимо сохранить файл главного модуля, который содержит описание проекта (файл с расширением dpr) с
помощью команды Save Project As…
– В процессе сохранения в папку проекта записываются вспомогательные файлы: файл с расширением res, описывающий
ресурсы; файл с расширением dfm, описывающий форму, и некоторые другие файлы.
5. Компиляция проекта в приложение
Сохраненный проект может выполняться только в самой системе программирования Delphi. Для того чтобы
преобразовать проект в приложение, которое может выполняться непосредственно в среде операционной системы,
необходимо сохранить проект в исполнимом файле (типа exe). Для компиляции проекта в исполнимый файл используется
команда [Project-Compile].
Среда разработки Delphi ориентирована на создание самых разнообразных приложений баз данных. Это и небольшие
локальные программы, и многоуровневые распределенные системы, использующие новейшие технологии. Но, несмотря на
сложность приложения, в его основе всегда лежит базовый механизм обеспечения доступа к данным. Он создается триадой
компонентов:
Компонент Набора данных (TTable, TQuery, TStoredProc);
компонент TDataSource;
один или несколько компонентов отображения данных..
В целом, благодаря компонентному подходу, разработка простого приложения баз данных оказывается ничуть не сложнее
создания обычного приложения. Достаточно разместить на форме несколько компонентов, настроить их свойства и вы
получаете работающее приложение баз данных.
.
Интегрированная среда разработки Visual Basic
Visual Basic — это система программирования, предназначенная для написания программ, работающих под
управлением операционной системы Windows. Используя Visual Basic, можно разрабатывать очень сложные приложения
практически для любой области современных компьютерных технологий: бизнес-приложения, игры, мультимедиа, базы данных
В Visual Basic реализована модель объектно-ориентированного событийно — управляемого программирования.
Программы, работающие в среде Windows, называются приложениями. На этапе разработки приложения в среде Visual Basic
называются проектами. Проект содержит набор взаимодействующих форм, программных модулей, модуль проекта и
вспомогательные файлы.
Форма – основной элемент внешнего интерфейса проекта, аналог окна Windows. Она имеет строку заголовка с кнопками
управления и системным меню, а также возможности управления мышью. Все это не надо программировать, форма,
включённая в проект, обладает этими свойствами. Обычно в проекте бывает несколько форм.
. Обычно в проекте один программный модуль, он хранится в файле с расширением bas.
.
Этапы разработки приложения в среде Visual Basic
Создание интерфейса (этап проектирования)
На этом этапе необходимо сначала продумать ожидаемый алгоритм работы пользователя с будущим приложением, возможные
события при работе приложения, порядок их возникновения. Кроме того, надо выявить будущих пользователей, максимально
точно описать предъявляемые ими требования к проекту. Тогда можно приступать непосредственно к проектированию, но и
здесь вопросов много: какие меню понадобятся, какого размера окно требуется для приложения, сколько всего будет
окон, должен ли пользователь иметь возможность изменять размеры окна, какие элементы управления рационально
использовать для решения задачи и как их разместить максимально удобно для пользователя?… В результате на форму с
помощью мыши помещаются те или иные управляющие элементы, а в окне Properties задаются их свойства. После того как
разработка интерфейса завершена, кнопки управления, поля и другие элементы, размещенные на форме, автоматически
будут распознавать действия пользователя, такие, например, как движение мыши или щелчок ее кнопки.
Создание программного кода
Теперь начинается процесс, похожий на традиционное программирование: создание программного кода для активизации
визуального интерфейса, подготовленного на первом этапе. Смысл состоит в том, что объекты Visual Basic только
распознают события типа щелчков мыши, а то, как они реагируют на эти события, зависит от программного кода,
написанного программистом. При этом, алгоритмическая часть Visual Basic практически ничем не отличается от
традиционного языка Basic.
Для создания или редактирования кода надо вызвать окно Редактора кода. Окно используется для размещения, просмотра и
редактирования всех текстовых элементов программы — описания констант, переменных, массивов, процедур и пр.
Отладка, тестирование, компиляция
Для выполнения отладки (проверки правильности функционирования проекта и исправления найденных ошибок) в Visual Basic
существует набор специальных инструментов, в первую очередь команды меню Debug. Отладка и тестирование – непременные
этапы работы над любым проектом, особенно большим и сложным. Проект может функционировать лишь в среде Visual Basic.
После его создания, отладки и тестирования выполняется компиляция: создается исполняемый модуль с расширением exe,
независимый от среды Visual Basic. Так получается готовое приложение.
Контрольные вопросы
. 1.Определение
2. Создание приложения в среде Delphi
3. Создание приложения в среде Visual Basic
Тема 5.4 Иерархия классов
Лекция№31. Классы объектно-ориентированного языка
программирования.
План лекции
1.Определение класса ООП
2.Виды классов
3. Отношения между классами
Иерархия классов в информатике означает
классификацию объектных типов, рассматривая объекты как реализацию классов (класс похож на заготовку, а
объект — это то, что строится на основе этой заготовки) и связывая различные классы отношениями наподобие
«наследует», «расширяет», «является его абстракцией», «определение интерфейса».
Класс — разновидность абстрактного
типа данных в объектно-ориентированном
программировании (ООП), характеризуемый способом своего построения. Класс определяет одновременно как интерфейс,
так и реализацию для всех своих экземпляров, а вызов метода-конструктора обязателен.
В объектно-ориентированной программе с применением классов каждый объект является «экземпляром» некоторого
конкретного класса, и других объектов не предусмотрено. То есть «экземпляр класса» в данном случае означает
не «пример некоторого класса» или «отдельно взятый класс», а «объект, типом которого является какой-то класс». В
современных объектно-ориентированных
языках программирования (в том числе в php, Java, C++, Oberon, Python, Ruby, Smalltalk, Object Pascal) создание класса сводится к написанию
некоторой структуры, содержащей набор полей и методов (среди последних особую роль играют конструкторы, деструкторы,
финализаторы). Практически класс может пониматься как некий шаблон, по которому создаются объекты — экземпляры
данного класса. Все экземпляры одного класса созданы по одному шаблону, поэтому имеют один и тот же набор полей и
методов.
Например, абстрактный тип данных «строка текста» может быть оформлен в виде класса, и тогда все строки текста в
программе будут являться объектами — экземплярами класса «строка текста».
. Сам класс в итоге определяется как список своих членов, а именно полей (свойств) и методов/функций/процедур. В зависимости от
языка программирования к этому списку могут добавиться константы, атрибуты и внешние определения.
Как и структуры, классы могут задавать поля — то есть переменные, принадлежащие либо непосредственно самому классу
(статические), либо экземплярам класса (обычные).
В ООП при использовании классов весь исполняемый код программы (алгоритмы) будет оформляться в виде так называемых
«методов», «функций» или «процедур», что соответствует обычному структурному программированию, однако
теперь они могут (а во многих языках обязаны) принадлежать тому или иному классу.
Как и поля, код в виде методов/функций/процедур, принадлежащих классу, может быть отнесен либо к самому классу, либо
к экземплярам класса. Метод, принадлежащий классу и соотнесенный с классом (статический метод) может быть вызван сам
по себе и имеет доступ к статическим переменным класса. Метод, соотнесенный с экземпляром класса (обычный метод),
может быть вызван только у самого объекта, и имеет доступ как к статическим полям класса, так и к обычным полям
конкретного объекта (при вызове этот объект передастся скрытым параметром метода). Объектно-ориентированный подход
за время своего развития накопил множество рекомендаций (паттернов) по созданию классов и иерархий классов.
Виды классов
1.Базовый (родительский) класс (суперкласс) —
класс, на основе которого создаются другие
классы. Классы, полученные на основе суперкласса, называются дочерними классами, производными
классами или подклассами.
Суперкласс позволяет создавать обобщенный интерфейс, заключающий в себе настраиваемую функциональность за счет
использования виртуальных функций.
Механизм суперклассов широко используется в объектно-ориентированном
программировании благодаря возможности
повторного использования, что достигается благодаря общим возможностям, инкапсулированным в модульные объекты.
Базовый класс — это класс, не имеющий суперкласса, и поэтому находится в основании дерева подклассов.
Большинство объектно-ориентированных систем программирования обеспечивает библиотеку классов, на основании которых
разработчик создает свои собственные.
В случае, когда язык или библиотека имеют лишь один базовый класс, то он именуется высшим типом.
В языке UML класс может иметь собственный набор корневых (root)
свойств для обозначения, что это именно базовый класс.
В C++-стиле (который используется в C# и других языках) термин «базовый класс» используется вместо термина
«суперкласс».
Языки программирования могут поддерживать абстрактные и конкретные суперклассы.
В ряде языков программирования все классы явно или неявно наследуются от некого базового класса. Smalltalk был одним из первых языков, в которых
использовалась эта концепция. К таким языкам Java (java.lang.Object), C# (System.Object), Delphi (TObject).
2. Производный класс (наследник, потомок)- это класс, наследующий некоторые (или все)
свойства от своего суперкласса.
Подклассы и суперклассы часто обозначаются как производные или порождённые (derived)
и базовые (base) классы соответственно, причём эти термины закреплены создателем C++ — Бьёрном Страуструпом, который нашёл эти термины
более интуитивно понимаемыми по сравнению с традиционной номенклатурой названий.
3. Абстрактный класс в объектно-ориентированном
программировании — базовый класс, который
не предполагает создания экземпляров. Абстрактные классы реализуют на практике один из принципов ООП — полиморфизм. Абстрактный класс можно
рассматривать в качестве интерфейса к
семейству классов, порождённому им, но, в отличие от классического интерфейса, абстрактный класс может иметь
определённые методы, а также свойства.
Абстрактные методы часто являются и виртуальными, в
связи с чем понятия «абстрактный» и «виртуальный» иногда путают.
В Delphi может быть объявлен абстрактный
класс с абстрактными методами:
TAbstractClass = class
procedure NonAbstractProcedure;
procedure AbstractProcedure; abstract;
end;
Для такого класса может быть создан объект, но обращение к методу AbstractProcedure этого объекта во время выполнения
вызовет ошибку.
.Интерфе́йс (от лат. inter — «между»,
и face — «поверхность») — синтаксическая конструкция
в коде программы, используемая для специфицирования услуг,
предоставляемых классом или компонентом.
Интерфейсы позволяют наладить множественное
наследование объектов и в то же время решить проблему ромбовидного наследования. В языке C++ она
решается через наследование классов с использованием ключевого слова virtual.
Отношения между классами
- Наследование (Генерализация) —
объекты дочернего класса наследуют все свойства родительского класса. - Ассоциация — объекты классов вступают во взаимодействие между собой.
- Агрегация — объекты одного класса
входят в объекты другого. - Композиция — объекты одного класса входят в объекты другого и зависят друг от друга по времени жизни.
- Класс-Метакласс — отношение, при котором экземплярами
одного класса являются другие классы.
Контрольные вопросы
1.Определение класса ООП
2.Виды классов
3. Отношения между классами
Лекция№32. Наследование. Перегрузка методов
План лекции
- Определение наследования
- Типы наследования
- Единый базовый класс
- Перегрузка (методов
Наследование — механизм языка, позволяющий описать новый класс на основе уже существующего
(родительского, базового) класса или интерфейса. Потомок может добавить собственные методы и свойства, а также
пользоваться родительскими методами и свойствами. Позволяет строить иерархии. Является одним из основных
принципов объектно-ориентированного
программирования.
Наследование обеспечивает в ООП полиморфизм и абстракцию данных.
Типы наследования
Простое наследование
Класс, от которого произошло наследование,
называется базовым или родительским (англ. base class). Классы, которые
произошли от базового, называютсяпотомками, наследниками или производными классами (англ. derived class).
В некоторых языках используются абстрактные
классы. Абстрактный класс — это класс, содержащий хотя бы один абстрактный метод, он описан в программе, имеетполя, методы и не может использоваться для
непосредственного создания объекта. То есть от
абстрактного класса можно только наследовать.
Множественное наследование
При множественном наследовании у класса может быть более одного предка. В этом случае класс наследует методы всех предков. Достоинства такого
подхода в большей гибкости. Множественное наследование реализовано в C++. Из других языков, предоставляющих эту возможность,
можно отметить Python и Eiffel. Множественное наследование поддерживается в языке UML.
Множественное наследование — потенциальный источник ошибок, которые могут возникнуть из-за наличия одинаковых имен
методов в предках
Большинство современных объектно-ориентированных языков программирования (C#, Java, Delphi и др.) поддерживают возможность
одновременно наследоваться от класса-предка и реализовать методы нескольких интерфейсов одним и тем же классом. Этот
механизм позволяет во многом заменить множественное наследование — методы интерфейсов необходимо переопределять
явно, что исключает ошибки при наследовании функциональности одинаковых методов различных классов-предков.
Единый базовый класс
В ряде языков программирования все классы явно или неявно наследуются от некого базового класса. Smalltalk был одним из первых языков, в которых
использовалась эта концепция. К таким языкам Java (java.lang.Object), C# (System.Object), Delphi (TObject).
Абсолютно все классы в Delphi являются потомками класса TObject. Если
класс-предок не указан, то подразумевается, что новый класс является прямым потомком класса TObject.
Множественное наследование в Delphi частично поддерживается за счёт использования классов-помощников (Сlass
Helpers).
Python
Python поддерживает как одиночное, так и множественное
наследование.
Перегрузка методов (процедур и функций)
— возможность использования одноимённых подпрограмм: процедур или функций в языках программирования.
Перегружаемые функции имеют одинаковое имя, но разное количество или типы аргументов. Это разновидность статического полиморфизма, при которой вопрос о том, какую
из функций вызвать, решается по списку её аргументов. Этот подход применяется в статически типизированных языках, которые
проверяют типы аргументов при вызове функции. Перегруженная функция фактически представляет собой несколько разных
функций, и выбор подходящей происходит на этапе компиляции. Перегрузку функций не следует путать с формами
полиморфизма, где правильный метод выбирается во время выполнения, например, посредством виртуальных функций, а не
статически.
Метод — это не что иное, как набор выполняемых инструкций. Методы также определяют интерфейс данных объекта. Ещё они
помогают обеспечивать структурный подход к программированию. Программа может быть разделена на различные методы,
которые является только логической группировкой связанных выполняемых инструкций. Методы помогают при отладке
программы, поскольку отладчик может непосредственно перейти к специфическому методу и сделать необходимые
исправления. Если программа размером в 1 КБ не содержит ни одного метода, то отладить такую программу будет
достаточно трудно.
Обратите внимание: Методы также называются функциями.
Преимущества
методов:
Есть два типа методов, а именно перегруженные методы и переопределённые методы.
Перегруженные
методы — это методы, которые находятся в том же самом классе и имеют то же самое имя, но различные списки
параметров. Переопределённые методы — это методы, которые находятся в суперклассе так же как и в подклассе.
Обратите внимание: Перегруженные методы постоянно находятся в одном классе. Они имеют одну и ту же область видимости
с классом.
Перегрузка ценна тем, что она позволяет обращаться к схожим методам по общему имени. Таким образом, имя abs
представляет общее действие, которое должно выполняться. Выбор нужной конкретной версии для данной ситуации — задача
компилятора. Программисту нужно помнить только об общем выполняемом действии. Полиморфизм позволяет свести несколько
имен к одному. Хотя приведенный пример весьма прост, если эту концепцию расширить, легко убедиться в том, что
перегрузка может облегчить выполнение более сложных задач.
При перегрузке метода каждая версия этого метода может выполнять любые необходимые действия. Не существует никакого
правила, в соответствии с которым перегруженные методы должны быть связаны между собой. Однако со стилистической
точки зрения перегрузка методов предполагает определенную связь. Таким образом, хотя одно и то же имя можно
использовать для перегрузки несвязанных методов, поступать так не следует. Например, имя sqr можно было бы
использовать для создания методов, которые возвращают квадрат целочисленного значения и квадратный корень значения с
плавающей точкой. Но эти две операции принципиально различны. Такое применение перегрузки методов противоречит ее
исходному назначению. В частности, следует перегружать только тесно связанные операции.
Контрольные вопросы
- Определение наследования
- Типы наследования
- Единый базовый класс
- Перегрузка (методов
Тема 5.5. Визуальное событийно-управляемое программирование
Лекция№33. Основные компоненты интегрированной среды разработки
План лекции
1.Преимущества Visual Basic
2. Элементы среды разработчика
.
Visual Basic — это система программирования, предназначенная для написания программ, работающих под
управлением операционной системы Windows. Используя Visual Basic, можно разрабатывать очень сложные приложения
практически для любой области современных компьютерных технологий: бизнес-приложения, игры, мультимедиа, базы
данных.
Причины столь широкой популярности и у Visual Basic, и у системы Windows примерно одинаковы: фирма Microsoft сумела
такую сложную технологию, как написание компьютерных программ, сделать доступной широкому кругу пользователей путем
применения графического интерфейса..
Одним из типов объектов Visual Basic являются элементы управления — это элементы, которые используются при разработке
пользовательского интерфейса. С их помощью можно дополнять программы новыми функциями, не вникая при этом в суть их
работыТаким образом, языки визуального программирования обладают неоспоримым преимуществом — можно
сконцентрироваться на том, что вы хотите получить от программы, а не на том, как это все запрограммировать.
Одним из основных преимуществ языка Visual Basic является возможность очень быстрого создания работоспособных
приложений. С появлением версии Visual Basic 6.0 сбылась мечта программиста — простые приложения можно создавать,
практически не прибегая к написанию программного кода, а в сложных приложениях рутинный процесс его создания сведен
к минимуму. Мастера, включенные в состав Visual Basic, дают возможность быстро создавать прототипы приложений,
готовых для обсуждения и согласования с заказчиком. Данная реализация языка ставит его практически в один ряд с
такими средствами разработки, как Visual C++, Delphi и другими.
Простота и мощность языка Visual Basic позволили сделать его встроенным языком для приложений Microsoft Office.
Многие независимые разработчики, например, известная своими программами в области бухгалтерского учета фирма «1C»,
приобретают лицензии на использование языка Visual Basic в качестве внутреннего языка своих приложений.
Фирма Microsoft интегрировала также специальную версию Visual Basic, известную под именем Visual Basic for
Application (VBA) во все компоненты пакета Microsoft Office, Microsoft Project и некоторые другие программы. Кроме
того, фирма Microsoft продала лицензию на VBA очень многим фирмам — производителям программного обеспечения. Поэтому
в настоящее время Basic уже не считается учебным языком — знание Visual Basic и его диалектов (VBA, VBScript)
становится необходимостью для современного программиста любого уровня
Среда Visual Basic является интегрированной: она предоставляет разработчику широкие возможности конструирования
графического интерфейса приложения, редактирования методов и свойств объектов, отладки, тестирования и выполнения
проекта.
Основными элементами среды являются:
А) Главное окно среды напоминает окна Windows: есть строка заголовка, главное меню и панель инструментов.
Заголовок состоит из названия системы программирования Microsoft Visual Basic, левее этих слов расположено название
проекта — Project1. Это название Visual Basic присвоил автоматически, его можно заменить каким-либо более
осмысленным. В правой части Строка меню и панель инструментов во многом совпадают с меню и панелью Windows, однако,
в них имеются меню и инструменты, которые обеспечивают доступ к специальным средствам Visual Basic.
Строка меню состоит из заголовков меню, которые содержат все команды, необходимые при работе с Visual Basic. Меню
File, View, Edit, Window и Help являются характерными для Windows и приложений Windows, но имеют ряд
особенностей.
Б) Окно конструктора форм: В серединной части экрана расположено окно проектов, озаглавленное Project1 — Form1
(Form). Оно является основным во время создания интерфейса будущего приложения. Внутри этого окна размещено окно
дизайнера (конструктора) форм, чаще его называют просто окном форм. Его название Form1, автоматически присваивается
Visual Basic, и должно быть впоследствии изменено. На этапе конструирования проекта на поверхности окна форм
размещают необходимое количество объектов, предназначенных для управления приложением. Поверхность окна форм в
режиме конструирования покрыта точками. Эти точки являются узлами координатной сетки и служат для облегчения
размещения объектов на форме.
В) Панель элементов (ToolBox): В левой части экрана вертикально расположена панель или палитра объектов (элементов).
Она содержит набор специальных инструментов — графических объектов, которые можно размещать в окне форм.
Г) Окно Свойства объекта (Properties): используется, чтобы задать свойства формы и размещенных на ней объектов на
этапе проектирования. Это окно содержит перечень тех свойств объекта, которые пользователь может изменить. Значения
свойств можно изменять непосредственно в окне Properties. Способы изменения свойств объекта:
в правое поле можно ввести значение свойства. Свойство будет изменено, если новое значение допустимо;
значение свойства можно выбрать из предложенного списка, нажав в правом поле кнопку с треугольником;
щелчок по кнопке с многоточием в правом поле вызовет стандартное диалоговое окно Windows, позволяющее выбрать
допустимое значение свойства.
Д) Окно Проводник проекта (Project Explorer): позволяет анализировать структуру проекта и его
состав. Приложение Visual Basic на этапе разработки состоит из нескольких файлов, которые все вместе составляют
проект.. Окно проводника содержит три кнопки — View Code — показать окно кодов, View Object — показать окно форм и
Toggle Folders — открыть/закрыть папку, содержащую список объектов.
Е) Еще одно окно — Code — окно Редактора кода в исходном состоянии среды не видно. Оно предназначено для создания и
редактирования кода программы и вызывается на экран по мере необходимости.
Контрольные вопросы
1.Преимущества Visual Basic
2. Элементы среды разработчика
Тема 5.5.
Лекция№34. События компонентов Процедуры, определенные
пользователем.
План лекции
- Интерфейсные объекты
- Графический интерфейс
- Событийная процедура
Рассмотрим простую и естественную модель событийно-управляемого и визуального программирования, характерную для языка
и среды Visual С++. В этой модели у приложения три составляющие: визуальная, системная и обработчик событий.
Визуальная составляющая задает образ на экране, с которым будет работать пользователь. Она, как правило,
разрабатывается визуальным инструментарием, позволяющим программисту создавать из элементов нужный образ на экране.
Эти элементы являются объектами со своими свойствами и поведение.
Визуальная составляющая определяет интерфейс пользователя. Такие элементы интерфейса, как кнопки, окна
редактирования, окна списков, называют элементами управления (controls). Эти и другие элементы интерфейса
стандартизированы в стандарте пользовательского интерфейса CUA (Common User Access) в рамках общего стандарта SSA
(system Application Architecture) фирмы IBM. Поэтому в разных средах разработки (Visual Basic, Visual C++,
Delphi и др.) визуальный инструментарий содержит одни и те же элементы интерфейса. Такие же элементы интерфейса
содержат разные приложения.
Элементы управления являются объектами, свойства и поведение которых определяется их переменными и методами. Они
относятся к интерфейсным объектам.
Пользователь — это возмутитель спокойствия в мире объектов приложения. Он «нажимает» на кнопки, выбирает
элементы списков, печатает тексты в окнах редактирования. Каждому его действию соответствует
некоторое событие. Системная составляющая приложения, которая включает в себя средства операционной системы
и средства среды программирования, определяет тип и параметры события и формирует сообщение объекту, с которым
связано событие. Иначе говоря, системная составляющая находит нужный объект и запускает функцию-обработчик сообщения
— соответствующий метод этого объекта. Таким образом, пользователь может взаимодействовать с элементами визуальной
составляющей, а само взаимодействие обеспечивается системной составляющей.
Обработчики событий и связанных с ними сообщений составляют третий компонент приложения. Когда пользователь
действует на элемент управления, происходит событие, распознаваемое системной составляющей, которая вызывает
обработчик события. В работе системной составляющей важную роль играют сообщения, связанные с происшедшими
событиями. В обработке события (т.е. в методе объекта) программист волен предусмотреть самые разные действия: может
изменять свойства других объектов, вызывать методы других объектов, добавлять или удалять объекты визуальной
составляющей, даже полностью изменить ее облик.
Программирование на Visual С++ полностью соответствует концепциям визуального и событийно-управляемого
программирования. Чтобы создать приложение на Visual С++, нужно сделать две вещи: разработать с помощью визуального
инструментария интерфейс пользователя и написать реакции на действия пользователя, т.е. для каждого возможного
события — обрабатывающий его метод.
В языках визуального объектно-ориентированного программирования (например, Visual Basic) применяется визуальный
метод создания графического интерфейса приложения и объектный метод построения его программного кода.
Графический интерфейс. Визуальное программирование позволяет делать графический интерфейс разрабатываемых приложений
на основе форм и управляющих элементов.
В роли основных объектов при визуальном программировании
выступают формы (Forms). Форма представляет собой окно, на котором размещаются управляющие элементы. Управляющие
элементы — это командные кнопки (CommandButton), переключатели, или «флажки» (Checkbox), поля выбора, или
«радиокнопки» (OptionsButton), списки (ListBox), текстовые поля (TextBox) и др.
Событийная
процедура. Важное место в технологии визуального объектно-ориентированного программирования занимают события. В
качестве события могут выступать щелчок кнопкой мыши на объекте, нажатие определенной клавиши, открытие документа и
т. д. В качестве реакции на события запускается определенная процедура, которая способна изменять свойства объекта,
вызывать его методы и т. д. Например, если пользователь производит какое-либо воздействие на элемент
графического интерфейса (нажимает командную кнопку), в качестве отклика выполняется некоторая последовательность
действий (событийная процедура).
Имя процедуры включает в себя имя объекта и имя события.
Объект_Событие
Каждая процедура представляет собой отдельный программный модуль, в начале и в
конце которого ставятся ключевые слова Sub и End:
Sub Объект_Событие() Программный код End
Sub Разрабатываемое на языке Visual Basic приложение называется проектом. Проект включает в себя не
только форму с размещенными на ней управляющими элементами, но и программные модули событийных процедур, которые
описывают поведение объектов приложения и взаимодействие объектов между собой.
Контрольные вопросы
- Интерфейсные объекты
- Графический интерфейс
- Событийная процедура
Тема 5.6. Разработка оконного приложения
Лекция№35. Создание интерфейса оконного приложения. Компиляция и
запуск приложения.
План лекции
- Программы с оконным интерфейсом на Linux
- Этапы создания программы с оконным интерфейсом
Программы с оконным интерфейсом на Linux
Многие современные программы имеют оконный интерфейс (или просто GUI – Graphical User Interface). Он делает намного
удобнее общение с пользователем, позволяет ускорить выполнение ряда рутинных процедур, да и просто GUI – это
эстетично и современно. Мы создадим одну очень короткую программу. Задача не в том, чтобы научиться создавать
программы с оконным интерфейсом (этому будут потом посвящены отдельные руководства), а в том, чтобы
продемонстрировать, что их сборка и распространение принципиально не отличаются от того, что мы видели в случае с
консольными программами.
Ключевым звеном оконных программ является X-сервер. Он отвечает за прорисовку графических элементов. Программы,
имеющие оконный интерфейс должны обращаться к X-серверу. Для обращения к нему служит специальный интерфейс (он
называется Xlib). Однако существуют средства, позволяющие избежать непосредственного обращения к низкоуровневому
интерфейсу (а работа с ним занимает длительное время). Два наиболее распространённых таких средства – это библиотеки
GTK и Qt.
Напишем очень короткую программу с оконным интерфейсом с использованием библиотеки GTK+ 2.0.
Чтобы простым способом её скомпилировать, нужно воспользоваться обычным компилятором gcc, сообщив ему, что надо
подключить библиотеку GTK+. Для этого мы вызываем утилиту pgk-config. Эта утилита ищет установленные в системе
библиотеки и возвращает путь к ним. В данном случае нам нужна библиотека gtk+, что мы и указываем в параметре.
gcc main.c -o gtkdemo `pkg-config —cflags —libs gtk+-2.0`
Теперь давайте подготовим эту программку для распространения.
Создайте каталог проекта. Назовите его gtkdemo. В нём, как обычно, каталог src. В каталоге src – вышеуказанный файл
main.c.
В корневом каталоге создайте файл configure.ac
Вы, конечно, заметили, что, помимо уже знакомых строк мы сюда включаем директиву PKG_CHECK_MODULES. Эта директива
запускает уже знакомую утилиту pkg-config. Путь к заголовочным файлам библиотеки gtk+ сохраняется в переменной
DEPS_CFLAGS. Путь к библиотечным файлам – в переменной DEPS_LIBS.
Ещё в корневом каталоге создайте файл Makefile.am
SUBDIRS = src
А в каталоге src – другой файл Makefile.am
bin_PROGRAMS = gtkdemo
gtkdemo_SOURCES = main.c
gtkdemo_LDADD = $(DEPS_LIBS)
AM_CPPFLAGS = $(DEPS_CFLAGS)
Здесь указывается, что адреса, найденные и сохранённые в этих двух переменных, должны быть подключены к проекту.
Всё! Собираем, как обычно.
aclocal
autoconf
touch NEWS README AUTHORS ChangeLog
autoheader
automake -a
./configure
make
make dist
Далее можете при желании инсталлировать программу.
make install
Или подготовить дистрибутив.
make dist
В этот дистрибутив будет входить и файл проекта (qtdemo.pro). Поэтому пользователи, получив такой пакет не должны
запускать configure. Всё, что им надо, это дать следующие команды:
qmake
make
Хотя qmake не требует обязательного присутствия в дистрибутиве файла README, желательно такой файл создать, и описать
подробно, как правильно устанавливать программу. Будет неприятно, если пользователь будет источать на вас весь запас
бранных слов только из-за того, что у него не установлена библиотека Qt, или из-за того, что его не предупредили,
что в данном пакете не надо запускать ./configure.
Курьёзно, что никто вам не запрещает пользоваться утилитой qmake для сборки самых обычных консольных программ, не
имеющих никакого отношения к Qt.
Если мы возьмём нашу последнюю версию, то там к программе требовалось подключить библиотеку libcalculate.
Однако не забывайте, что для сборки программы из архива, полученного таким путём, пользователь также должен иметь
программу qmake на своём компьютере. Поэтому пользуйтесь такими нестандартными возможностями с большой
осторожностью.
Контрольные вопросы
- Программы с оконным интерфейсом на Linux
- Этапы создания программы с оконным интерфейсом
Список литературы
- Семакин И.Г. Основы алгоритмизации и программирования. Учебник для студентов учреждений среднего
профессионального образования — 3-е изд., стер. — М. : Издательский центр «Академия», 2013. 400 с. - Семакин И., Шестаков А. Основы алгоритмизации и
программирования. Практикум. Учебное пособие М.: Академия, 2013г. – 144с.
- Голицына О.Л., Попов И.И. Основы алгоритмизации и
программирования: учебное пособие. М.: Форум, 2014г. – 432с.
- С. Скиена. Алгоритмы. Руководство по разработке. СПб.: БХВ-
Петербург, 2011г. – 720с.
- Т.Х. Кормен. Алгоритмы. Вводный курс. М.: Вильямс. 2014г. – 208с.
- Кнут Д. Искусство программирования. Том 1. Основные алгоритмы.
М.: Вильямс, 2010г.-720с.
Второе издание Алгоритмы Руководство по разработке XYZXYZS YZXYZS ZXYZS XYZ$ YZ$ Z$ s Стивен Скиена Springer hy
Стивен С. Скиена Алгоритмы Руководство по разработке 2-е издание Санкт-Петербург «БХВ-Петербург» 2011
УДК 681.3.06 ББК 32.973.26-018.2 С42 Скиена С. С42 Алгоритмы. Руководство по разработке. — 2-е изд.: Пер. с англ. — СПб.: БХВ-Петербург. 2011. — 720 с.: ил. ISBN 978-5-9775-0560-4 Книга является наиболее полным руководством по разработке эффективных ал- горитмов. Первая часть книги содержит практические рекомендации по разработке алгоритмов: приводятся основные понятия, дается анализ алгоритмов, рассматри- ваются типы структур данных, основные алгоритмы сортировки, операции обхода графов и алгоритмы для работы со взвешенными графами, примеры использования комбинаторного поиска, эвристических методов и динамического программирова- ния. Вторая часть книги содержит обширный список литературы и каталог из 75 наиболее распространенных алгоритмических задач, для которых перечислены существующие программные реализации. Приведены многочисленные примеры задач. Книгу можно использовать в качестве справочника по алгоритмам для про- граммистов, исследователей и в качестве учебного пособия для студентов соответ- ствующих специальностей. Для программистов, исследователей и студентов УДК 681.3.06 ББК 32.973.26-018.2 Translation from the English language edition: "The Algorithm Design Manual" by Steven S. Skiena: ISBN 978-1-84800- 069-8 Copyright 2008 Springer. The Netherlands as a part of Springer Science f Business Media. All rights reserved Russian edition copyright (O 2011 year by BHV - St.Petersburg All rights reserved. No part of this book may be reproduced or transmitted in any form or by any means, electronic or mechanical, including photocopying, recording or by any information storage retrieval system, without permission from the Publisher. Перевод английской редакции книги The Algorithm Design Manual, Steven S Skiena; ISBN 978-1-84800-069-8. Cop- yright «' 2008 Springer. The Netherlands as a part of Springer Science+Business Media. Все права защищены. Русская редакция издания выпущена издательством "БХВ-Петербург" в 2011 году. Все права защищены. Никакая часть настоящей книги не может быть воспроизведена или передана в какой бы то ни было форме и какими бы го ни было средствами, будь то электронные или механические, включая фотокопирование и запись на магнитный но- ситель, если надо нет письменного разрешения издательства. ISBN 978-1-84800-069-8 (англ.) ISBN 978-5-9775-0560-4 (рус.) © 2008 Sponger, The Netherlands as a part of Springer Scieiice+Business Media © Перевод на русский язык "БХВ-ПетербурГ'. 2011
Оглавление Предисловие...............................................................13 Читателю..................................................................13 Преподавателю........................................................... 15 Благодарности..............................................................16 Ограничение ответственности...............................................17 ЧАСТЬ I. ПРАКТИЧЕСКАЯ РАЗРАБОТКА АЛГОРИТМОВ...............................19 Глава 1. Введение в разработку алгоритмов.................................21 1.1. Оптимизация маршрута робота..........................................22 1.2. Задача календарного планирования.....................................26 1.3. Обоснование правильности алгоритмов................................. 29 1.3.1. Представление алгоритмов........................................30 1.3.2. Задачи и свойства...............................................31 1.3.3. Демонстрация неправильности алгоритма...........................32 1.3.4. Индукция и рекурсия.............................................33 1.3.5. Суммирование....................................................35 1.4. Моделирование задачи.................................................37 1.4.1. Комбинаторные объекты...........................................37 1.4.2. Рекурсивные объекты.............................................39 1.5. Истории из жизни.....................................................40 1.6. История из жизни. Моделирование проблемы ясновидения.................41 1.7. Упражнения...........................................................45 Глава 2. Анализ алгоритмов................................................49 2.1. Модель вычислений RAM................................................49 2.1.1. Анализ сложности наилучшего, наихудшего и среднего случая.......50 2.2. Асимптотические обозначения..........................................52 2.3. Скорость роста и отношения доминирования.............................55 2.3.1. Отношения доминирования.........................................56 2.4. Работа с асимптотическими обозначениями..............................58 2.4.1. Сложение функций................................................58 2.4.2. Умножение функций...............................................58 2.5. Оценка эффективности.................................................59 2.5.1. Сортировка методом выбора.......................................59 2.5.2. Сортировка вставками............................................60 2.5.3. Сравнение строк.................................................61 2.5.4. Умножение матриц................................................63
6 Оглавление 2.6. Логарифмы и их применение.............................................64 2.6.1. Логарифмы и двоичный поиск......................................64 2.6.2. Логарифмы и деревья.............................................64 2.6.3. Логарифмы и биты................................................65 2.6.4. Логарифмы и умножение...........................................65 2.6.5. Быстрое возведение в степень....................................66 2.6.6. Логарифмы и сложение............................................66 2.6.7. Логарифмы и система уголовного судопроизводства.................67 2.7. Свойства логарифмов...................................................68 2.8. История из жизни. Загадка пирамид.....................................69 2.9. Анализ высшего уровня (*).............................................72 2.9.1. Малораспространенные функции....................................73 2.9.2. Пределы и отношения доминирования...............................74 2.10. Упражнения...........................................................75 Глава 3. Структуры данных..................................................84 3.1. Смежные и связные структуры данных....................................84 3.1.1. Массивы.........................................................85 3.1.2. Указатели и связные структуры данных............................86 3.1.3. Сравнение.......................................................89 3.2. Стеки и очереди..................................................... 90 3.3. Словари...............................................................91 3.4. Двоичные деревья поиска...............................................95 3.4.1. Реализация двоичных деревьев....................................96 3.4.2. Эффективность двоичных деревьев поиска.........................100 3.4.3. Сбалансированные деревья поиска................................101 3.5. Очереди с приоритетами...............................................102 3.6. История из жизни. Триангуляция.......................................104 3.7. Хэширование и строки.................................................107 3.7.1. Коллизии.......................................................108 3.7.2. Эффективный метод поиска строк посредством хэширования.........110 3.7.3. Выявление дубликатов с помощью хэширования......................112 3.8. Специализированные структуры данных. .................................113 3.9. История из жизни. Геном человека.....................................114 3.10. У пражнения.........................................................118 Глава 4. Сортировка и поиск...............................................123 4.1. Применение сортировки................................................123 4.2. Практические аспекты сортировки......................................126 4.3. Пирамидальная сортировка.............................................128 4.3.1. Пирамиды.......................................................129 4.3.2. Создание пирамиды..............................................132 4.3.3. Наименьший элемент пирамиды....................................133 4.3.4. Быстрый способ создания пирамиды (*)...........................135 4.3.5. Сортировка вставками......................................... 137 4.4. История из жизни. Билет на самолет...................................138 4.5. Сортировка слиянием. Метод "разделяй и властвуй".....................141 4.6. Быстрая сортировка. Рандомизированная версия.........................143 4.6,1. Ожидаемое время исполнения алгоритма быстрой сортировки........146
Оглавление 7 4.6.2. Рандомизированные алгоритмы.......................................147 4.6.3. Действительно ли алгоритм быстрой сортировки работает быстро?.....150 4.7. Сортировка распределением. Метод блочной сортировки....................150 4.7.1. Нижние пределы для сортировки.....................................151 4.8. История из жизни. Адвокат Скиена.......................................152 4.9. Двоичный поиск и связанные с ним алгоритмы ............................154 4.9.1. Частота вхождения элемента........................................155 4.9.2. Односторонний двоичный поиск......................................155 4.9.3. Корни числа.......................................................156 4.10. Метод "разделяй и властвуй"...........................................156 4.10.1. Рекуррентные соотношения.........................................157 4.10.2. Рекуррентные соотношения метода "разделяй и властвуй"............158 4.10.3. Решение рекуррентных соотношений типа "разделяй и властвуй" (*)..159 4.11. Упражнения........................................................ 161 Глава 5. Обход графов.......................................................168 5.1. Разновидности графов...................................................169 5.1.1. Граф дружеских отношений..........................................172 5.2. Структуры данных для графов............................................174 5.3. История из жизни. Жертва закона Мура...................................178 5.4. История из жизни. Создание графа.......................................181 5.5. Обход графа............................................................184 5 6 Обход в ширину ...... . . ... .. .................... .. .............185 5.6.1. Применение обхода.................................................187 5.6.2. Поиск путей .... .. . ......................................... 188 5.7. Применение обхода в ширину.............................................189 5.7.1. Компоненты связности..............................................189 5.7.2. Раскраска графов двумя цветами....................................191 5.8. Обход в глубину........................................................192 5.9. Применение обхода в глубину............................................195 5.9.1. Поиск циклов......................................................196 5.9.2. Шарниры графа................................................... 196 5.10. Обход в глубину ориентированных графов............................... 200 5.10.1. Топологическая сортировка........................................202 5.10.2. Сильно связные компоненты........................................203 5.11. Упражнения............................................................207 Глава 6. Алгоритмы для работы со взвешенными графами........................213 6.1. Минимальные остовные деревья...........................................214 6.1.1. Алгоритм Прима....................................................215 6.1.2. Алгоритм Крускала.................................................218 6.1.3. Поиск-объединение.................................................220 6.1.4. Разновидности остовных деревьев ..................................223 6.2. История из жизни. И все на свете только сети...........................224 6.3. Поиск кратчайшего пути.................................................227 6.3.1. Алгоритм Дейкстры.................................................228 6.3.2. Кратчайшие пути между всеми парами вершин.........................231 6.3.3. Транзитивное замыкание............................................233
8 Оглавление 6.4. История из жизни. Печатаем с помощью номеронабирателя.................234 6.5. Потоки в сетях и паросочетание в двудольных графах....................239 6.5.1. Паросочетание в двудольном графе................................239 6.5.2. Вычисление потоков в сети.......................................240 6.6. Разрабатывайте не алгоритмы, а графы ............................... 244 6.7. Упражнения............................................................246 Глава 7. Комбинаторный поиск и эвристические методы........................251 7.1. Перебор с возвратом....................................................251 7.1.1. Генерирование всех подмножеств..................................254 7.1.2. Генерирование всех перестановок............................... 255 7.1.3. Генерирование всех путей в графе................................256 7.2. Отсечение вариантов поиска............................................258 7.3. Судоку.............................................................. 259 7.4. История из жизни. Покрытие шахматной доски............................264 7.5. Эвристические методы перебора .................................... 267 7.5.1. Произвольная выборка............................................268 7.5.2. Локальный поиск.................................................271 7.5.3. Имитация отжига.................................................274 7.5.4. Применение метода имитации отжига. ... ........................ 278 7.6. История из жизни. Только это не радио.................................280 7.7. История из жизни. Отжиг массивов.................................... 283 7.8. Другие эвристические методы поиска....................................286 7.9. Параллельные алгоритмы............................................... 287 7.10. История из жизни. "Торопиться в никуда"..............................289 7.11. Упражнения...........................................................290 Глава 8. Динамическое программирование................................... 293 8.1. Кэширование и вычисления..............................................294 8.1.1. Генерирование чисел Фибоначчи методом рекурсии..................294 8.1.2. Генерирование чисел Фибоначчи посредством кэширования...........295 8.1.3. Генерирование чисел Фибоначчи посредством динамического программирования.................................................... 297 8.1.4. Биномиальные коэффициенты..................................... 298 8.2. Поиск приблизительно совпадающих строк.............................. 300 8.2.1. Применение рекурсии для вычисления расстояния редактирования ...301 8.2.2. Применение динамического программирования для вычисления расстояния редактирования..............................................302 8.2.3. Восстановление пути.............................................304 8.2.4. Разновидности расстояния редактирования........................ 306 8.3. Самая длинная возрастающая последовательность.........................310 8.4. История из жизни. Эволюция омара......................................312 8.5. Задача разбиения......................................................315 8.6. Синтаксический разбор.................................................318 8.6.1. Триангуляция с минимальным весом................................320 8.7. Ограничения динамического программирования. Задача коммивояжера...... 322 8.7.1. Вопрос правильности алгоритмов динамического программирования.....323 8.7.2. Эффективность алгоритмов динамического программирования.........324 8.8. История из жизни. Динамическое программирование и язык Prolog. 325
Оглавление 9 8.9. История из жизни. Сжатие текста для штрих-кодов...................328 8.10. Упражнения.......................................................332 Глава 9. Труднорешаемые задачи и аппроксимирующие ал! ори гмы...........338 9.1. Сведение задач....................................................338 9.1.1. Ключевая идея................................................339 9.1.2. Задачи разрешимости.............................'.............340 9.2, Сведение для создания новых алгоритмов............................341 9.2.1. Поиск ближайшей пары.........................................341 9.2.2. Максимальная возрастающая подпоследовательность..............342 9.2.3. Наименьшее общее кратное.....................................343 9.2.4. Выпуклая оболочка (*)........................................344 93. Простые примеры сведения сложных задач............................345 9.3.1. Гамильтонов цикл.............................................345 9.3.2. Независимое множество и вершинное покрытие...................347 9.3.3. Задача о клике...............................................350 9.4. Задача выполнимости булевых формул................................351 9.4.1. Задача выполнимости в 3-конъюнктивной нормальной форме.......352 9.5. Нестандартные сведения............................................353 9.5.1. Целочисленное программирование...............................354 9.5.2. Вершинное покрытие...........................................356 9.6. Искусство доказательства сложности................................358 9.7. История из жизни. Наперегонки со временем.........................360 9.8. История из жизни. Полный провал................................. 362 9.9. Сравнение классов сложности Р и NP................................364 9.9.1. Верификация решения и поиск решения..........................365 9.9.2. Классы сложности Р и NP......................................365 9.9.3. Почему задача выполнимости является самой сложной из всех сложных задач?..............................................................366 9.9.4. NP-сложность по сравнению с NP-полнотой......................367 9.10. Решение NP-полных задач..........................................367 9.10.1. Аппроксимация вершинного покрытия......................... 368 9.10.2. Задача коммивояжера в евклидовом пространстве............. 370 9.10.3. Максимальный бесконтурный подграф...........................371 9.10.4. Задача о покрытии множества.................................372 9.11. Упражнения.......................................................375 Глава 10. Как разрабатывать алгоритмы..................................380 ЧАСТЬ И. КАТАЛОГ АЛГОРИТМИЧЕСКИХ ЗАДАЧ.................................385 Глава 11. Описание каталога............................................387 Глава 12. Структуры данных.............................................389 12.1. Словарь..........................................................389 12.2. Очереди с приоритетами...........................................395 12.3. Суффиксные деревья и массивы.....................................398 12.4. Графы.......................................................... 402 12.5. Множества........................................................406 12.6. Kd-деревья.......................................................410
10 Оглавление Глава 13. Численные задачи................................................415 13.1. Решение системы линейных уравнений..................................416 13.2. Уменьшение ширины ленты матрицы.....................................419 13.3. Умножение матриц....................................................422 13.4. Определители и перманенты...........................................425 13.5. Условная и безусловная оптимизация..................................427 13.6. Линейное программирование...........................................431 13.7. Генерирование случайных чисел.......................................435 13.8. Разложение на множители и проверка чисел на простоту................440 13.9. Арифметика произвольной точности....................................443 13.10. Задача о рюкзаке...................................................448 13.11. Дискретное преобразование Фурье....................................451 Глава 14. Комбинаторные задачи............................................455 14.1. Сортировка.. ............................................. . ... .456 14.2. Поиск...............................................................461 14.3. Поиск медианы и выбор элементов.....................................465 14.4. Генерирование перестановок..........................................468 14.5 Генерирование подмножеств.......................................... 472 14.6. Генерирование разбиений.............................................475 14.7. Генерирование графов................................................479 14.8. Календарные вычисления..............................................484 14.9. Календарное планирование............................................486 14.10. Выполнимость.......................................................489 Глава 15. Задачи на графах с полиномиальным временем исполнении...........493 15.1 Компоненты связности............................................. 494 15.2. Топологическая сортировка...........................................497 15.3. Минимальные остовные деревья........................................500 15.4. Поиск кратчайшего пути..............................................505 15.5. Транзитивное замыкание и транзитивная редукция......................511 15.6. Паросочетание.......................................................514 15.7. Задача поиска эйлерова цикла и задача китайского почтальона.........517 15.8. Реберная и вершинная связность......................................521 15.9. Потоки в сети.................................................... 524 15.10. Рисование графов...................................................528 15.11. Рисование деревьев.................................................531 15.12. Планарность........................................................534 Глава 16. Сложные задачи на графах........................................538 16.1. Задача о клике......................................................539 16.2. Независимое множество...............................................542 16.3. Вершинное покрытие..................................................544 16.4. Задача коммивояжера.................................................547 16.5. Гамильтонов цикл....................................................551 16.6. Разбиение графов....................................................554 16.7. Вершинная раскраска.................................................557 16.8. Реберная раскраска................................................. 561 16.9. Изоморфизм графов...................................................563 Oaj 16. 16. Гл: 17. 17.: 17.: 17/ 17/ 17.( 17.: 17.1 17.S 17.1 17.1 17.1 17.1 17.1 17.1 17.1 Гла 18.1 18.2 18.3 18.4 18.5 18.6 18.7 18.8 18.9. Гла 19.1. 19.2. 19.3. 19.4. Спи» Прех
Оглавление 11 I6.10. Дерево Штейнера....................................................568 I6.l I. Разрывающее множество ребер или вершин............................572 Глава 17. Вычислительная геометрия........................................576 17.1. Элементарные задачи вычислительной геометрии........................577 17.2. Выпуклая оболочка...................................................581 17.3. Триангуляция........................................................585 17.4. Диаграммы Вороного..................................................589 17.5. Поиск ближайшей точки...............................................592 17.6. Поиск в области.....................................................596 17.7. Местоположение точки................................................599 17.8. Выявление пересечений...............................................603 17.9. Разложение по контейнерам...........................................607 17.10. Преобразование к срединной оси.....................................610 17.11. Разбиение многоугольника на части..................................613 17.12. Упрощение многоугольников..........................................615 17.13. Выявление сходства фигур...........................................619 17.14. Планирование перемещений...........................................621 17.15. Конфигурации прямых................................................625 17,16. Сумма Минковского..................................................628 Глава 18. Множества и строки..............................................631 18.1. Поиск покрытия множества............................................631 18.2. Задача укладки множества............................................635 18.3. Сравнение строк.....................................................638 18.4. Нечеткое сравнение строк............................................641 18.5. Сжатие текста.......................................................647 18.6. Криптография........................................................651 18.7. Минимизация конечного автомата......................................656 18.8. Максимальная общая подстрока........................................659 18.9. Поиск минимальной общей надстроки...................................663 Глава 19. Ресурсы.........................................................666 19.1. Программные системы.................................................666 19.1.1. Библиотека LEDA...............................................666 19.1.2. Библиотека CGAL...............................................667 19.1.3. Библиотека Boost..............................................668 19.1.4. Библиотека GOBLIN.............................................668 19.1.5. Библиотека Netlib.............................................668 19.1.6. Коллекция алгоритмов ассоциации АСМ...........................669 19.1.7. Сайты SourceForge и CPAN......................................669 19.1.8. Система Stanford GraphBase....................................669 19.1.9. Пакет Combinatorica...........................................670 19.1.10. Программы из книг............................................670 19.2. Источники данных....................................................672 19.3. Библиографические ресурсы...........................................673 19.4. Профессиональные консалтинговые услуги..............................673 Список литературы.........................................................675 Предметный указатель......................................................713
Предисловие Многие профессиональные программисты, с которыми я встречался, не очень хорошо подготовлены к решению проблем разработки алгоритмов. Это прискорбно, так как методика разработки алгоритмов является одной из важнейших технологий вычисли- тельной техники. Создание правильных, эффективных и реализуемых алгоритмов для решения реальных задач требует от разработчика знаний в двух областях: ♦ Методика. Хорошие разработчики алгоритмов знают несколько фундаментальных приемов, в число которых входят работа со структурами данных, динамическое программирование, поиск в глубину, перебор с возвратами и эвристика. Пожалуй, самым важным техническим приемом является моделирование — искусство абстра- гирования от запутанного реального приложения к четко сформулированной задаче, поддающейся алгоритмическому решению. ♦ Ресурсы. Хорошие разработчики алгоритмов пользуются коллективным опытом предыдущих поколений разработчиков. Вместо того, чтобы создавать "с нуля" алго- ритм для стоящей перед ними задачи, они сначала узнают, что уже известно о ней и ищут существующие реализации решения, чтобы использовать их в качестве отправной точки. Они знают много классических алгоритмических задач, кото- рые служат исходным материалом для моделирования практически любого прило- жения. Эта книга была задумана как руководство по разработке алгоритмов, в котором я пла- нировал изложить технологию разработки комбинаторных алгоритмов. Она рассчитана как на студентов, так и на профессионалов. Книга состоит из двух частей. Часть I явля- ет собой общее руководство по техническим приемам разработки и анализа компью- терных алгоритмов. Часть II предназначена для использования в качестве справочного и познавательного материала и состоит из каталога алгоритмических ресурсов, реали- заций и обширного списка литературы. Читателю Меня очень обрадовал теплый прием, с которым было встречено первое издание книги "Руководство по разработке алгоритмов", опубликованное в 1997 г. Она была признана уникальным руководством по применению алгоритмических приемов для решения за- дач, которые часто возникают в реальной жизни. Но с тех пор многое изменилось в этом мире. Если считать, что начало современной разработке и анализу алгоритмов было положено приблизительно в 1970 г., то получается, что около 30% современной истории алгоритмов приходится на время, прошедшее после первой публикации руко- водства.
14 Предисловие Читатели одобрили три аспекта руководства: каталог алгоритмических задач, истории из жизни и электронную версию книги. Эти элементы были сохранены в настоящем издании. ♦ Каталог алгоритмических задач. Не так-то просто узнать, что уже известно о стоя- щей перед вами задаче. Именно поэтому в книге имеется каталог 75 наиболее важ- ных задач, часто возникающих в реальной жизни. Просматривая его, студент или специалист-практик может быстро выяснить название своей задачи и понять, что о ней известно и как приступить к ее решению. Чтобы облегчить идентификацию, каждая задача в каталоге сопровождается рисунками состояния "до и после", иллю- стрирующими требуемые характеристики входа и выхода. За этот каталог задач один остроумный рецензент предложил назвать мою книгу "Автостопом по алго- ритмам". Каталог задач является самой важной частью этой книги. Обновляя каталог для это- го издания, я собрал отзывы и рекомендации ведущих мировых экспертов по каж- дой содержащейся в нем задаче. Особое внимание было уделено обновлению про- граммных реализаций решений задач. ♦ Истории из жизни. На практике алгоритмические задачи редко возникают в начале работы над большим проектом. Наоборот, они обычно появляются в виде подзадач, когда становится ясно, что программист не знает, что делать дальше, или что при- нятое решение ошибочно. Чтобы продемонстрировать, как алгоритмические задачи возникают в реальной жизни, в материал книги были включены правдивые истории, описывающие наш опыт по решению практических задач. При этом преследовалась цель показать, что разработка и анализ алгоритмов представляют собой не отвлеченную теорию, а важный инструмент, требующий умелого обращения. В этом издании сохранены все первоначальные истории из жизни (обновленные по мере необходимости), а также были добавлены новые истории, имеющие отноше- ние к внешней сортировке, работе с графами, методу имитации отжига и другим ал- горитмическим областям. ♦ Электронная версия. Поскольку человек практичный обычно ищет готовую про- грамму, а не алгоритм, в книге даются ссылки на все имеющиеся рабочие реализа- ции алгоритмов. Для удобства эти реализации собраны на одном веб-сайте http://www.cs.sunysb.edu/~algorith. После публикации книги этот веб-сайт долгое время был одним из первых в результатах поиска в Google по слову "algorithm". Кроме этого, в книге даны рекомендации, как найти подходящий код для решения той или иной задачи. Наличие данных реализаций сводит проблему разработки ал- горитма к правильному моделированию приложения и избавляет разработчика от необходимости разбираться в подробностях самого алгоритма. Этот подход приме- няется на всем протяжении книги. Важно обозначить то, чего нет в этой книге. Мы не уделяем большой внимания мате- матическому обоснованию алгоритмов и в большинстве случаев ограничиваемся не- формальными рассуждениями. В этой книге вы не найдете ни одной теоремы. Более подробную информацию вы можете получить, изучив приведенные программы или справочный материал. Цель данного руководства в том, чтобы как можно быстрее ука- зать читателю верное направление движения.
Предисловие 15 Преподавателю Эта книга содержит достаточно материала для курса "Введение в алгоритмы". Предпо- лагается, что читатель обладает знаниями, полученными при изучении таких курсов, как "Структуры данных" или "Теория вычислительных машин и систем". На сайте http://www.algorist.com можно загрузить полный набор лекционных слайдов для преподавания материала этой книги. Кроме того, я выложил аудио- и видеолекции с использованием этих слайдов для преподавания полного курса продолжительностью в один семестр. Таким образом, вы можете через Интернет воспользоваться моей по- мощью в преподавании вашего курса. Главное внимание в книге уделяется разработке алгоритмов, а их анализ стоит на вто- ром плане. Книгу можно использовать как для обычных лекционных курсов, так и для активного обучения, при котором преподаватель не читает лекцию, а руководит реше- нием реальных задач в группе студентов. Истории из жизни предоставляют введение вактивный метод обучения. Книга была полностью переработана с целью облегчить ее использование в качестве учебника. Для настоящего издания характерны: ♦ подробное изложение материала. По сравнению с предыдущим изданием, объем первой части книги был увеличен вдвое. Однако дополнительный материал не уве- личивает количество обсуждаемых тем, а служит для более полного и тщательного изложения основ; ♦ обсуждение основ. Учебники по разработке алгоритмов обычно представляют об- щеизвестные алгоритмы как нечто само собой разумеющееся и не обсуждают идеи, лежащие в их основе, или слабые места других подходов. Истории из жизни, при- водимые в этой книге, иллюстрируют процесс выбора алгоритма на примерах решения некоторых прикладных задач, но я применил аналогичный подход и к из- ложению материала, касающегося классических алгоритмов; ♦ остановки для размышлений. В этих разделах я изложил ход собственных рассуж- дений (включая тупиковые решения) в процессе выполнения конкретного домашне- го задания. Разделы "Остановка для размышлений" разбросаны по всему тексту, чтобы повысить активность читателей по решению задач. Ответы к задачам даются тут же; ♦ переработаннные и новые домашние задания. Настоящее издание книги содержит вдвое больше упражнений для домашней работы, чем предыдущее. Нечетко сфор- мулированные задания были исправлены или заменены новыми. Каждой задаче присвоен уровень сложности от I до 10; ♦ экзамены на основе материала книги. Студентам моих курсов по изучению алго- ритмов я обещаю, что все вопросы текущих контрольных работ и экзаменов в конце семестра будут взяты из домашних заданий в этой книге. Таким образом, студенты точно знают, что нужно изучать, чтобы успешно сдать экзамен. Для действенности этого подхода я тщательно подобрал количество, тип и сложность домашних зада- ний, следя затем, чтобы количество задач было оптимальным; ♦ разделы с подведением итогов. В этих разделах делается акцент на основных поня- тиях, которые нужно усвоить в данной главе;
16 Предисловие ♦ ссылки на задачи по программированию. В конце упражнений для каждой главы даются ссылки на 3-5 задач по программированию, взятых с веб-сайта http://www.programming-challenges.com. Эти задания можно использовать, чтобы добавить практический компонент реализации алгоритмов к теоретическому курсу; ♦ большой объем работающего программного кода вместо псевдокода. В этой книге увеличено количество алгоритмов, написанных на реальных языках программиро- вания (в частности, на языке С), за счет уменьшения объема псевдокода. Я считаю, что корректность и надежность проверенной работающей реализации дают ей пре- имущество над неформальным представлением простых алгоритмов. Полностью реализованные алгоритмы доступны на сайте http://www.algorist.com, ♦ замечания к главам. Каждая глава завершается кратким разделом с замечаниями, содержащими ссылки на основные источники информации и дополнительный спра- вочный материал. Благодарности Обновление посвящения через десять лет после выхода первого издания книги застав- ляет задуматься о скоротечности времени. С тех пор Рени стала моей женой, а потом матерью наших двух детей, Бонни и Эбби. Мой отец ушел в мир иной, но моя мама и мои братья Лен и Роб продолжают играть важную роль в моей жизни. Я посвящаю эту книгу членам моей семьи, новым и старым, тем, кто с нами и тем, кто покинул нас. Я бы хотел поблагодарить нескольких людей за их непосредственный вклад в новое издание. Эндрю Гон (Andrew Gaun) и Бетсон Томас (Betson Thomas) оказали мне по- мощь, в частности, при создании инфраструктуры нового сайта http:// www.cs.sunysb.edu/~algorith и при решении некоторых вопросов по подготовке ру- кописи. Дэвид Грайз (David Gries) дал ценные рекомендации в объеме, превышающем мои ожидания. Химаншу Гупта (Himanshu Gupta) и Бин Танг (Bin Tang) отважно ис- пользовали рукописную версию этого издания в своих академических курсах. Я также выражаю благодарность редакторам издательства Springer-Verlag Уэйну Уиллеру (Wayne Wheeler) и Алану Уайлду (Allan Wylde). Группа экспертов по разработке алгоритмов изучила материал книги, делясь со мной своими знаниями и извещая меня о новостях в этой области. Я благодарен всей группе, в состав которой входили: Ами Амир (Ami Amir). Хёрв Бронниманн (Herve Bronnimann), Бернар Шазель (Bernard Chazelle), Крис Чу (Chris Chu), Скотт Коттон (Scott Cotton), Ефим Диниц (Yefim Dinitz), Комеи Фукуда (Komei Fukuda), Майкл Гудрич (Michael Goodrich). Ленни Хит (Lenny Heath), Сихат Имамоглу (Cihat Imamoglu). Tao Жянг (Tao Jiang), Дэвид Каргер (David Karger), Джузеппе Лиотта (Giuseppe Liotta), Альберт Мао (Albert Мао). Сильвано Мартелло (Silvano Martello), Кэтрин Мак-Геох (Catherine McGeoch), Курт Мельхорн (Kurt Mehlhorn), Скотт А. Митчелл (Scott A. Mitchell). Насер Мескини (Naceur Meskini), Джин Майерс (Gene Myers), Гонзало Наварро (Gonzalo Navarro), Стивен Норт (Stephen North), Джо О'Рурк (Joe O’Rourke), Майк Патерсон (Mike Paterson), Тео Павлидис (Theo Pavlidis), Сет Петти (Seth Pettie), Мишель Почиола (Michel Pocchiola), Барт Пренил (Bart Preneel), Томаш Радзик
Предисловие 17 (Tomasz Radzik), Эдвард Рейнголд (Edward Reingold), Фрэнк Раски (Frank Ruskey), Питер Сэндерс (Peter Sanders). Жоао Сетубал (Joao Setubal), Джонатан Шевчук (Jonathan Shewchuk), Роберт Скил (Robert Skeel), Дженз Стон (Jens Stoye). Торстен Суэл (Torsten Suel). Брюс Уотсон (Bruce Watson) и Ури Цвик (Uri Zwick). Многие упражнения были созданы по подсказке коллег или под влиянием других ра- бот. Восстановление первоначальных источников годы спустя является задачей не из легких, но на веб-сайте книги дается ссылка на первоисточник каждой задачи (на- сколько мне удавалось его вспомнить). Было бы невежливо не поблагодарить людей, внесших важный вклад в первое издание книги. Рики Брэдли (Ricky Bradley) и Дарио Влах (Dario Vlah) создали мощную инфра- структуру для веб-сайта, логически стройную и легко расширяемую. Жонг Ли (Zhong Li) сделал большинство рисунков каталога задач с помощью графического редактора xfig. Ричард Крэндол (Richard Crandall), Рон Дэниэльсон (Ron Danielson), Такие Метак- сас (Takis Metaxas), Дэйв Миллер (Dave Miller), Гири Нарасимхан (Giri Narasimhan) и Джо Закари (Joe Zachary) проверили черновые версии первого издания. Их содержа- тельные отзывы и рекомендации помогли мне сформировать содержимое данного из- дания. Большую часть моих знаний об алгоритмах я получил, изучая их совместно с моими аспирантами. Многие из них— Йо-Линг Лин (Yaw-Ling Lin). Сундарам Гопалакриш- нан (Sundaram Gopalakrishnan), Тинг Чен (Ting Chen), Фрэнсин Иване (Francine Evans), Харальд Pay (Harald Rau), Рики Брэдли (Ricky Bradley) и Димитрис Маргаритис (Dimitris Margaritis) — являются персонажами историй, изложенных в книге. Мне все- гда было приятно работать и общаться с моими друзьями и коллегами из Университета Стоуни Брук — Эсти Аркином (Estie Arkin), Майклом Бэндером (Michael Bender). Джи Гао (Jie Gao) и Джо Митчеллом (Joe Mitchell). И, наконец, хочу сказать спасибо Майк- лу Брокстайну (Michael Brochstein) и остальным жителям города за то, что познакоми- ли меня с настоящего жизнью далеко за пределами Стоуни Брук. Ограничение ответственности Традиционно вину за любые недостатки в книге великодушно принимает на себя ее автор. Я же делать этого не намерен. Любые ошибки, недостатки или проблемы в этой книге являются виной кого-то другого; но я буду признателен, если вы поставите меня в известность о них, с тем. чтобы я знал, кто виноват. Стивен С. Скиена Кафедра вычислительной техники Университет Стоуни Брук Стоуни Брук, Нью-Йорк 11794-4400 http://www.cs.sunysb.edu/~skiena Апрель 2008 г.
ЧАСТЬ I Практическая разработка алгоритмов
ГЛАВА 1 Введение в разработку алгоритмов Что такое алгоритм? Это процедура выполнения определенной задачи. Алгоритм явля- ется основопола! ающей идеей любой компьютерной программы. Чтобы представлять интерес, алгоритм должен решать общую, корректно поставлен- ную задачу. Определение задачи, решаемой с помощью алгоритма, дается описанием всего множества экземпляров, которые должен обработать алгоритм, и выхода, т. е. результата, получаемого после обработки одного из этих экземпляров. Описание одно- го из экземпляров задачи может заметно отличаться от формулировки обшей задачи. Например, постановка задача сортировки делается таким образом: ЗАДАЧА. Сортировка. Вход. Последовательность из п элементов: а\,.... а». Выход. Перестановка элементов входной последовательности таким образом, что для ее членов справедливо соотношение а'| < сА < ... < а'„ । < а'„. Экземпляром задачи сортировки может быть массив имен, например {Mike. Bob, Sally. Jill, Jan}, или набор чисел, например {154, 245, 568, 324. 654, 324}. Первым шагом к решению — определить общую задачу. Алгоритм— это процедура, которая принимает любой из возможных входных экземп- ляров и преобразует его в соответствии с требованиями, указанными в условии задачи. Для решения задачи сортировки существует много разных алгоритмов. В качестве примера одного из таких алгоритмов можно привести метод сортировки вставками. Сортировка этим методом заключается во вставке в требуемом порядке элементов из неотсортированной части списка в отсортированную часть, первоначально содержа- щую один элемент. Реализация этого алгоритма на языке С представлена в листин- ге 1.1. Листинги. Реализация алгоритма сортировки вставками insert ion_sort (item s[], int n) 1 int i,j; /* Счетчики */ for (i=l; i<n; i++) { j=i; while ( (j>0) && (s[j] < s[j-l])) ( swap(&s[j], &s[ j-1]) ; 3 = j-i; )
22 Часть I. Практическая разработка алгоритмов А на рис. 1.) показано применение этого алгоритма для сортировки определенного эк- земпляра— строки INSERTIONSORT. Рис. 1.1. Пример использования алгоритма сортировки вставками (шкала времени направлена вниз) Обратите внимание на универсальность этого алгоритма. Его можно применять как для сортировки слов, так и для сортировки чисел, используя соответствующую операцию сравнения для определения, какое из двух значений поставить первым. Можно с лег- костью убедиться, что этот алгоритм правильно сортирует любой возможный набор входных величин в соответствии с нашим определением задачи сортировки. Хороший алгоритм должен обладать тремя свойствами: быть корректным, эффектив- ным и легкореализуемым. Получение комбинации всех трех свойств сразу может ока- заться недостижимой задачей. В производственной обстановке любая программа, ко- торая предоставляет достаточно хорошие результаты и не замедляет работу системы, в большинстве случаев является приемлемой, независимо оз того, возможны ли улуч- шения этих показателей. В этой области вопрос получения самых лучших возможных результатов или достижения максимальной эффективности обычно возникает только в случае серьезных проблем с производительностью или с законом. В этой главе основное внимание уделяется вопросу корректности алгоритмов, а их эф- фективность рассматривается в главе 2. Способность определенного алгоритма пра- вильно решить поставленную задачу, т. е. его корректность, редко поддается очевид- ной оценке. Алгоритмы обычно сопровождаются доказательством их правильности в виде объяснения, почему для каждого экземпляра задачи будет выдан требуемый ре- зультат. Но прежде чем продолжить обсуждение темы, мы продемонстрируем, что ар- гумент "это очевидно" никогда не является достаточным доказательством правильно- сти алгоритма. 1.1. Оптимизация маршрута робота Рассмотрим задачу, которая часто возникает на производстве и транспорте. Допустим, что нам нужно запрограммировать роботизированный манипулятор, который применя- ется для припаивания контактов интегральной схемы к контактным площадкам на пе- чатной плате. Чтобы запрограммировать манипулятор для выполнения этой задачи, сначала нужно установить порядок, в котором манипулятор припаивает первый кон-
"лава 1. Введение в разработку алгоритмов 23 такт, потом второй, третий и т. д.. пока не будут припаяны все. После обработки по- следнего контакта манипулятор возвращается к исходной позиции первого контакта для обработки следующей платы. Таким образом, маршрут манипулятора является замкнутым маршрутом, или циклом. Так как роботы являются дорогостоящими устройствами, мы хотим минимизировать время, затрачиваемое манипулятором на обработку платы. Будет логичным предполо- жить. что манипулятор перемещается с постоянной скоростью; соответственно, время перемещения от одной точки к другой прямо пропорционально расстоянию между точ- ками. То есть, нам нужно найти алгоритмическое решение такой задачи: ВАДАЧА. Оптимизация маршрута робота. Вход. Множество S' из » точек, лежащих на плоскости. Выход. Самый короткий маршрут посещения всех точек множества S. Прежде чем приступать к программированию маршрута манипулятора, нам нужно раз- работать алгоритм решения этой задачи. На ум может прийти несколько подходящих алгоритмов. Но самым подходящим будет эвристический алгоритм ближайшего сосе- да (nearest-neighbor heuristic). Начиная с какой-либо точки /?о. мы идем к ближайшей к ней точке (соседу) р\. От точки р\ мы идем к ее ближайшему еще не посещенному со- седу. таким образом исключая точку р0 из числа кандидатов на посещение. Процесс повторяется до тех пор, пока не останется ни одной не посещенной точки, после чего мы возвращаемся в точку рц, завершая маршрут. Псевдокод эвристического алгоритма ближайшего соседа представлен в листинге 1.2. Листинг 1.2. Эвристический алгоритм ближайшего соседа uarestNeighbor (Р) Из множества Р выбираем и посещаем произвольную начальную точку р Р = Р- 1 = О Пока остаются непосещенные точки 1-1 + 1 Выбираем и посещаем непосещенную точку р1( ближайшую к точке p,-i Посещаем точку pi Возвращаемся в точку ро от точки pn-i Этот алгоритм можно рекомендовать к применению по многим причинам. Он прост в понимании и легко реализуется. Вполне логично сначала посетить близлежащие точ- ки, чтобы сократить общее время прохождения маршрута. Алгоритм дает отличные результаты для входного экземпляра, показанного на рис. 1.2. Алгоритм ближайшего соседа достаточно эффективен, т. к. в нем каждая пара точек (р„р,) рассматривается, самое большее, два раза: первый раз при добавлении в мар- шрут точки р:, а второй — при добавлении точки р,. При всех этих достоинствах алго- ритм имеет всего лишь один недостаток — он совершенно неправильный. Неправильный? Да как он может быть неправильным? Поясню: несмотря на то. что алгоритм всегда создает маршрут, этот маршрут не обязательно будет самым коротким возможным маршрутом, или хотя бы приближающимся к таковому. Рассмотрим мно- жество точек, расположенных в линию, как показано на рис. 1.3.
24 Часть I Практическая разработка алгоритмов Рис. 1.2. C;iv чай удачного входного экземпляра для эвристического алюритма ближайшего соседа Рис. 1.3. Пример неудачного входного экземпляра для эвристического алгоритма ближайшего соседа (а) и оптимальное решение (6) Цифры на рисунке обозначают расстояние от начальной точки до соответствующей точки справа или слева. Начав обход с точки 0 и посещая затем ближайшего непосе- щенного соседа текущей точки, мы будем метаться вправо-влево через нулевую точку, т. к. алгоритм не содержит указания, что нужно делать в случае одинакового расстоя- ния между точками. Гораздо лучшее (более того, оптимальное) решение данного эк- земпляра задачи — начать обход с крайней левой точки и двигаться направо, посещая каждую точку, после чего возвратиться в исходное положение. Представьте себе реакцию вашего начальника при виде манипулятора, мечущегося вправо-влево при выполнении такой простой задачи. Можно сказать, что в данном случае проблема заключается в неудачном выборе от- правной точки маршрута. Почему бы не начать маршрут с самой левой точки в качест- ве точки рн? Это даст нам оптимальное решение данного экземпляра задачи. Верно на все 100%, но лишь до тех пор, пока мы не развернем множество точек на 90 градусов, сделав все точки самыми левыми. А если к тому же немного сдвинуть первоначальную точку 0 влево, то она опять будет выбрана в качестве отправной. Те- перь вместо дергания из стороны в сторону манипулятор будет скакать вверх-вниз, но время прохождения маршрута по-прежнему оставляет желать лучшего. Таким образом.
Глава 1. Введение в разработку алгоритмов 25 независимо от того, какую точку мы выберем в качестве исходной, алгоритм ближай- шего соседа обречен на неудачу с некоторыми экземплярами задачи (т. е. с некоторы- ми множествами точек), и нам нужно искать другой подход. Условие, заставляющее всегда искать ближайшую точку, является излишне ограничивающим, т. к. оно прину- ждает нас выполнять нежелательные переходы. Задачу можно попробовать решить другим способом— соединяя пары самых близких точек, если такое соединение не вызывает никаких проблем, например, досрочного завершения цикла. Каждая вершина рассматривается как самостоятельная одновершинная цепочка. Соединив все вместе, мы получим одну цепочку, содержащую все точки. Соединив две конечные точки, мы получим цикл. На любом этапе выполнения этого эвристического алгоритма ближай- ших пар у нас имеется множество отдельных вершин и не имеющих общих вершин цепочек, которые можно соединить. Псевдокод соответствующего алгоритма показан в листинге 1.3. Листинг 1.3. Эвристический алгоритм ближайших пар CksestPair (Р четь п - количество точек множества Р. ; : . n — do d - - рог каждой пары точек(s, t), не имеющих общих вершин цепей if dist (s, t) Д d then sm = s, t„ = t и d = dist (s, t) Соединяем (sm, t, ребром Соединяем две конечные точки ребром Для экземпляра задачи, представленного на рис. 1.3. этот алгоритм работает должным образом. Сначала точка 0 соединяется со своими ближайшими соседями, точками I н-1. Потом соединение следующих ближайших пар точек выполняется поочередно вле- во-вправо. расширяя центральную часть по одному сегменту за проход. Эвристический алгоритм ближайших пар чуть более сложный и менее эффективный, чем предыдущий, но. по крайней мере, для данного экземпляра задачи он дает правильный результат. Впрочем, это верно не для всех экземпляров. Посмотрите на результаты работы алго- ритма на рис. 1.4. а. Данный входной экземпляр состоит из двух рядов равномерно расположенных точек. Расстояние между рядами (I -е) несколько меньше, чем рас- стояние между смежными точками в рядах (I + е). Таким образом, пары наиболее близких точек располагаются не по периметру, а напротив друг друга. Сперва проти- воположные точки соединяются попарно, а затем полученные пары соединяются по- очередно по периметру. Общее расстояние маршрута алгоритма ближайших пар в этом случае будет равно 3(1 -е) + 2(1 + е) + -J(l - с)2 + (2 + 2е)~ . Этот маршрут на 20% длин- нее. чем маршрут на рис. 1.4.6. когда е~ 0. Более того, есть входные экземпляры, дающие значительно худшие результаты, чем этот. Таким образом, этот алгоритм тоже не годится. Какой из этих двух алгоритмов более эффективный? На этот вопрос нельзя ответить, просто посмотрев на них. Но очевидно, что оба алгоритма могут выдать очень плохие маршруты на некоторых с виду простых входных экземплярах. Но каков же правильный алгоритм решения этой задачи? Можно попробовать пере- числить все возможные перестановки множества точек, а потом выбрать перестановку.
26 Часть I. Практическая разработка алгоритмов сводящую г минимуму длину маршрута. Псевдокод этого алгоритма показан в листин- ге 1.4. Рис. 1.4. Неудачный входной экземпляр для алгоритма ближайших пар (а) и оптимальное решение (б) i Листинг 1.4. Оптимальный алгоритм поиска маршрута OptimalTSP(F Г] гое аждс перестановки Р из общего числа перестановок п! множества точек Р If 'cost(P ) d) then d - cost(P и Pmlr P, return P Так как рассматриваются все возможные упорядочения, то получение самого коротко- го маршрута гарантировано. Поскольку мы выбираем самую лучшую комбинацию, алгоритм правильный. В то же самое время он чрезвычайно медленный. Например, самый быстрый компьютер в мире не сможет в течение дня перечислить все 20! = 2 432 902 008 176 640 000 возможных перестановок 20 точек. А о реальных си- туациях. когда количество точек печатной платы достигает тысячи, можно и не гово- рить. Все компьютеры в мире, работая круглосуточно, не смогут даже приблизиться к решению этой задачи до конца существования Вселенной, а тогда решение этой зада- чи, скорее всего, уже не будет актуальным. Поиском эффективного алгоритма решения этой задачи, называющейся задачей ком- мивояжера (traveling salesman problem. TSP), мы будем заниматься на протяжении большей части этой книги. Если же вам не терпится узнать решение уже сейчас, то вы можете посмотреть его в разделе 16 4 Подведение итогов Алгоритмы. которые всегда выдают правильное решение, коренным образом отличаются от эвристических алгоритмов. которые обычно выдают достаточно хорошие, но не га- рантированные результаты. 1.2. Задача календарного планирования Теперь рассмотрим задачу календарного планирования. Представьте, что вы кинозвез- да и вам наперебой предлагают роли в разных кинофильмах, общее количество кото-
Глава 1. Введение в разработку алгоритмов 27 рых равно н. Каждое предложение имеет условие, что вы должны посвятить себя ему с первого до последнего дня съемок. Поэтому вы не можете сниматься одновременно в фильмах с полностью или частично накладывающимися периодами съемок. Ваш критерий для принятия того или иного предложения довольно прост: вы хотите заработать как можно больше денег. Поскольку вам платят одинаково за каждый фильм, то вы стремитесь получить роли в как можно большем количестве фильмов, периоды съемок которых не конфликтуют. На рис. 1.5 перечислены фильмы, в которых вам предлагают роли. Tarjan of the Jungle The Four Volume Problem The President's Algorisl Steiner's Tree Process Terminated Halting State Programming Challenges Рис. 1.5. Экземпляр задачи планирования пепересекающихся календарных периодов В данном случае очевидно, что вы можете сниматься, самое большее, в четырех филь- мах— "Discrete Mathematics", "Programming Challenges", "Calculated Bets", а потом в "Halting State" или в "Steiner's Tree". А в менее очевидных случаях вам (или вашему менеджеру) нужно будет решить сле- дующую алгоритмическую задачу календарного планирования: ЗАДАЧА. Планирование съемок в фильмах. Вход. Множество / интервалов времени п в линейном порядке. Выход. Самое большое подмножество пепересекающихся интервалов времени, кото- рое возможно во множестве I. Наум может прийти несколько способов решения этой задачи. Один из них основан на представлении, что надо работать всегда, когда это возможно. Это означает, что вам нужно брать роль в фильме, съемки которого начинаются раньше всех других. Псевдо- код этого алгоритма представлен в листинге 1.5. Листинг 1.5. Алгоритм первой возможной работы EarliestJobFirst (I) Из множества фильмов I берем роль в фильме j с самым ранним началом съемок, период которых не пересекается с периодом ваших предыдущих обязательств. Поступаем таким образом до тех пор, пока больше не останется таких фильмов. Этот подход выглядит логично, по крайней мере, до тех пор, пока вы не осознаете, что хватаясь за самую раннюю работу, можете пропустить несколько других, если первый фильм является сериалом. Пример такой ситуации показан на рис. 1.6, а, где самым ранним фильмом является киноэпопея "War and Peace", которая закрывает перед вами все другие перспективы. Этот пример заставляет искать другое решение. Проблема с фильмом "War and Peace" заключается в том, что его съемки длятся слишком долго. В таком случае, может быть.
28 Часть I. Практическая разработка алгоритмов вам следует орать роли только в фильмах с самыми короткими периодами съемок? Разве не очевидно, что чем быстрее вы закончите сниматься в одном фильме, тем раньше можно начать сниматься в другом, максимизируя таким образом количество фильмов в любой выбранный период времени? Псевдокод этого алгоритма представ- лен в листинге 1.6. War and Peace a) б) Рис. 1.6. Неудачные экземпляры задач для применения эвристики: самых ранних периодов (а). самых коротких периодов (б) Листинг 1.6. Алгоритм самого короткого периода Shortest TobFirst <I) While ; . 0) do Из ВС' . множества фильмов I берем фильм j с самым оотким периодом съемок Удаляем фильм i и любой другой фильм, период съемок которого пересекается г фильмом j, из множества доступных фильмов I Но и этот подход окажется действенным только до тех пор, пока вы не увидите, что можете упустить возможность заработать больше (рис. 1.6,6). Хотя в данном случае потери меньше, чем в предыдущем случае, тем не менее вы получите только половину возможных заработков. На данном этапе может показаться заслуживающим внимания алгоритм, который пе- ребирает все возможные комбинации. Если отвлечься от проверки подмножеств ин- тервалов (т. е. периодов съемок) на пересечение, этот алгоритм можно выразить псев- докодом, представленным в листинге 1.7. Листинг 1.7. Алгоритм полного перебора ExhaustiveScheduling(I) 0 For каждог" из 2 подмножеств Si множества интервалов I If (S непересекающееся) и (size(Si) > j) then size (Si) и Sr.4X = Si Return Sr™ Но насколько эффективным будет такой алгоритм? Здесь ключевым ограничением яв- ляется необходимость выполнить перечисление 2" подмножеств п элементов. А поло- жительным обстоятельством является то, что это намного лучше, чем перечисление всех и! порядков п элементов, как предлагается в задаче оптимизации маршрута робо- тизированного манипулятора. В данном случае при п = 20 имеется около миллиона подмножеств, которые можно за несколько секунд перебрать на современном компью-
Глава 1. Введение е разработку алгоритмов 29 терс. Но когда выбор фильмов возрастет до п = 100, то 211Ю будет намного больше, чем значение 20'. которое положило нашего робота на лопатки в предыдущей задаче. Разница между задачей составления маршрута и задачей календарного планирования заключается в том, что для последней имеется алгоритм, который решает задачу и пра- вильно, и эффективно. Для этого вам нужно брать роли только в фильмах с самым ранним окончанием съемок, т. е. выбрать такой временной интервал х, у которого пра- вая конечная точка находится левее правых конечных точек всех прочих временных интервалов. Таким фильмом в нашем примере (см. рис. 1.5) является "Discrete Mathematics". Вполне возможно, что съемки других фильмов начались раньше, чем съемки фильма х. но все они должны пересекаться друг с другом (по крайней мере, частично), поэтому мы можем выбрать, самое большее, один фильм изо всей группы. Съемки фильма х закончатся раньше всего, поэтому остальные фильмы с наклады- вающимися съемочными периодами потенциально блокируют другие возможности, расположенные справа от них. Очевидно, что выбрав фильм х, вы никак не можете проиграть. Псевдокод эффективного алгоритма правильного решения задачи кален- дарного планирования будет выглядеть так, как показано в листинге 1.8. Листинг 1.8. Оптимальный алгоритм календарного планирования 3₽timalScheduling (I) While (I * 0) do Из всего множества фильмов I выбираем фильм j с самым ранним окончанием съемок 'даляем фильм j и любой другой фильм, съемки которого пересекаются с фильмом j, из множества доступных фильмов I Обеспечение оптимального решения для всего диапазона возможных входных экземп- зяров является трудной, но. как правило, выполнимой задачей. Важной частью процес- са разработки такого алгоритма является поиск входных экземпляров задачи, которые опровергают наше допущение о правильности алгоритма-претендента на решение. Эффективные алгоритмы часто скрываются где-то совсем рядом, и в этой книге мы ютим помочь вам развить навыки их обнаружения. Подведение итогов Кажущиеся вполне логичными алгоритмы очень легко могут оказаться неправильными. Правильность алгоритма требует тщательного доказательства. 1.3. Обоснование правильности алгоритмов Будем надеяться, что предшествующие примеры продемонстрировали вам всю слож- ность темы правильности алгоритмов. Правильные алгоритмы выделяются из общего числа с помощью специальных инструментов, главный из которых называется доказа- тельством. Адекватное математическое доказательство состоит из нескольких частей. Прежде все- го, требуется ясная и четкая формулировка того, что вы пытаетесь доказать. Потом не- обходим набор предположений, которые всегда считаются верными и поэтому исполь-
30 Часть I. Практическая разработка алгоритмов зуются как часть доказательства. Далее, цепь умозаключений приводит нас от началь- ных предположений к конечному утверждению, которое мы пытаемся доказать. Нако- нец. небольшой черный квадрат в тексте указывает на конец доказательства. В этой книге формальным доказательствам не уделяется большого внимания, т. к. пра- вильное формальное доказательство привести очень трудно, а неправильное может вас сильно дезориентировать. На самом деле, доказательство является демонстрацией. Доказательства полезны только тогда, когда они простые и незамысловатые — ясные и лаконичные аргументы, объясняющие, почему алгоритм удовлетворяет требованию нетривиальной правильности. Правильные алгоритмы требуют тщательного изложения и определенных усилий для доказательства как их правильности, так и того факта, что они не являются неправиль- ными В последующих разделах мы разработаем инструменты для достижения этих целей. 1.3.1. Представление алгоритмов Цепь логических умозаключений об алгоритме невозможно построить без тщательного описания последовательности шагов, которые необходимо выполнить. Для этой цели наиболее часто употребляются, по отдельности или в совокупности, три формы пред- ставления алгоритма: обычный язык, псевдокод и язык программирования. Самым за- гадочным из этих средств представления алгоритма является псевдокод; это средство лучше всего можно определить как язык программирования, который никогда не выда- ет сообщений о синтаксических ошибках. Все три способа являются полезными, т. к. существует естественное стремление к компромиссу между легкостью восприятия и точностью представления алгоритма. Наиболее простым для понимания "языком про- граммирования" является обычный язык, но в то же время он наименее точен. С другой стороны, такие языки, как Java или C/C++, позволяют точно выразить алгоритм, но создавать и понимать алгоритмы на этих языках задача не из легких. В отношении сложности применения и понимания псевдокод представляет золотую середину между этими двумя крайностями. Выбор самого лучшего способа представления алгоритма зависит от ваших предпочте- ний. Я, например, сначала описываю свои алгоритмические идеи на обычном языке, а затем перехожу на более формальный псевдокод наподобие языка программирования или даже на настоящий язык программирования для уточнения сложных деталей. Не допускайте ошибку, которую часто делают мои студенты, — используют псевдо- код, чтобы приукрасить плохо определенную идею и придать ей более формальный и солидный вид. При описании алгоритма следует стремиться к ясности. Например, ал- горитм ExhaustiveScheduling (см. листинг 1.7) можно было бы выразить на обычном языке так, как показано в листинге 1.9. Листинг 1.9. Алгоритм полного перебора ExhaustiveScheduling(I) Протестировать все 2" подмножеств множества I и возвратить самое большое подмножество непересекающихся интервалов.
Глава 1. Введение в разработку алгоритмов 31 Подведение итогов В основе любого алгоритма лежит идея. Если в описании алгоритма не просматривается ясно ваша идея, значит, вы используете для ее выражения нотацию слишком низкого уровня. 1.3.2. Задачи и свойства Чтобы продемонстрировать правильность алгоритма, одного его описания недостаточ- но. Нам также нужно подробное описание задачи, которая подлежит решению. Постановка задачи состоит из двух частей: набора допустимых входных экземпляров и требований к выходу алгоритма. Невозможно доказать правильность алгоритма для нечетко поставленной задачи. Иными словами, поставьте неправильно задачу, и вы получите неправильный ответ. Постановки некоторых задач допускают слишком широкий диапазон входных экземп- ляров. Допустим, что в нашей задаче календарного планирования съемки фильмов мо- гут прерываться на некоторое время. Например, съемки фильма могут быть запланиро- ваны на сентябрь и ноябрь, а октябрь свободен. Тогда календарный план съемок для любого фильма будет состоять из набора временных интервалов. В этом случае мы можем браться за роли в двух фильмах с чередующимися, но не пересекающимися периодами съемок. Такую общую задачу календарного планирования нельзя решить с помощью алгоритма самого раннего окончания съемок. Более того, для решения этой общей задачи вообще не существует эффективного алгоритма. Подведение итогов Важным и заслуживающим внимания приемом разработки алгоритмов является сужение множества допустимых экземпляров задачи до тех пор, пока не будет найден правильный и эффективный алгоритм. Например, задачу общих графов можно свести к задаче деревь- ев, или двумерную геометрическую задачу свести к одномерной. При указании требований выхода алгоритма часто допускаются две ошибки. Первая из них— плохая формулировка вопроса. Примером может служить вопрос о наилучше.м маршруте между двумя точками на карте при отсутствии определения, что значит "наилучший". Лучший в каком смысле? В смысле самого короткого расстояния, самого короткого времени прохождения или, может быть, минимального количества пово- ротов? Второй ошибкой является формулирование составных целей. Каждый из трех только что упомянутых критериев наилучшего маршрута является четко определенной целью правильного эффективного алгоритма оптимизации. Но в качестве требования к реше- нию из этих критериев можно выбрать только один. Например, формулировка: "Найти самый короткий маршрут от точки к точке, содержащий количество поворотов не бо- лее чем в два раза превышающее необходимое" является вполне четкой постановкой задачи. Но решить такую задачу очень трудно, т. к. решение требует сложного анализа. Для примеров постановки задач читателю настоятельно рекомендуется ознакомиться с постановкой каждой из 75 задач во второй части этой книги. Правильная постановка задачи является важной частью ее решения. Изучение постановок всех этих классиче- ских задач поможет вам распознать задачи, постановкой и решением которых кто-то уже занимался до вас.
32 Часть I. Практическая разработка алгоритмов 1.3.3. Демонстрация неправильности алгоритма Самый лучший способ доказать неправильность алгоритма— найти экземпляр задачи, для которого алгоритм выдает неправильный ответ. Такие экземпляры задачи называ- ются контрпримерами. Никто не бросится на защиту алгоритма, для которого был предоставлен контрпример. Вполне разумно выглядящие алгоритмы можно момен- тально опровергнуть посредством очень простых контрпримеров. Хороший контрпри- мер должен обладать двумя важными свойствами: ♦ проверяемостью. Чтобы продемонстрировать, что некий входной экземпляр задачи является контрпримером для определенного алгоритма, требуется вычислить ответ, который алгоритм выдаст для данного экземпляра, и предоставить лучший ответ, с тем. чтобы доказать, что алгоритм не смог его найти; ♦ простотой. Хороший контрпример не содержит ничего лишнего и ясно демонстри- рует. почему именно предлагаемый алгоритм неправильный. Поэтому обнаружен- ный контрпример следует упростить. Контрпример на рис. 1.6, а можно упростить и улучшить, сократив количество пересекающихся периодов с четырех до двух. Развитие навыков поиска контрпримеров будет стоить затраченного времени. Этот процесс в чем-то подобен разработке наборов тестов для проверки компьютерных про- грамм, но в нем главную роль играет удачная догадка, а не перебор вариантов. Приве- ду несколько советов. ♦ Ищите мелкомасштабные решения. Обратите внимание, что мои контрпримеры для задач поиска маршрута содержат шесть или меньше точек, а для задач кален- дарного планирования — только три временных интервала. Это обстоятельство ука- зывает на то, что если алгоритм неправильный, то доказать это можно на очень про- стом экземпляре. Алгористы-любители нередко создают большой запутанный эк- земпляр. с которым потом не могут справиться. А профессионалы внимательно изучат несколько небольших экземпляров, т. к. их легче осмыслить и проверить. ♦ Рассмотрите все решения. Для наименьшего нетривиального значения п имеется только небольшое количество экземпляров. Например, существуют только три представляющих интерес способа расположения двух интервалов на прямой: непе- рекрываюшиеся интервалы, частично перекрывающиеся интервалы и полностью перекрывающиеся интервалы. Все случаи размещения на прямой трех интервалов (включая контрпримеры для обоих эвристических алгоритмов календарного плани- рования) можно создать, по-разному добавляя третий интервал к этим двум, распо- ложенным указанными тремя способами. ♦ Ищите слабое звено. Если рассматриваемый вами алгоритм работает по принципу "всегда берем самое большее" (так называемый жадный алгоритм), то подумайте, почему этот подход может быть неправильным. ♦ Ищите ограничения. "Жадный" эвристический алгоритм можно сбить с толку не- обычным методом предоставления входного экземпляра, содержащего одинаковые элементы. В этой ситуации алгоритму будет не на чем основывать свое решение и он, возможно, возвратит неоптимальное решение. ♦ Рассматривайте крайние случаи. Многие контрпримеры представляют собой ком- бинацию большого и малого, правого и левого, многих и немногих, далекого и
Гпава 1 Введение в разработку алгоритмов 33 близкого. Обычно легче осмыслить и проверить примеры по краям диапазона, чем из его середины. Рассмотрим в качестве примера два скопления точек, расстояние d между которыми намного превышает расстояние между точками в любом из них. Длина оптимального маршрута коммивояжера в этой ситуации будет практически равна 2d. независимо от количества посещаемых точек, т. к. длина маршрута внутри каждого скопления точек несущественна. Подведение итогов Лучший способ опровергнуть правильность эвристического алгоритма — испытать его на контрпримерах. 1.3.4. Индукция и рекурсия Один факт, что для данного алгоритма не был найден контрпример, вовсе не означает, что алгоритм правильный. Для этого требуется доказательство или демонстрация пра- вильности. Для доказательства правильности алгоритма часто применяется математи- ческая индукция. Мои первые впечатления о математической индукции были таковы, словно это какое- то шаманство. Вы берете формулу типа i = n(n + 1) / 2, доказываете ее для базово- го случая, например, 1 или 2. потом допускаете, что утверждение справедливо для л - 1, и на основе этого допущения доказываете формулу для общего п. Это называется доказательством? Полнейший абсурд! Мои первые впечатления о рекурсии в программировании были точно такими же — чистое шаманство. Программа проверяет входной аргумент на равенство базовому значению, например. 1 или 2. При отрицательном результате такой проверки более сложный случай решается путем разбивки его на части и вызова этой же программы в качестве подпрограммы для решения этих частей. Это называется программой? Пол- нейший абсурд! Причиной, благодаря которой как рекурсия, так и математическая индукция кажутся шаманством, является тот факт, что рекурсия и есть математическая индукция. В обе- их имеются общие и граничные условия, при этом общее условие разбивает задачу на все более мелкие части, а граничное, или начальное, условие завершает рекурсию. Если вы понимаете одно из двух,— или рекурсию, или индукцию.— вы в состоянии понять, как работает другое. Мне приходилось слышать, что программист— это математик, который умеет строить доказательства только методом индукции. Частично дело в том, что программисты — никудышные построители доказательств, но главным образом в том. что многие изучаемые ими алгоритмы являются или рекурсивными, или инкрементными (поэтап- ными). Рассмотрим, например, правильность алгоритма сортировки вставками, представлен- ного в начапе этой главы. Его правильность можно обосновать методом индукции: ♦ базовый экземпляр содержит всего лишь один элемент, а по определению одноэле- ментный массив является полностью отсортированным; 2 Зак 3741
34 Часть I. Практическая разработка алгоритме* ♦ мы можем допустить, что после первых п- 1 итераций сортировки вставками пер вые п- 1 элементов массива Л будут полностью отсортированы; ♦ чтобы определить, куда следует вставить последний элемент х, нам нужно найт! уникальную ячейку между наибольшим элементом, не превышающим х, и паи меньшим элементом, большим чем х. Для этого мы сдвигаем все большие элементь назад на одну позицию, создавая место для элементах в требуемой позиции. Но к индуктивным доказательствам нужно относиться с большой осторожностью, т. к в цепь рассуждений могут вкрасться трудно распознаваемые ошибки. Прежде всего это граничные ошибки. Например, в приведенном выше доказательстве правильного алгоритма сортировки вставками мы самоуверенно заявили, что между двумя элемен тами массива А имеется однозначно определяемая ячейка, в которую можно вставит! наш элемент х, когда массив в нашем базовом экземпляре содержит только одну ячей ку. Поэтому для правильной обработки частных случаев вставки минимальных ил1 максимальных элементов необходимо соблюдать большую осторожность. Другой, более распространенный, тип ошибок в индуктивных доказательствах связан < небрежным подходом к расширению экземпляра задачи. Добавление всего лишь одно го элемента к экземпляру задачи может полностью изменить оптимальное решение Соответствующий пример для задачи календарного планирования показан на рис. 1.7. Рис. 1.7. Оптимальное решение (прямоугольники) до и после внесения изменений (пунктирная линия) в экземпляр задачи После добавления нового временного интервала в экземпляр задачи новое оптималь- ное расписание может не содержать ни одного из временных интервалов любого опти- мального решения, предшествующего изменению. Бесцеремонное игнорирование таких аспектов может вылиться в очень убедительное доказательство полностью не- правильного алгоритма. Подведение итогов Математическая индукция обычно является верным способом проверки правильности рекурсивного алгоритма. Остановка для размышлений: Правильность инкрементных алгоритмов ЗАДАЧА. Доказать правильность рекурсивного алгоритма для инкрементации нату- ральных чисел, т. c.y—ty + 1, представленного в листинге 1.10. F".....................-------------_ . Листинг 1.10. Алгоритм для инкрементации натуральных чисел Increment (у) if у = 0 then return(1) else if (у mod 2) = 1 then return (2 • Increment ([Ly/2_l) ) else return(у + 1)
Глава 1 Введение в разработку алгоритмов 35 Решение. Лично мне правильность этого алгоритма определенно не очевидна. Но т. к. это рекурсивный алгоритм, а я — программист, мое естественное побуждение будет доказать его методом индукции. Абсолютно очевидно, что алгоритм правильно обрабатывает базовый случай, когда у = 0. Совершенно ясно, что 0+1 = 1 и, соответственно, возвращается значение 1. Теперь допустим, что функция работает правильно для общего случая, когда .у = и - 1. На основе этого допущения нам нужно продемонстрировать правильность алгоритма для случая, когда у = п. Для половины случаев алгоритм доказывается легко, в част- ности для четных чисел (для которых (у mod 2) = о), т. к. у + 1 возвращается явно. Но для нечетных чисел решение зависит от результата, возвращаемого выражением ’ncrement ([у/2]). Здесь нам хочется воспользоваться нашим индуктивным допущени- ем, но оно не совсем правильно. Мы сделали допущение, что функция increment рабо- тает правильно. когда у = п— 1, но для значения у, равного приблизительно половине этого значения, мы этого не допускали. Теперь мы можем усилить наше допущение, объявив, что общий случай выдерживается для всех у < и - 1. Это усиление никоим образом не затрагивает сам принцип, но необходимо, чтобы установить правильность алгоритма. Теперь случаи нечетных у (т. е. у - 2т + 1 для целого числа т) можно обработать, как показано в листинге 1.11. |Листинг 1.11, Алгоритм для инкрементации нечетных натуральных чисел /•Increment ([ (2m + 1) /2]) = 2* Increment (Lm + 1/2_|) = 2•Increment(m) = 2(m + 1) =2m+2=y+l решая, таким образом, общий случай. 1.3.5. Суммирование При анализе алгоритмов, часто приходится прибегать к математическим формулам суммирования. А процесс доказательства правильности формулы суммирования пред- ставляет собой классический пример применения математической индукции. В конце этой главы дается несколько упражнений, в которых требуется доказать формулу с по- мощью индукции. Чтобы сделать эти упражнения более понятными, напомню основ- ные принципы суммирования. Формулы суммирования представляют собой краткие выражения, описывающие сло- жение сколь угодно большой последовательности чисел. В качестве примера можно привести такую формулу: ЁЛО = /(!) +/(2)+ ... + /(«) /=1 Суммирование многих алгебраических функций можно выразить простыми формула- ми в замкнутой форме. Например, поскольку сумма п единиц равна п. то:
36 Часть I Практическая разработка алгоритмов п I?- ,=| Сумму первых п целых чисел можно выразить через целые числа / и (и - i + 1) следую- щим образом: н п/2 = ^ (/ + (« -i + 1)) = н(п + 1)/ 2 <=i <=1 Очень пригодится в области анализа алгоритмов умение распознавать два основных класса формул суммирования: ♦ Арифметические прогрессии. Арифметическую прогрессию в виде формулы = ' = «(« + ')7 2 можно встретить в анализе алгоритма сортировки методом выбора. По большому счету, важным фактом является наличие квадратичной сум- мы, а не то, что константа равняется 1/2. В общем, для р > I: \(а7,/Д=^/'>=0(«'’+1) I Таким образом, сумма квадратов кубичная, а сумма кубов— "четверичная" (если вы не против употребления такого слова). Нотация 0(х) (тета большое) подробно рассматривается в разделе 2.2. Для р < -1 эта сумма всегда стремится к константе, даже когда п—>со. ♦ Геометрические прогрессии. В геометрических прогрессиях индекс цикла играет роль показателя степени, т. е.: G(n.a) = У^а1 = а(а"+1 - 1) / (а - 1) 1=0 Сумма прогрессии зависит от ее знаменателя, т. е. числа а. При а < 1 эта сумма стремится к константе, даже когда /?—><». Данная сходимость последовательности оказывается большим подспорьем в анали- зе алгоритмов. Это означает, что если количество элементов растет линейно, то их сумма необязательно будет расти линейно, а может быть ограничена константой Например. 1 + 1/2 + 1/4 + 1/8 + ... < 2 независимо от количества элементов последо- вательности. При а > 1 сумма стремительно возрастает при добавлении каждого нового элемента, например, 1 + 2 + 4 + 8+ 16 +32 = 63. В самом деле, для а > 1 G(n, а) = 0(а" ’ *). Остановка для размышлений. Формулы факториала ЗАДАЧА, Докажите методом индукции, что х/1= (и + !)!-!. Решение. Индукционная парадигма прямолинейна: сначала подтверждаем базовыи случай (здесь мы принимаем п = 1, хотя случай п = 0 был бы еще более общим): । £/хП=1 = (1 + 1)!-1 = 2-1 = 1
Гпава 1 Введение в разработку алгоритмов 37 Теперь допускаем, что данное утверждение верно для всех чисел вплоть до п. Для до- казательства общего случая n + 1 видим, что если мы вынесем наибольший член из-под знака суммы W+I п / zх/! = (и + |)х(и + 1)! + У f х?! /=| <=i то получим левую часть нашего индуктивного допущения. Заменяя правую часть, по- лучаем: п+1 У< х i\ = (n + 1)х(и + 1)! + {п +1)! — I >=1 = (л + 1)!х((и +1)+1)-1 = (л + 2)!-1 Этот общий прием выделения наибольшего члена суммы для выявления экземпляра индуктивного допущения лежит в основе всех таких доказательств. 1.4. Моделирование задачи Моделирование является искусством формулирования приложения в терминах точно поставленных, хорошо понимаемых задач. Правильное моделирование является клю- чевым аспектом применения методов разработки алгоритмов решения реальных задач. Более того, правильное моделирование может сделать ненужным разработку или даже реализацию алгоритмов, соотнося ваше приложение с ранее решенной задачей. Пра- вильное моделирование является ключом к эффективному использованию материала во второй части этой книги. В реальных приложениях применяются реальные объекты. Вам, может быть, придется работать над системой маршрутизации сетевого трафика, планированием использова- ния классных комнат учебного заведения или поиском закономерностей в корпоратив- ной базе данных. Большинство же алгоритмов спроектировано для работы со строго определенными абстрактными структурами, такими как перестановки, графы или мно- жества. Чтобы извлечь пользу из литературы по алгоритмам, вам нужно научиться вы- полнять постановку задачи абстрактным образом, в терминах процедур над фундамен- тальными структурами. 1.4.1. Комбинаторные объекты Очень велика вероятность, что над решаемой вами алгоритмической задачей уже рабо- тали другие, хотя, возможно, и в совсем иных контекстах. Но не надейтесь узнать, что известно о вашей конкретной "задаче оптимизации процесса/", поискав в книге сло- восочетание "процесс Л". Вам нужно сформулировать задачу оптимизации вашего процесса в терминах вычислительных свойств стандартных структур, таких как: ♦ перестановка— упорядоченное множество элементов. Например, {1, 4, 3, 2} и {4. 3, 2, 1} являются двумя разными перестановками одного множества целых чи- сел. Мы уже видели перестановки в задачах оптимизации маршрута манипулятора и
38 Часть I Практическая разработка алгоритмов календарного планирования. Перестановки будут вероятным исследуемым объек- том в задаче поиска "размещения", "маршрута", "границ" или "последовательности"; ♦ подмножество— выборка из множества элементов. Например, множества {1, 3, 4} и {2} являются двумя разными подмножествами множества первых четырех целых чисел. В отличие от перестановок, порядок элементов подмножества не имеет зна- чения, поэтому подмножества {1, 3, 4} и {4, 3, 1} являются одинаковыми. В про- блеме календарного планирования нам пришлось иметь дело с подмножествами. Подмножества будут вероятным исследуемым объектом в задаче поиска "кластера", "коллекции", "комитета", "группы", "пакета" или "выборки"; Рис. 1.8. Моделирование реальных структур с помощью деревьев и графов ♦ дерево— иерархическое представление взаимосвязей между объектами. Реальное применение деревьев показано на примере родословного древа семейства Скиена на рис. 1.8, а. Деревья будут вероятным исследуемым объектом в задаче поиска "иерархии", "отношений доминирования", "отношений предок/потомок" или "так- сономии"; ♦ граф— представление взаимоотношений между произвольными парами объектов. На рис. 1.8, б показана модель дорожной сети в виде графа, где вершины представ- ляют населенные пункты, а ребра— дороги, соединяющие населенные пункты. Граф будет вероятным исследуемым объектом в задаче поиска "сети", "схемы", "инфраструктуры" или "взаимоотношений"; ♦ точка— представление места в некотором геометрическом пространстве. Напри- мер, расположение автобусных остановок можно описать точками на карте (плоско- сти). Точка будет вероятным исследуемым объектом в задаче поиска "местонахож- дения", "позиции", "записи данных" или "расположения"; ♦ многоугольник— представление области геометрического пространства. Например, с помощью полигона можно описать границы страны на карте. Многоугольники и многогранники будут вероятными исследуемыми объектами в задаче поиска "фор- мы", "региона", "очертания" или "границы"; ♦ строка— последовательность символов или шаблонов. Например, имена студентов можно представить в виде строк. Строка будет вероятным исследуемым объектом при работе с "текстом", "символами", "шаблонами" или "метками". Для всех этих фундаментальных структур имеются соответствующие алгоритмы, кото- рые представлены в каталоге задач во второй части этой книги. Важно ознакомиться с
Глава 1. Введение в разработку алгоритмов 39 этими задачами, т. к. они они изложены на языке, типичном для моделирования при- ложений. Чтобы научиться свободно владеть этим языком, просмотрите задачи в ката- логе и изучите рисунки входа и выхода для каждой из них. Разобравшись в этих зада- чах, хотя бы на уровне рисунков и формулировок, вы будете знать, где искать возмож- ный ответ в случае возникновения проблем в разработке вашего приложения. В книге также представлены примеры успешного применения моделирования прило- жений в виде описаний решений реальных задач. Однако здесь необходимо высказать одно предостережение. Моделирование сводит разрабатываемое приложение к одной из многих существующих задач и структур. Такой процесс по своей природе является ограничивающим, и некоторые нюансы модели могут не соответствовать вашей кон- кретной задаче. Кроме того, встречаются задачи, которые можно моделировать не- сколькими разными способами. Моделирование является всего лишь первым шагом в разработке алгоритма решения задачи. Внимательно отнеситесь к отличиям вашего приложения от потенциальной модели, но не спешите с заявлением, что ваша задача является уникальной и особен- ной. Временно игнорируя детали, которые не вписываются в модель, вы сможете найти ответ на вопрос, действительно ли они являются принципиально важными. Подведение итогов Моделирование разрабатываемого приложения в терминах стандартных структур и алго- ритмов является важнейшим шагом в поиске решения. 1.4.2. Рекурсивные объекты Научившись мыслить рекурсивно, вы научитесь определять большие сущности, со- стоящие из меньших сущностей точно такого же типа, как и большие. Например, если рассматривать дом как набор комнат, то при добавлении или удалении комнаты дом остается домом. В мире алгоритмов рекурсивные структуры встречаются повсеместно. Более того, каж- цую из ранее описанных абстрактных структур можно считать рекурсивной. На рис. 1.9 видно, как легко они разбиваются на составляющие. Перечислим возможные рекурсивные объекты. ♦ Перестановки. Удалив первый элемент перестановки {1,...,и}, мы получим переста- новку остающихся п - 1 элементов. ♦ Подмножества. Каждое множество элементов {!,...,/?} содержит подмножество { 1- 1}, являющееся результатом удаления элемента п, если такой имеется. ♦ Деревья. Что мы получим, удалив корень дерева? Правильно, коллекцию меньших деревьев. А что мы получим, удалив какой-либо лист дерева? Немного меньшее де- рево. ♦ Графы. Удалите любую вершину графа, и вы получите меньший граф. Теперь разо- бьем вершины графа на две группы, правую и левую. Разрезав все ребра, соеди- няющие эти группы, мы получим два меньших графа и набор разорванных ребер.
40 Часть I. Практическая разработка алгоритмов (4.1,5.2,3) (1.2.7.9) 4+( 1.4.2,3) 9+11,2.7) Рис. 1.9. Рекурсивное разложение комбинаторных объектов: перестановки, подмножества деревья, графы, множества точек, многоугольники и строки ♦ Точки. Возьмем облако точек и разобьем его линией на две группы. Теперь у нас есть два меньших облака точек. ♦ Многоугольники. Соединив хордой две несмежные вершины простого многоуголь- ника с п вершинами, мы получим два меньших многоугольника. ♦ Строки. Что мы получим, удалив первый символ строки? Другую, более короткую строку. Для рекурсивного описания объектов требуются как правила разложения, так и базо- вые объекты, а именно спецификация простейшего объекта, далее которого разложе- ние не идет. Такие базовые объекты обычно легко определить. Перестановки и под- множества нулевого количества объектов обозначаются как {}. Наименьшее дерево или граф состоят из одной вершины, а наименьшее облако точек состоит из одной точ- ки. С многоугольниками немного посложнее; наименьшим многоугольником является треугольник. Наконец, пустая строка содержит нулевое количество знаков. Решение о том, содержит ли базовый объект нулевое количество элементов или один элемент, является, скорее, вопросом вкуса и удобства, нежели какого-либо фундаментального принципа. Такие рекурсивные разложения применяются для определения многих алгоритмов, рассматриваемых в этой книге. Обращайте на них внимание. 1.5. Истории из жизни Самый лучший способ узнать, какое огромное влияние тщательная разработка алго- ритма может иметь на производительность, — ознакомиться с примерами из реальной жизни. Внимательно изучая опыт других людей, мы учимся использовать этот опыт в нашей собственной работе. В различных местах этой книги приводится несколько рассказов об успешном (а ино- гда и неуспешном) опыте нашей команды в разработке алгоритмов решения реальных задач. Я надеюсь, что вы сможете перенять и усвоить опыт и знания, полученные нами
Глава 1. Введение в разработку алгоритмов 41 в процессе этих разработок, чтобы использовать их в качестве моделей для ваших соб- ственных решений. Все, изложенное в этих рассказах, действительно произойти Конечно же, в изложе- нии истории слегка приукрашены, и диалоги в них были отредактированы. Но я при- ложил все усилия, чтобы правдиво описать весь процесс, начиная от постановки задачи № выдачи решения, чтобы вы могли проследить его в действии. В своих рассказах я пытаюсь уловить образ мышления алгориста в процессе решения ’адачи. Эти истории из жизни обычно затрагивают, по крайней мере, одну, а часто несколько, задач из каталога во второй части книги. Когда такая задача встречается, дается ссылка на соответствующий раздел каталога. Таким образом подчеркиваются достоинства мо- делирования разрабатываемого приложения в терминах стандартных алгоритмических задач. Пользуясь каталогом задач, вы сможете в любое время получить известную ин- формацию о решаемой задаче. 1.6. История из жизни. Моделирование проблемы ясновидения Я занимался обычными делами, когда на меня нежданно-негаданно свалился этот за- прос в виде телефонного звонка. — Профессор Скиена, я надеюсь, что Вы сможете помочь мне. Я — президент компа- нии Lotto System Group, Inc., и нам нужен алгоритм решения проблемы, возникающей в нашем последнем продукте. — Буду рад помочь Вам, — ответил я. В конце концов, декан моего инженерного фа- кульгета всегда призывает преподавательский состав к более широкому взаимодейст- вию с производством. — Наша компания продает программу, предназначенную для развития способностей наших клиентов к ясновидению и предсказанию выигрышных лотерейных номеров'. Стандартный лотерейный билет содержит шесть номеров, выбираемых из большего количества последовательных номеров, например, от 1 до 44. Таким образом, шансы выигрыша любой комбинации номеров очень небольшие. Но после соответствующей тренировки наши клиенты могут мысленно увидеть, скажем, 15 номеров из 44 возмож- ных, по крайней мере, четыре из которых будут на выигрышном билете. Вы все пока понимаете? -Скорее нет, чем да,— сказал я, но тут вспомнил, что наш декан призывает нас взаимодействовать с производством. — Наша проблема заключается в следующем. После того, как ясновидец сузил выбор номеров от 44 до 15, из которых он уверен в правильности, по крайней мере, четырех, нам нужно найти эффективный способ применить эту информацию. Допустим, уга- давшему, по крайней мере, три правильных номера выдается денежный приз. Нам ну- Это реальный случай.
42 Часть I. Практическая разработка алгоритмов жен алгоритм, чтобы составить наименьший набор билетов, которые нужно купить, чтобы гарантировать выигрыш, по крайней мере, одного приза. — При условии, что ясновидец не ошибается? — Да, при условии, что ясновидец не ошибается. Нам нужна программа, которая рас- печатывает список всех возможных комбинаций выигрышных номеров билетов, кото- рые он должен купить с минимальными затратами. Можете ли вы помочь нам с реше- нием этой задачи? Может быть, они и в самом деле были ясновидцами, т. к. они обратились как раз туда, куда надо. Определение наилучшего подмножества номеров билетов попадает в разряд комбинаторных задач. А точнее, это какой-то тип задачи о покрытии множества, в ко- торой каждый покупаемый билет "покрывает" некоторые из возможных четырехэле- ментных подмножеств увиденного ясновидцем пятнадцатиэлементного множества. Определение наименьшего набора билетов для покрытия всех возможностей представ- ляет собой особый экземпляр NP-полной задачи о покрытии множества (рассматри- вается в разделе 18.1) и считается вычислительно неразрешимой. Данная задача действительно была особым экземпляром задачи о покрытии множест- ва, полностью определяемая всего лишь четырьмя числами: размером п возможного множества S (п~ 15), количеством номеров к на каждом билете (к — 6), количеством обещаемых ясновидцем выигрышных номеров j из множества S (J• = 4) и, наконец, ко- личеством совпадающих номеров /, необходимых, чтобы выиграть приз (/ = 3). На рис. 1.10 показано покрытие экземпляра меньшего размера, где п — 5, j = к = 3,1 = 2. Рис. 1.10. Покрытие всех пар множества {1, 2. 3, 4. 5) номерами билетов [ 1, 2, 3), {1, 4, 5}, {2, 4, 5}, 13. 4. 5} — Хотя найти точный минимальный набор билетов с помощью эвристических методов будет трудно, я должен буду дать вам покрывающий набор номеров, достаточно близ- кий к самому дешевому, — ответил я ему. — Вам этого будет достаточно? — При условии, что ваша программа создает лучший набор номеров, чем программа моего конкурента, это будет то, что надо. Его система не всегда гарантирует выигрыш. Очень признателен за вашу помощь, профессор Скиена.
Глава 1. Введение в разработку алгоритмов 43 — Один вопрос напоследок. Если с помощью вашей программы люди могут натрени- роваться выбирать выигрышные лотерейные билеты, то почему вы сами не пользуетесь ею, чтобы выигрывать в лотерею? — Надеюсь встретиться с Вами в ближайшее время, профессор Скиена. Благодарю за помощь. Я повесил трубку и начал обдумывать дальнейшие действия. Задача выглядела, как идеальный проект для какого-либо смышленого студента. После моделирования задачи посредством множеств и подмножеств основные компоненты решения выглядели до- вольно просто. ♦ Нам было нужна возможность генерировать все подмножества к номеров из потен- циального множества S. Алгоритмы генерирования и ранжирования подмножеств представлены в разделе 14 5. ♦ Нам была нужна правильная формулировка, что именно означает требование иметь покрывающее множество приобретенных билетов. Очевидным критерием того, что требование выполнено, мог быть наименьший набор билетов, включающий, по крайней мере, один билет, содержащий каждое из (/) /-подмножеств множества S', за которое может выдаваться приз. ♦ Нам нужно было отслеживать уже рассмотренные призовые комбинации. Нам нуж- ны такие комбинации номеров билетов, чтобы покрыть как можно больше еще не охваченных призовых комбинаций. Текущие охваченные комбинации являются подмножеством всех возможных комбинаций. Структуры данных для подмножеств рассматриваются в разделе 12.5. Наилучшим кандидатом выглядел вектор битов, который бы за постоянное время давал ответ, охвачена ли уже данная комбинация. ♦ Нужен был механизм поиска, чтобы решить, какой следующий билет покупать. Для небольших множеств можно было проверить все возможные подмножества комби- наций и выбрать из них самую меньшую. Для множеств большего размера выиг- рышные комбинации номеров билетов для покупки можно было выбирать с по- мощью какого-либо процесса рандомизированного поиска наподобие имитации от- жига (см. раздел 7.5.3), чтобы покрыть как можно больше комбинаций. Выполняя эту рандомизированную процедуру несколько раз и выбирая самые лучшие реше- ния, мы смогли бы, скорее всего, получить хороший набор комбинаций номеров билетов. Опуская детали механизма поиска, требуемый алгоритм можно выразить на псевдоко- де, как показано в листинге 1.12. --------- ---------------------------------------------..--...— Листинг 1.12. Поиск набора призовых комбинаций , LottoTicketSet (n,k,l) Инициализируем как "ложь" все элементы (^-элементного вектора разрядов V While V содержит элементы "ложь" Выбираем k-элементное подмножество Т множества {1,..., п} в качестве следующей комбинации номеров покупаемого билета
44 Часть I. Практическая разработка алгоритмов For каждого из 1-элементньк подмножеств Т, множества Т, V(rank(TJ] = "истина" Выдаем набор комбинаций номеров билетов для покупки Способный студент Фаяз Юнас (Fayyaz Younas) принял вызов реализовать алгоритм. На основе представленной основы он реализовал алгоритм поиска методом полного перебора и смог найти оптимальные решения задач, в которых п < 5. Для решения за- дачи с большим значением п он реализовал процедуру рандомизированного поиска, которую отлаживал, пока не остановился на самом лучшем варианте. Наконец насту- пил день, когда мы могли позвонить в компанию Lotto Systems Group и сказать им, что мы решили задачу. — Результат работы нашей программы таков: оптимальным решением для п- 15, k = 6,j = 4,1 = 3 будет покупка 28 билетов. — Двадцать восемь билетов! — выразил недовольство президент. — В вашей про- грамме, должно быть, есть ошибка. Вот пять билетов, которых будет достаточно, что- бы покрыть все варианты дважды: {2, 4, 8. 10, 13, 14}, {4, 5, 7, 8, 12, 15}, {1,2, 3. 6, 11. 13}, {3, 5,6,9. 10, 15}. {1,7,9, И, 12, 14}. Мы повозились с этим примером немного и должны были признать, что он был прав. Мы неправильно смоделированы задачу! В действительности, нам не нужно было по- крывать явно все возможные выигрышные комбинации. Объяснение представлено на рис. 1.11в виде двухбилетного решения нашего предыдущего четырехбилетного при- мера. Рис. 1.11. Гарантирование выигрышной комбинации из двух номеров множества {1. 2, 3, 4, 5; при использовании только комбинации номеров {I. 2, 3J и {1, 4, 5', Такие малообещающие комбинации, как {2, 3, 4} и {3, 4, 5}, имеют совпадающие пары комбинаций номеров в билетах, показанных на рис. 1.11. Мы пытались покрыть слиш- ком много комбинаций, а дрожащие над каждой копейкой ясновидцы не желали пла- тить за такую расточительность. К счастью, у этой истории хороший конец. Общий принцип нашего поискового реше- ния оставался применимым для реальной задачи. Все, что нужно было сделать, так это
Гиава 1. Введение в разработку алгоритмов 45 исправить, какие подмножества засчитываются за покрытие данным набором комби- наций номеров билетов. Сделав это исправление, мы получили требуемые результаты. В компании Lotto Systems Group с благодарностью приняли нашу программу для вне- дрения в свой продукт. Будем надеяться, что они сорвут большой куш с ее помощью. Мораль этой истории заключается в том, что необходимо удостовериться в правильно- сти моделирования задачи, прежде чем пытаться решить ее. В нашем случае мы разра- ботали правильную модель, но недостаточно глубоко проверили ее, перед тем, как на- чинать создавать программу на ее основе Наше заблуждение было бы быстро выявле- но, если бы. прежде чем приступать к решению реальной проблемы, мы проработали небольшой пример вручную и обсудили результаты с нашим клиентом. Но мы смогли выйти из этой ситуации с минимальными отрицательными последствиями благодаря правильности нашей первоначальной формулировки и использованию четко опреде- ленных абстракций для таких задач, как генерирование ^-элементных подмножеств методом ранжирования, структуры данных множества и комбинаторного поиска. Замечания к главе В каждой хорошей книге, посвященной алгоритмам, отражается подход ее автора к их разработке. Тем, кто хочет ознакомиться с альтернативными точками зрения, особенно рекомендуется прочитать книги [CLRS01], [К.Т06] и [Мап89]. Формальные доказательства правильности алгоритма являются важными и заслужива- ют более полного рассмотрения, чем можно предоставить в этой главе. Методы вери- фикации программ обсуждаются в книге [Gri89], Задача календарного планирования ролей в фильмах является особым случаем общей задачи независимого множества, которая рассматривается в разделе 16.2. В качестве входных экземпляров допускаются только интервальные графы, в которых вершины графа G можно представить линейными интервалами, a (z, j) является ребром G тогда и только тогда, когда интервалы пересекаются. Этот интересный и важный класс графов подробно рассматривается в книге [Gol04]. Колонка "Программистские перлы" Джона Бентли (Jon Bentley) является, наверное, самой известной коллекцией историй из жизни разработчиков алгоритмов. Колонка первоначально публиковалась в журнале "Communications of the АСМ", а потом ее ма- териалы были изданы в двух книгах, [Веп90] и [Веп99]. Еще одна прекрасная коллек- ция историй из жизни собрана в книге [Вго95]. Хотя эти истории имеют явный уклон в сторону разработки и проектирования программного обеспечения, они также пред- ставляют собой кладезь мудрости. Все программисты должны прочитать эти книги, чтобы получить и пользу, и удовольствие. Наше решение задачи о покрытии множества лотерейных билетов более полно пред- ставлено в книге [YS96]. 1.7. Упражнения Поиск контрпримеров I. [3] Докажите, что значение а + Ь может быть меньшим, чем значение min(a, Ь). 2. [3] Докажите, что значение а* b может быть меньшим, чем значение min(a, b).
46 Часть I. Практическая разработка алгоритмов 3. [5] Начертите сеть дорог с двумя точками а и Л, такими, что маршрут между ними, пре- одолеваемый за кратчайшее время, не является самым коротким. 4. [5] Начертите сеть дорог с двумя точками а и Ь, самый короткий маршрут между кото- рыми не является маршрутом с наименьшим количеством поворотов. 5. [4] Задача о рюкзаке: имея множество целых чисел S = {sb s2, .... s,,} и целевое число Т, найти такое подмножество множества 5, сумма которого в точности равна Т. Например, множество 5 = {1, 2, 5, 9, 10} содержит такое подмножество, сумма элементов которого равна Т= 22, но не Т= 23. Найти контрпримеры для каждого из следующих алгоритмов решения задачи о рюкзаке, т. е., нужно найти такое множество 5 и число Т, при которых подмножество, выбранное с помощью данного алгоритма, не до конца заполняет рюкзак, хотя правильное решение и существует: • вкладывать элементы множества S в рюкзак в порядке слева направо, если они под- ходят (т. е., алгоритм "первый подходящий"); • вкладывать элементы множества S в рюкзак в порядке от наименьшего до наиболь- шего (т. е., используя алгоритм "первый лучший"); • вкладывать элементы множества S в рюкзак в порядке от наибольшего до наимень- шего. 6. [5] Задача о покрытии множества: имея семейство подмножеств S.. S„, универсального множества U = {!,...,«}, найдите семейство подмножеств Т с S наименьшей мощности, чтобы U,, е Т'' =U . Например, для семейства подмножеств S) = {1, 3, 5}, S2 = {2, 4}, S) = {1,4} и S.t = {2, 5} покрытием множества будет семейство подмножеств ,9, и 52. При- ведите контрпример для следующего алгоритма выбираем самое мощное подмножество для покрытия, после чего удаляем все его элементы из универсального множества; по- вторяем добавление подмножества, содержащего наибольшее количество неохваченных элементов, пока все элементы не будут покрыты. Доказательство правильности 7. [3] Докажите правильность следующего рекурсивного алгоритма умножения двух нату- ральных чисел для всех целочисленных констант с > 2: function multiply(у, z) comment Return произведение yz. 1. if z = 0 then return(0) else 2. return(multiply(cy,[z/c]) + у(z mod c)) 8. [3] Докажите правильность следующего алгоритма вычисления полинома Р(х) = = a„xn + а„ ix" ' + ...+ atx + а0: function homer (А, х) Р = Ап for i from п-1 to 0 р = р * х + Ai return р 9. [3] Докажите правильность следующего алгоритма сортировки: function bubblesort (А : list[l... п]) var int i,j
глаеа 1. Введение в разработку алгоритмов 47 for i from n to 1 for j from 1 to i - 1 if (A[ j]>A[j+l] меняем местами значения A[j] и A[j + 1] Математическая индукция Для доказательства пользуйтесь методом математической индукции. 10. [3] Докажите, что ^"=|/ = п(п +1)/2 для и>0. 11. [3] Докажите, что у" = л(л + 1)(2л +1) / 6 для п>0. 12. [3] Докажите, что = «‘(” + /4 для л>0. 13 [3] Докажите, что £ i(i +1)(/ + 2) = п(п + 1)(и + 2)(л + 3) / 4. /=| " а"+1 — 1 14. [5] Докажите, что Уо1 =----для n>l,a* 1. ,0 и ] п 15 [3] Докажите, что V---=-----для л>0. 7^/(/+1) л + 1 16. [3] Докажите, что п3 + 2л делится на 3 для л>0. 17. [3] Докажите, что дерево с п вершинами имеет в точности п — 1 ребер. 18. [3] Докажите, что сумма кубов первых п положительных целых чисел равна квадрату суммы этих целых чисел, т. е.. П п 5У=(5>)2 J=l /=| Приблизительные подсчеты 19. [3] Содержат ли все ваши книги, по крайней мере, миллион страниц? Каково общее ко- личество страниц всех книг в вашей институтской библиотеке? 20. [3] Сколько слов содержит эта книга? 21. [3] Сколько часов составляет один миллион секунд? А сколько дней? Выполните все необходимые вычисления в уме. 22. [3] Сколько городов и поселков в Соединенных Штатах? 23. [3] Сколько кубических километров воды изливается из устья Миссисипи каждый день? Не пользуйтесь никакой справочной информацией. Опишите все предположения, сде- ланные вами для получения ответа. 24. [3] В каких единицах измеряется время доступа к жесткому диску, в миллисекундах (тысячных долях секунды) или микросекундах (миллионных долях секунды)? Сколько времени занимает доступ к слову в оперативной памяти вашего компьютера, больше или меньше микросекунды? Сколько инструкций может выполнить центральный про-
48 Часть I. Практическая разработка алгоритмов цессор вашего компьютера в течение года, если компьютер постоянно держать вклю- ченным? 25. [4] Алгоритм сортировки выполняет сортировку I 000 элементов за I секунду. Сколько времени займет сортировка 10 000 элементов. • если время исполнения алгоритма прямо пропорционально и2? • если время исполнения алгоритма, по грубым оценкам, пропорционально «log/?? Проекты по реализации 26. [5] Реализуйте два эвристических алгоритма решения задачи коммивояжера из разде- ла 1.1. Какой из них выдает на практике более качественные решения? Можете ли вы предложить эвристический алгоритм, работающий лучше любого из них? 27. (5] Опишите способ проверки достаточности покрытия данным множеством комбина- ций номеров лотерейных билетов из задачи в разделе 1.6. Напишите программу поиска хороших множеств комбинаций номеров билетов. Задачи, предлагаемые на собеседовании 28. [5] Напишите функцию деления целых чисел, которая не использует ни оператор деле- ния (/). ни оператор умножения (*). Функция должна быть быстродействующей. 29. |5] У вас есть 25 лошадей. В каждой скачке может участвовать не больше 5 лошадей. Требуется определить первую, вторую и третью по скорости лошадь. Найдите мини- мальное количество скачек, позволяющих решить эту задачу. 30. [3] Сколько настройщиков пианино во всем мире? 31. [3] Сколько бензоколонок в Соединенных Штагах? 32. [3] Сколько весиз лед на хоккейном поле? 33. [3] Сколько километров дорог в Соединенных Штатах? 34. [3] Сколько раз, в среднем, нужно открыть наугад телефонный справочник Манхэттена, чтобы найти определенного человека? Задачи по программированию Эти задачи доступны на сайтах http://www.programniing-challenges.com и http://uva.onlinejudge.org. Идентификатор задачи на соответствующем сайте указыва- ется после названия задачи в форме "число/число". Так как сайты англоязычные, на- звания задач даются на исходном языке. 1. TheЗп+ 1 problem. 110101/100. 2. The Trip. 110103/10137. 3. Australian Voting. 110108/10142.
ГЛАВА 2 Анализ алгоритмов Алгоритмы являются принципиально важным компонентом информатики, т. к. их изу- чение не требует использования языка программирования или компьютера. Это озна- чает необходимость в методах, позволяющих сравнивать эффективность алгоритмов, не прибегая к их реализации. Самыми значимыми из этих инструментов являются мо- дель вычислений RAM и асимптотический анализ сложности наихудших случаев. Для оценки производительности алгоритмов применяется асимптотическая нотация. Хотя практик может прийти в ужас от самой идеи теоретического анализа алгоритмов, этот материал представлен здесь в силу его исключительной ценности при работе (алгоритмами. Этот способ оценки производительности является наиболее трудным материалом в данной книге. Но когда вы поймете основу этих идей на интуитивном уровне, вам будет намного легче разобраться в формальной составляющей. 2.1. Модель вычислений RAM Разработка машинно-независимых алгоритмов основывается на гипотетическом ком- пьютере. называющемся машиной с произвольным доступом к памяти (Random Access Machine) или RAM-машиной. Согласно этой модели наш компьютер работает таким образом: ♦ для исполнения любой простой операции (+, +, =. if, call) требуется ровно один временной шаг; ♦ циклы и подпрограммы не считаются простыми операциями, а состоят из несколь- ких простых операций. Нет смысла считать подпрограмму сортировки одношаговой операцией, т. к. для сортировки 1 000 000 элементов потребуется определенно на- много больше времени, чем для сортировки десяти элементов. Время исполнения цикла или подпрограммы зависит от количества итераций или специфического харак- тера подпрограммы; ♦ каждое обращение к памяти занимает один временной шаг. Кроме этого, наш ком- пьютер обладает неограниченным объемом оперативной памяти. Кэш и диск в мо- дели RAM не применяются. Время исполнения алгоритма в RAM-модели вычисляется по общему количеству ша- гов, требуемых алгоритму для решения данного экземпляра задачи. Допуская, что наша RAM-машина исполняет определенное количество шагов/операций за секунду, количе- ство шагов легко перевести в единицы времени. Может показаться, что RAM-модель является слишком упрощенным представлением работы компьютеров. В конце концов, на большинстве процессоров умножение двух
50 Часть I. Практическая разработка алгоритмов чисел занимает больше времени, чем сложение, что не вписывается в первое предпо- ложение модели. Второе предположение может быть нарушено удачной оптимизацией цикла компилятором или гиперпотоковыми возможностями процессора. Наконец, вре- мя обращения к данным может значительно разниться в зависимости от расположения данных: в кэше, в оперативной памяти или на диске. Таким образом, по сравнению с настоящим компьютером, все три основные допущения для RAM-машины неверны. Тем не менее, несмотря на эти несоответствия настоящему компьютеру, RAM-модель является превосходной моделью для понимания того, как алгоритм будет работать на настоящем компьютере. Она обеспечивает хороший компромисс, отражая поведение компьютеров и одновременно являясь простой в использовании. Эти характеристики делают RAM-модель полезной для практического применения. Любая модель полезна лишь в определенных рамках. Возьмем, например, модель пло- ской Земли. Можно спорить, что это неправильная модель, т. к. еще древние греки зна- ли, что в действительности Земля круглая. Но модель плоской Земли достаточно точна для закладки фундамента дома. Более того, в данном случае с моделью плоской Земли настолько удобнее работать, что использование модели сферической Земли1 для этой цели даже не приходит в голову. Та же самая ситуация наблюдается и в случае с RAM-моделью вычислений— мы соз- даем, вообще говоря, очень полезную абстракцию. Довольно трудно создать алгоритм, для которого RAM-модель выдаст существенно неверные результаты. Устойчивость RAM-модели позволяет анализировать алгоритмы машинно-независимым способом. Подведение итогов Алгоритмы можно изучать и анализировать, не прибегая к использованию конкретного языка программирования или компьютерной платформы. 2.1.1. Анализ сложности наилучшего, наихудшего и среднего случая С помощью RAM-модели можно подсчитать количество шагов, требуемых алгоритму для исполнения любого экземпляра задачи. Но чтобы получить общее представление о том. насколько хорошим или плохим является алгоритм, нам нужно знать, как он ра- ботает со всеми экземплярами задачи. Чтобы понять, что означает наилучший, наихудший и средний случай сложности алго- ритма (т. е. время его исполнения в соответствующем случае), нужно рассмотреть ис- полнение алгоритма на всех возможных экземплярах входных данных. В случае задачи сортировки множество входных экземпляров состоит из всех возможных компоновок ключей п по всем возможным значениям п. Каждый входной экземпляр можно пред- ставить в виде точки графика (рис. 2.1), где ось х представляет размер входа задачи (для сортировки это будет количество элементов, подлежащих сортировке), а ось у — количество шагов, требуемых алгоритму для обработки данного входного экземпляра. В действительности. Земля не совсем сферическая, но модель сферической Земли удобна для ра- боты с такими понятиями, как широта и долгота.
Гпава 2. Анализ алгоритмов 51 Количество шагов Размер задачи Рис. 2.1. Наилучший, наихудший и средний случай сложности алгоритма Эти точки естественным образом выстраиваются столбцами, т. к. размер входа может быть только целым числом (т. е., сортировка 10,57 элементов лишена смысла). На гра- фике этих точек можно определить три представляющих интерес функции: ♦ сложность алгоритма в наихудшем случае— это функция, определяемая макси- мальным количеством шагов, требуемых для обработки любого входного экземпля- ра размером л Этот случай отображается кривой, проходящей через самую высшую точку каждого столбца; ♦ сложность алгоритма в наилучшем случае— это функция, определяемая мини- мальным количеством шагов, требуемых для обработки любого входного экземпля- ра размером л. Этот случай отображается кривой, проходящей через самую низшую точку каждого столбца; ♦ сложность алгоритма в среднем случае — это функция, определяемая средним ко- личеством шагов, требуемых для обработки всех экземпляров размером л. На практике наиболее важной является оценка сложности алгоритма в наихудшем слу- чае Для многих людей это противоречит здравому смыслу. Рассмотрим пример. Вы собираетесь посетить казино, имея в кармане «долларов. Каковы возможные исходы? В наилучшем случае вы выиграете само казино, но хотя такое развитие событий и воз- можно, вероятность его настолько ничтожна, что даже не стоит думать об этом. В наи- худшем случае вы проиграете все свои л долларов; вероятность такого события удру- ающе высока и ее легко вычислить. Средний случай, когда типичный игрок проигры- вает в казино 87.32% взятых с собой денег, трудно рассчитать и даже определить. Что именно означает средний? Глупые люди проигрывают больше, чем умные, так вы ум- нее или глупее, чем средний человек и насколько? Игроки в очко, которые умеют от- слеживать сданные карты, в среднем выигрывают больше, чем игроки, которые не от- казываются от нескольких бесплатных рюмок алкогольного напитка, регулярно пред- лагаемых симпатичными официантками. Чтобы избежать всех этих усложнений и получить самый полезный результат, мы и рассматриваем только наихудший случай. Важно осознавать то, что в каждом случае сложность алгоритма определяется число- вой функцией, соотносящей время с размером задачи. Эти функции определены так же
52 Часть I. Практическая разработка алгоритмов строго, как и любые другие числовые функции, будь то уравнение у = х2 - 2.r + 1 или цена акций Google в зависимости от времени. Но функции временной сложности на- столько трудны для понимания, что перед началом работы их нужно упростить. Для этой цели используются асимптотические обозначения, в частности обозначение "О-большое". 2.2. Асимптотические обозначения Временную сложность наилучшего, наихудшего и среднего случая для любого алго- ритма можно представить как числовую функцию от размеров возможных экземпляров задачи. Но работать с этими функциями очень трудно, т. к. они обладают такими свой- ствами: ♦ являются слишком волнистыми. Время исполнения алгоритма, например, двоично- го поиска, обычно меньше для массивов, имеющих размер и= 2*- I, где к— целое число. Эта особенность не имеет большого значения, но служит предупреждением, что точная функция временной сложности любого алгоритма вполне может иметь неровный график с небольшими выпуклостями и впадинами, как показано на рис. 2.2; Рис. 2.2. Верхняя и нижняя границы, действительные для п > сглаживают волнистость сложных функций ♦ требуют слишком много информации для точного определения. Чтобы сосчитать точное количество инструкций RAM-машины, исполняемых в худшем случае, нуж- но, чтобы алгоритм был расписан в подробностях полной компьютерной програм- мы. Более того, точность ответа зависит от маловажных деталей кодировки (напри- мер, был ли употреблен оператор case вместо вложенных операторов if). Точный анализ наихудшего случая, например, такого: Ди) = 12754и2 + 4353и + 8341g2n + 13546 очевидно, был бы очень трудной задачей, решение которой не предоставляет нам никакой дополнительной информации, кроме той, что "с увеличением и временная сложность возрастает квадратически".
Глава 2 Анализ алгоритмов 53 Оказывается, намного легче работать с верхней и нижней границами функций времен- ной сложности, используя для этого асимптотические обозначения ((^-большое и Q- большое соответственно). Асимптотические обозначения позволяют упростить анализ, поскольку игнорируют детали, которые не влияют на сравнение эффективности алго- ритмов. Властности, в асимптотических обозначениях игнорируется разница между постоян- ными множителями. Например, в анализе с применением асимптотического обозначе- ния функции fin) = 2я и g(ri) = п являются идентичными. Эго вполне логично в нашей ситуации. Допустим, что определенный алгоритм на языке С исполняется вдвое быст- рее. чем тот же алгоритм на языке Java. Этот постоянный множитель, равняющийся двум, не предоставляет нам никакой информации собственно об алгоритме, т к. в обо- их случаях исполняется один и тот же алгоритм. При сравнении алгоритмов такие по- стоянные коэффициенты не принимаются во внимание. Формальные определения, связанные с асимптотическими обозначениями, выглядят жим образом: ♦ Ди) = O(g(«)) означает, что функция /(я) ограничена сверху функцией с • g(n). Ины- ми словами, существует такая константа с, для которой /(я) < с • g(n) при достаточно большом значении п (т. е. п > я0 для некоторой константы Яо); I f[n) = Q(g(n)) означает, что функция /(я) ограничена снизу функцией с • g(n). Иными словами, существует такая константа с, для которой /(я) > с • g(n) для всех п > /г«; ♦ Ди) = 0(g(n)) означает, что функция /(л) ограничена сверху функцией щ • g(n), a сни- зу функцией ci • g(n) для всех п > я0. Иными словами, существуют константы щ и сэ, для которых /(я) < ci g(n) и Дя)> c2-g(/?). Следовательно, функция g(n) дает нам хорошие ограничения для функции /(я). Графическая иллюстрация этих определений дается на рис. 2.3. Рис. 2.3. Графическая иллюстрация асимптотических обозначений: Оболыпое (а), D-большое (6). 0-большое (в) В каждом из этих определений фигурирует константа яо, после которой эти определе- ния всегда верны. Нас не интересуют небольшие значения я, т. е. значения слева от яп. В конце концов, нам безразлично, что один алгоритм может отсортировать, скажем. шесть или восемь элементов быстрее, чем другой. Мы ищем алгоритм для быстрой сортировки 10 000 или 1 000 000 элементов В этом отношении асимптотические обо-
54 Часть I. Практическая разработка алгоритмов значения позволяют нам игнорировать несущественные детали и концентрироваться на общей картине. Подведение итогов Анализ наихудшего случая и асимптотические обозначения являются инструментами, ко- торые сушественно упрощают задачу сравнения эффективности алгоритмов. Обязательно убедитесь, что вы понимаете смысл асимптотических обозначений, изу- чив последующие примеры. Конкретные значения константе и и0 были выбраны пото- му, что они хорошо иллюстрируют ситуацию, но можно использовать и другие значе- ния этих констант с точно таким же результатом. Вы можете выбрать любые другие значения констант, при которых сохраняется первоначальное неравенство; в идеале, следует выбирать такие значения, при которых очевидно, что неравенство соблюда- ется: Зп2 - 100« + 6 = О(п ). т. к. выбрано с = 3 и Зп2 > Зп2 - 1 ООи + 6; Зп- - 100м + 6 = О(п\ т. к. выбрано с = 1 и п’> Зп2 - 100м + 6 при и > 3; Зи" - 100м + 6 ф О(п). т. к. для любого значения с выбрано см < Зм2 при м > с; Зп2 - 100м + 6 = Q(M2), т. к. выбрано с = 2 и 2м2 < Зм2 - 100м + 6 при м > 100; Зм2 - 100м + 6 Ф £1(п\ т. к. выбрано с = 3 и Зм2 - 100м + 6 < м2 при м > 3: Зм2 - 100м + 6 = Q(m), т. к. для любого значения с выбрано см < Зп2 - 100м + 6 при м > 100с; Зм" - 100м + 6 = ®(м2). т. к. применимо как О, так и Q; Зм2 — 100м + 6 Ф 0(м3), т. к. применимо только О; Зп2 - 100/7 + 6 # 0(м). т. к. применимо только Q. Асимптотические обозначения позволяют получить приблизительное представление о равенстве функций при их сравнении. Выражение типа м2 = О(п*) может выглядеть странно, но его значение всегда можно уточнить, пересмотрев его определение в тер- минах верхней и нижней границ. Возможно, это обозначение будет более понятным, если в данном случае рассматривать символ равенства (= ) как означающий "одна из функций, принадлежащих к множеству функций". Очевидно, что м2 является одной из функций, принадлежащих множеству функций О(м3). Остановка для размышлений. Возвращение к определению ЗАДАЧА. Верно ли равенство 2"т 1 = 0(2")? Решение. Для разработки оригинальных алгоритмов требуется умение и вдохновение. Но при использовании асимптотических обозначений лучше всего подавить все свои творческие инстинкты. Все задачи по асимптотическим обозначениям можно правиль- но решить, работая с первоначальным определением. ♦ Верно ли равенство 2"+ 1 = 0(2")? Очевидно, что/(и) = O(g(«)) тогда и только тогда, когда существует такая константа с, при которой для всех достаточно больших зна-
Гпава 2. Анализ алгоритмов 55 чений п функция fin) < с g(n). Существует ли такая констан га? Заметим, что 2" 1 = = 2 • 2". откуда следует, что 2 • 2" < с • 2"для всех с > 2. ♦ Верно ли равенство 2” 1 = £2(2”)? Согласно определению, fin) = £l(g(n)) тогда и только тогда, когда существует такая константа с > 0. при которой для всех доста- точно больших значений п функция fin) > с g(n). Это условие удовлетворяется для любой константы 0 < с < 2. Границы О-большое и Q-большое совместно подразуме- вают 2"*1 =0(2"). Остановка для размышлений. Квадраты ЗАДАЧА. Верно ли равенство (х + у)2 = (fix2 + у~)? Решение. При малейших трудностях в работе с асимптотическим обозначением не- медленно возвращаемся к его определению, согласно которому это выражение дейст- вительно тогда и только тогда, когда мы можем найти такое значение с. при котором (х+у)2<с(х2+у2). Лично я первым делом раскрыл бы скобки в левой части уравнения, т. е. (х + у)" = х" + + 2ху+у2. Если бы в развернутом выражении отсутствовал член 2ху, то вполне очевид- но, что неравенство соблюдалось бы для любого значения с > 1. Но т. к. этот член име- ется, то нам нужно рассмотреть его связь с выражением х* + у . Если х < у. то 2ху< <1у < 2(х~ + у). А если х > у, то 2ху < 2х2 < 2(х2 + у2). В любом случае теперь мы мо- жем ограничить это выражение двойным значением функции с правой стороны Это означает, что (х + у)' < 3(х2 + у2), поэтому результат остается в силе. 2.3. Скорость роста и отношения доминирования Используя асимптотические обозначения, мы пренебрегаем постоянными множителя- ми, не учитывая их при вычислении функций. При таком подходе функции fin) = =0.001и2 и g(n) = 1000г?2 для нас одинаковы, несмотря на то, что значение функции gfi) в миллион раз больше значения функции fin) для любого п. Причина, по которой достаточно грубого анализа, предоставляемого обозначением 0-большое. приводится в табл. 2.1, в которой перечислены наиболее распространенные функции и их значения для нескольких значений п. В частности, здесь можно увидеть время исполнения fin) операций алгоритмов на быстродействующем компьютере, ис- полняющем каждую операцию за одну наносекунду (1О’У секунд). На основе представ- ленной в таблице информации можно сделать следующие выводы: ♦ время исполнения всех этих алгоритмов примерно одинаково для значений п - 10; ♦ любой алгоритм с временем исполнения г?! становится бесполезным для значений «>20; ♦ диапазон алгоритмов с временем исполнения 2" несколько шире, но они также ста- новятся непрактичными для значений п> 40; ♦ алгоритмы с квадратичным временем исполнения п2 применяются при п < 10 000, после чего их производительность начинает резко ухудшаться. Эти алгоритмы, ско- рее всего, будут бесполезны для значений n > 1 000 000;
56 Часть I. Практическая разработка алгоритмов ♦ алгоритмы с линейным и логарифмическим временем исполнения остаются полез- ными при обработке миллиарда элементов; ♦ в частности, алгоритм ()(1g/?) без труда обрабатывает любое вообразимое количест- во элементов. Таблица 2.1. Скорость роста основных функций IgH п »lg« п~ 2" и! 10 0.003 мкс 0.01 мкс 0,033 мкс 0.1 мкс 1 мкс 3.63 мс 20 0.004 мкс 0,02 мкс 0.086 мкс 0.4 мкс 1 мс 77.1 лет 30 0.005 мкс 0.03 мкс 0.147 мкс 0.9 мкс 1 с 8.4- H0,s лет 40 0.005 мкс 0.04 мкс 0.213 мкс 1.6 мкс 18.3 мин 50 0.006 мкс 0.05 мкс 0.282 мкс 2,5 мкс 13 дней 100 0.007 мкс 0.1 мкс 0.644 мкс 10 мкс 4x10й лет 1 000 0.010 мкс 1.00 мкс 9,966 мкс 1 мс 10 000 0.013 мкс 10 мкс 130 мкс 100 мс 100 000 0.017 мкс 0.10 мс 1.67 мс Юс 1 000 000 0.020 мкс 1 мс 19,93 мс 16,7 мин 10 000 000 0.023 мкс 0,01 с 0.23 с 1,16 дней 100 000 000 0.027 мкс 0.10 с 2.66 с 115.7 дней 1 000 000 000 0.030 мкс 1 с 29,90 с 31.7 лет Из этого можно сделать основной вывод, что, даже игнорируя постоянные множители, мы получаем превосходное общее представление о годности алгоритма для решения задачи определенного размера. Алгоритм с временем исполнениями) = п секунд побе- дит алгоритм с временем исполнения g(») = 1000 000-»' секунд только при » < 1 000 000. На практике такая громадная разница между постоянными множителями алгоритмов встречается намного реже, чем задачи по обработке большего объема входных данных. 2.3.1. Отношения доминирования Посредством асимптотических обозначений функции разбиваются на классы, в каждом из которых сгруппированы функции с эквивалентным асимптотическим обозначением. Например, функции /(») = 0.34» и g(») = 234,234» принадлежат к одному и тому же классу, а именно к классу порядка 0(и). Кроме того, когда функции/и g принадлежат к разным классам, они являются разными относительно нашего обозначения. Иными словами, справедливо либо Ди)= O(g(»)). либо g(») = О(Д»)). но не оба равенства од- новременно. Говорят, что функция с более быстрым темпом роста доминирует над менее быстро растущей функцией точно так же, как более быстро растущая страна в итоге начинает доминировать над отстающей. Когда функции f и g принадлежат к разным классак
Гпава 2. Анализ алгоритмов 57 (т. е. Дя) Ф0(g(w))). говорят, что функция f доминирует над функцией g, когда fi,n) ()(у(л}\ Это отношение иногда обозначается как g К счастью, процесс базового анализа алгоритмов обычно порождает лишь небольшое количество классов функций, достаточное для покрытия почти всех алгоритмов, рас- сматриваемых в этой книге. Далее приводятся эти классы в порядке возрастания доми- нирования. ♦ Функции-константы, фл) = 1. Такие функции могут измерять трудоемкость сложе- ния двух номеров, распечатывания какого-либо текста или рост таких функции, как Ди) = min(/?,100). По большому счету, зависимость от параметра л отсутствует. ♦ Логарифмические функции, фп)~ log/?. Логарифмическая временная сложность про- является в таких алгоритмах, как двоичный поиск. С увеличением л такие функции возрастают довольно медленно, но быстрее, чем функции-константы (которые во- обще не возрастают). Логарифмы рассматриваются более подробно в разделе 2.6. ♦ Линейные функции, фл) = л. Такие функции измеряют трудоемкость просмотра каждого элемента в массиве элементов один раз (или два раза, или десять раз), на- пример, для определения наибольшего или наименьшего элемента или для вычис- ления среднего значения. ♦ Сулерлинейные функции, фл) = «lg/г. Этот важный класс функций возникает в таких алгоритмах, как Quicksort и Mergesort. Эти функции возрастают лишь немного бы- стрее, чем линейные (см. табл. 2.1), но достаточно быстро, чтобы составить другой класс доминирования. ♦ Квадратичные функции, фл) = п. Эти функции измеряют трудоемкость просмотра большинства или всех пар элементов в универсальном множестве из п элементов. Они возникают в таких алгоритмах, как сортировка вставками или сортировка ме- тодом выбора. ♦ Кубические функции, фл) = п. Эти функции возникают при перечислении всех три- ад элементов в универсальном множестве из я элементов. Они также возникают в определенных алгоритмах динамического программирования, которые рассматри- ваются в главе 8. ♦ Показательные функции, фл) = с", константа с> 1. Эти функции возникают при перечислении всех подмножеств множества из л элементов. Как мы видели в табл. 2.1. экспоненциальные алгоритмы быстро становятся бесполезными с увели- чением количества элементов л. Впрочем, не так быстро, как функции из следую- щего класса. ♦ Факториальные функции, фл) = я!. Факториальные функции определяют все пере- становки я элементов. Тонкости отношений доминирования рассматриваются в разделе 2.9.2, но в действи- тельности вы должны помнить лишь следующее отношение: я! » 2" » л л2» nlogn :» я » log/? >> 1 Подведение итогов Хотя анализ алгоритмов высокого уровня может порождать экзотические функции. в большинстве практических алгоритмов применяется лишь небольшой ассортимент функций временной сложности, которого является вполне достаточно
58 Часть I. Практическая разработка алгоритмов 2.4. Работа с асимптотическими обозначениями Для работы с асимптотическими обозначениями нужно повторить тему упрощения ал- гебраических выражений, которая изучается в школе. Большинство полученных в школе знаний также применимо и для асимптотических обозначений. 2.4.1. Сложение функций Сумма двух функций определяется доминантной функцией, а именно: CV(»)) + <>(g(«)) = O(max(/(n),(g(H))) П(/(и)) + Q(g(»)) = Q(maxQ/(/7),(g(w))) ©(/(«)) + O(g(«)) = 0(max(X«),(g(n))) Это обстоятельство очень полезно при упрощении выражений, т. к. оно подразумевает, что п' + if + п + I = О(п). По сравнению с доминантным членом все остальное являет- ся несущественным. На интуитивном уровне это объясняется таким образом. По крайней мере, половина общего объема суммы fin) + g(n) должна предоставляться большей функцией. По опре- делению, при и —» оо доминантная функция будет предоставлять большую часть объема суммы функций. Соответственно, отбросив меньшую функцию, мы уменьшим значе- ние суммы максимум в два раза, что равносильно постоянному множителю 1/2. Пред- положение, что Ди) = О(и~) и g(n) = О(п~) также влечет за собой, что Д/г) + g(n) - О(п~). 2.4.2. Умножение функций Умножение можно рассматривать, как повторяющееся сложение. Рассмотрим умноже- ние на любую константу с > 0, будь это 1,02 или 1 000 000. Умножение функции на константу не может повлиять на ее асимптотическое поведение, т. к. в анализе функ- ции с'Д/г) с применением обозначения "(7-большое" мы можем умножить ограничи- вающие константы на 1/с, что даст нам необходимые константы для анализа. Таким образом: О(сД/г)) -» О(Д/7)) Q(c/(«)) -> Г2(Д/г)) 0(сДи)) -► 0(Ди)) Конечно же, во избежание неприятностей, константа с должна быть строго положи- тельной, т. к. мы можем свести на нет даже самую быстрорастущую функцию, умно- жив ее на ноль. С другой стороны, когда обе функции произведения возрастают, то обе являются важ- ными. Функция O(w!log,z) доминирует над функцией и! точно так же, как и функция log// доминирует над константой 1. В общем, 6>(A«))*O(g(/0) -> C*(/(>/)*g(//)) Q(/(/7))*Q(g('O) -> Q(/(/7)*g(n)) 0(/(/7))*©(g(«)) -> 0(/(/7)*g(/?))
Глава 2. Анализ алгоритмов 59 Остановка для размышлений. Транзитивность ЗАДАЧА. Доказать, что отношение О-большое обладает транзитивностью, т. е. если Дп) = (fig(n)) и g(n) = О(/?(?7)). тогда fin) = (fih(n)). Решение. Работая с асимптотическими обозначениями, мы всегда обращаемся к опре- делению. Нам нужно показать, что fin) < c-fi(n) при п > п2 при условии, что fin) < C\g(ri) и g{n)< c2h(n) при п> п\ и п> л2 соответственно. Из этих неравенств получаем: Дл)< C]g(n) <cic2h(n) при п > «з = max(«|,n2)- 2.5. Оценка эффективности Как правило, общую оценку времени исполнения алгоритма можно легко получить при наличии точного письменного изложения алгоритма. В этом разделе рассматривается несколько примеров, возможно даже более подробно, чем необходимо. 2.5.1. Сортировка методом выбора В этом разделе анализируется алгоритм сортировки методом выбора. При сортировке этим способом определяется наименьший неотсортированный элемент и помещается в конец отсортированной части массива. Процедура повторяется до тех пор, пока все элементы массива не будут отсортированы. Графическая иллюстрация работы алго- ритма представлена на рис. 2.4, а соответствующий код на языке С в листинге 2.1. Рис. 2.4. Графическая иллюстрация работы алгоритма сортировки методом выбора Листинг 2.1. Реализация алгоритма сортировки методом выбора на языке С relection_sort (int s[], int n) int i,j; /* Счетчики */ int min; /* Указатель наименьшего элемента • for (1=0; i<n; i++) { min=i; for (j=i+l; j<n; j++)
60 Часть I. Практическая разработка алгоритмов ;f s[j] s[min]) min=j; ?wap(bal- ,&s[min]); I Внешний цикл исполняется n раз. Внутренний цикл исполняется п - i- 1 раз, где i — счетчик внешнего цикла. Точное количество исполнений оператора if определяется следующей формулой: W-1 п-1 п-1 Х|=Еи-/_| ,=0 /=/+1 ,=0 Это формула сложения целых чисел в убывающем порядке, начиная с п - 1. т. е.: S(n) = (п- 1) + (п - 2) + (и — 3) + ... + 2 + 1 Какие выводы мы можем сделать на основе такой формулы? Чтобы получить точное значение, необходимо применить методы, рассматриваемые в разделе 1 3.5. Но при работе с асимптотическими обозначениями нас интересует только степень выражения. Один из подходов— считать, что мы складываем п- 1 элементов, среднее значение которых равно приблизительно п/2. Таким образом мы получаем S(n) ~ п(п - 1)/2. Другим подходом будет использование верхней и нижней границ. Мы имеем не более п элементов, значение каждого из которых не превышает п- 1. Таким образом, S(n) < < п(п - 1) = О(п~). Также мы имеем п/2 элементов, чье значение больше чем п/2. Соот- ветственно. S(n)> (и/2) х (и/2) = Q(n2). Все это говорит нам, что время исполнения рав- но ©(/?"), т. е. сложность сортировки методом выбора является квадратичной. 2.5.2. Сортировка вставками Основное практическое правило при асимптотическом анализе гласит, что время ис- полнения алгоритма в наихудшем случае получается умножением наибольшего воз- можного количества итераций каждого вложенного цикла. Рассмотрим, например, ал- горитм сортировки вставками из листинга 1.1, внутренние циклы которого приведены в листинге 2.2. Листинг 2.2. Внутренние циклы алгоритма сортировки вставками на языке С for (1=1; i<n; i++) | .=1 while ( (j>0) && (s[j] < s[j-l])) ( svap(&s[j] ,&s[j-l]); 1 = j-iz 1 Сколько итераций осуществляет внутренний цикл while? На этот вопрос сложно дать однозначный ответ, т. к. цикл может быть остановлен досрочно, если произойдет вы- ход за границы массива (j >о) или элемент окажется на должном месте в отсортирован- ной части массива (s[j]<s[j-l]). Так как в анализе наихудшего случая мы ищем верх- нюю границу времени исполнения, то мы игнорируем досрочное завершение и полага-
Гпава 2. Анализ алгоритмов 61 ем. что количество исполняемых в этом цикле итераций всегда будет /. Более того, мы можем допустить, что всегда исполнятся п итераций, т. к. i < п. А т. к. внешний цикл исполняется н раз. то алгоритм сортировки вставками должен быть квадратичным, т. е. Такой грубый анализ методом округления всегда оказывается результативным, в том смысле, что полученная верхняя граница времени исполнения ((Лбольшое) всегда бу- дет правильной. Иногда этот результат может быть даже завышен в худшую сторону, т. е. в действительности время исполнения худшего случая окажется меньшим, чем результат, полученный при анализе. Тем не менее, я настоятельно рекомендую этот подход в качестве основы для простого анализа алгоритмов. 2.5.3. Сравнение строк Сравнение комбинаций символов — основная операция при работе с текстовыми стро- ками. Далее приводится алгоритм для реализации функции поиска определенного тек- ста, которая является обязательной частью любого веб-браузера или текстового редак- тора. ЗАДАЧА. Найти подстроку в строке. Вход. Текстовая строка / и строка для поиска р (рис. 2.5). Выход. Содержит ли строка t подстроку р, и, если содержит, в каком месте? а b abb а abba a a b a b b а Рис. 2.5. Пример: поиск подстроки abba в тексте cuibabba Пример практического применения этого алгоритма— поиск упоминания определен- ной фамилии в новостях. Для данного экземпляра задачи текстом t будет статья, а строкой р для поиска — указанная фамилия. Эта задача решается с помощью довольно простого алгоритма (листинг 2.3), который допускает, что строка р может начинаться в любой возможной позиции в тексте /, и выполняет проверку, действительно ли, начиная с этой позиции, текст содержит иско- мую строку. ,. ........ ...»....ч. .... ............. ..ж............. ....г»..-. .... - ... ....................... .............. Листинг 2.3. Реализация алгоритма поиска строки в тексте findmatchfchar *р, char *t) int i,j; int m, n; m = strlen(p); strlen(t); /* Счетчики ♦/ /• Длины строк */
62 Часть I. Практическая разработка алгоритмов for i=0; . =(n-m); i=i+l) , >0; while ((i<m) && (t[i+j]==p[j]) ) 7 j+1: f m) return(i); I return(-1); ) Каким будет время исполнения этих двух вложенных циклов в наихудшем случае? Внутренний цикл while исполняется максимум т раз, а возможно, и намного меньше, если поиск заканчивается неудачей. Кроме оператора while, внешний цикл содержит еще два оператора. Внешний цикл исполняется самое большее п — т раз, т. к. после продвижения направо по тексту оставшийся фрагмент будет короче искомой строки. Общая временная сложность — произведение значений оценки временной сложности внешнего и вложенного циклов, что дает нам время исполнения в худшем случае О((п — т)(т + 2)). При этом мы не учитываем время, потраченное на определение длины строк с по- мощью функции strien. Так как мы не знаем, каким образом реализована эта функция, мы можем лишь строить предположения относительно времени ее работы. Если мы явно считаем количество символов, пока не достигнем конца строки, то отношение между временем исполнения этой операции и длиной строки будет линейным. Значит, время работы будет равно ()(п + т + (и — т)(т + 2)). С помощью асимптотических обозначений это выражение можно упростить. Так как т + 2 = ©(/»), выражение " + 2" не представляет интереса, поэтому останется только О(п + т + (п ~ т)т). Выполнив умножение, получаем выражение О(п + т + пт - т~). которое выглядит довольно непривлекательно. Но мы знаем, что в любой представляющей интерес задаче п> т. т. к. невозможно, чтобы искомая строка р была длиннее, чем текст /, в котором выполняется ее поиск. Одним из следствий этого обстоятельства является отношение (п + т) < 2п = 0(и). Та- ким образом, формула времени исполнения для наихудшего случая упрощается дальше до О(п + пт - т~). Еще два замечания. Обратите внимание, что п < пт, т. к. для любой представляющей интерес строки поиска т> 1. Таким образом, п+ пт= &(пт). и мы можем опустить дополняющее п. упростив формулу анализа до (Хпт — /и2). Кроме того, заметьте, что член - т~ отрицательный, вследствие чего он только умень- шает значение выражения внутри скобок. Так как "О-большое" задает верхнюю грани- цу. то любой отрицательный член можно удалить, не искажая оценку верхней границы. Тот факт, что п > т подразумевает, что тп > т~. поэтому отрицательный член недоста- точно большой, чтобы аннулировать любой другой оставшийся член. Таким образом, время исполнения этого алгоритма в худшем случае можно выразить просто как О(пт). Накопив достаточно опыта, вы сможете выполнять такой анализ алгоритмов в уме, да- же не прибегая к изложению алгоритма в письменной форме. В конце концов, одной из составляющих разработки алгоритма для решения предоставленной задачи является перебор в уме разных способов и выбор самого лучшего из них. Такое умение приоб-
Гпава 2. Анализ алгоритмов 63 ретается с опытом, но если у вас недостаточно практики и вы не понимаете, почему время исполнения данного алгоритма равно то сначала распишите подробно формулу, а потом выполните последовательность логических рассуждений, как было продемонстрировано в этом разделе. 2.5.4. Умножение матриц При анализе алгоритмов с вложенными циклами часто приходится иметь дело с вло- женными операциями суммирования. Рассмотрим задачу умножения матриц: ЗАДАЧА. Умножить матрицы. Вход. Матрица А (размером х ху)ц матрица В (размером у * z). Выход. Матрица С размером х х z, где С[/][/] является скалярным произведением стро- ки i матрицы А и столбца j матрицы В. Умножение матриц является одной из основных операций в линейной алгебре; пример задачи на умножение матриц рассматривается в разделе 13.3. А в листинге 2.4 приво- дится пример реализации простого алгоритма для умножения матриц с использованием вложенных циклов. Листинг 2.4. Умножение матриц for (1=1; 1<=х; 1++) for (j=l; j<=y; j++) | C[i] [j] - 0; for (k=l; k<=z; k++) C[i] [j] += A[i] [k] * B[k] [j]; Анализ временной сложности этого алгоритма выполняется следующим образом. Ко- личество операций умножения М(х, у, z) определяется такой формулой: M(x,y,z) = Y £ £1 ,=1 у = 1 *=| Суммирование выполняется справа налево. Сумма z единиц равна z. поэтому можно написать: <=1 у=1 Сумма у членов z вычисляется так же просто. Она равна yz, тогда M(x,y,z) = У- <=1 Наконец, сумма х членов yz равна xyz. Таким образом, время исполнения этого алгоритма для умножения матриц равно Ofxyz). В общем случае, когда все три измерения матриц одинаковы, это сводится к 0(и),т. е. алгоритм имеет кубическую формулу времени исполнения
64 Часть I. Практическая разработка алгоритмов 2.6. Логарифмы и их применение Слово "логарифм"— почти анаграмма слова "алгоритм". Но наш интерес к логариф- мам вызван не этим обстоятельством. Возможно, что кнопка калькулятора с обозначе- нием "log"— единственное место, где вы сталкиваетесь с логарифмами в повседнев- ной жизни. Также возможно, что вы уже не помните назначение этой кнопки. Лога- рифм — это функция, образ ная показательной. То есть, выражение Ьх = у эквивалентно выражению г = log/,y. Более того, из определения логарифма следует, что b'°6h> = у . Показательные функции возрастают чрезвычайно быстро, как может засвидетельство- вать любой, кто когда-либо выплачивал долг по кредиту. Соответственно, функции, обратные показательным, т. е. логарифмы, возрастают довольно медленно. Логариф- мические функции возникают в любом процессе, содержащем деление пополам. Да- вайте рассмотрим несколько таких примеров. 2.6.1. Логарифмы и двоичный поиск Двоичный поиск является хорошим примером алгоритма с временной логарифмиче- ской сложностью (9(logn). Чтобы найти определенного человека по имени р в телефон- ной книге, содержащей п имен, мы сравниваем имя р с выбранным именем посередине книги (т. е. с и/2-м именем), скажем. Monroe, Marilyn. Независимо от того, находится ли имя р перед выбранным именем (например. Dean. James) или после него (например, Preslej, Elvis), после этого сравнения мы можем отбросить половину всех имен в книге. Этот процесс повторяется с половиной книги, содержащей искомое имя и т. д., пока не останется всего лишь одно имя. которое и будет искомым. По определению количество таких делений равно logy;. Таким образом, чтобы найти любое имя в телефонной книге Манхэттена (содержащей миллион имен), достаточно выполнить всего лишь двадцать сравнений. Потрясающе, не так ли? Идея двоичного поиска является одной из наиболее плодотворных в области разработ- ки алгоритмов. Эта мощь становится очевидной, если мы представим, что в окружаю- щем нас мире имеются только неотсортированные телефонные книги. Как видно из табл. 1.1, алгоритмы с временной сложностью O(logn) можно применять для решения задач с практически неограниченным размером входных данных. 2.6.2. Логарифмы и деревья Двоичное дерево высотой в один уровень может иметь две концевые вершины (листа), а дерево высотой в два уровня может иметь до четырех листов. Какова высота h дво- ичного дерева, имеющего п листов? Обратите внимание, что количество листов удваи- вается при каждом увеличении высоты дерева на один уровень. Таким образом, зави- симость количества листов п от высоты дерева h выражается формулой п = 2Л, откуда следует, что h = login. Теперь перейдем к общему случаю. Рассмотрим деревья, которые имеют d потомков (для двоичных деревьев d- 2). Такое дерево высотой в один уровень может иметь d количество листов, а дерево высотой в два уровня может иметь d~ количество листов. Количество листов на каждом новом уровне можно получить, умножая на d количество
Гпава 2. Анализ алгоритмов 65 [листов предыдущего уровня. Таким образом, количество листов п выражается форму- лой и = </. т. е. высота находится по формуле h = logjrz (рис. 2.6). I xii Рис. 2.6. Дерево высотой h и количеством потомков d для каждого узла имеет ct‘ листьев. В данном случае /г 2, d = 3 Из вышеизложенного можно сделать вывод, что деревья небольшой высоты могут иметь очень много листьев. Это обстоятельство является причиной того, что двоичные деревья лежат в основе всех быстро обрабатываемых структур данных. 2.6.3. Логарифмы и биты Положим, имеются две однобитовые комбинации (0 и I) и четыре двухбитовые комби- нации (00. 01, 10 и 11). Сколько битов и' потребуется, чтобы представить любую из п возможных разных комбинаций, будь то один из п элементов или одно из целых чисел от I до 77? Ключевым наблюдением здесь является то обстоятельство, что нужно иметь, по край- ней мере, п разных битовых комбинаций длиной ж Так как количество разных бито- вых комбинаций удваивается с добавлением каждого бита, то нам нужно, по крайней мере, w битов, где 2й’ = п, т. е. нам нужно w = log2n битов. 2.6.4. Логарифмы и умножение I Логарифмы имели особенно большую важность до распространения карманных каль- куляторов. Применение логарифмов было самым легким способом умножения боль- ших чисел вручную, либо с помощью логарифмической линейки, либо с использовани- ем таблиц. Но и сегодня логарифмы остаются полезными для выполнения операций умножения, особенно для возведения в степень. Вспомните, что ioga(xy) = logXx.) + log^Q’), т. е„ что логарифм произведения равен сумме логарифмов сомножителей. Прямым следствием этого является формула: logn и'’ = £-log„« Выясним, как вычислить а для любых а и Ь, используя функции ехр(х) и 1п(л) на кар- манном калькуляторе, где ехр(л) = е' и 1п(х) = log(.(x). Мы знаем, что: ah = ехр(1п(с/)) = ехр(Мпа) Таким образом, задача сводится к одной операции умножения с однократным вызовом каждой из этих функций. Вак 3711
66 Часть I. Практическая разработка алгоритмов 2.6.5. Быстрое возведение в степень Допустим, что нам нужно вычислить точное значение ап для достаточно большого значения п. Такие задачи, в основном, возникают в криптографии при проверке числа на простоту (см. раздел 13.8). Проблемы с точностью не позволяют нам воспользовать- ся ранее рассмотренной формулой возведения в степень. Самый простой алгоритм выполняет л- 1 операций умножения (а х а х ... х а). Но можно указать лучший способ решения этой задачи, приняв во внимание, что w = |_«/2j + |"«/2”|. Если п четное, тогда а" = (а"12)2. А если п нечетное, то тогда а" = а(с^п12^)2. В любом случае значение показателя степени было уменьшено наполо- вину, а вычисление сведено к. самое большее, двум операциям умножения. Таким об- разом, для вычисления конечного значения будет достаточно O(lg/?) операций умноже- ния. Псевдокод соответствующего алгоритма показан в листинге 2.5. Листинг 2.5. Алгоритм быстрого возведения и степень .................................................................. и function power(а, п) if (п = 0) return(1) х = power (а, |_л/2J) if (п is even) then return (х2) else return (a ~ x2) Этот простой алгоритм иллюстрирует важный принцип "разделяй и властвуй". Разде- ление задачи на (по возможности) равные подзадачи, всегда окупается. Этот принцип применим и в реальном мире. Когда значение п отлично от 2, то входные данные не всегда можно разделить точно пополам, но разница в один элемент между двумя поло- винами не вызовет никакого серьезного нарушения баланса. 2.6.6. Логарифмы и сложение Гармоническое число представляет собой особый случай арифметической прогрессии, а именно Н(п) = S(n, — I). Это сумма обратных величин первых п последовательных чи- сел натурального ряда: Н{п) = 1 / / ~ In и /=| Гармонические числа помогают объяснить, откуда берутся логарифмы в алгебраиче- ских операциях. Например, ключевым фактором в анализе среднего случая сложности алгоритма быстрой сортировки Quicksort является следующее суммирование: ^(«) = «X"=i1// Применение тождества гармонического числа сразу же упрощает это выражение до ®(wlog«).
Глава 2 Анализ алгоритмов 67 2.6.7. Логарифмы и система уголовного судопроизводства В табл. 2.2 приводится пример использования логарифмов. Таблица 2.2. Рекомендуемые наказания в федеральных судах США за преступления финансового мошенничества Понесенные убытки Повышение уровня наказания (А) 2 000 долларов или меньше Уровень не повышается (В) Свыше 2 000 долларов Повысить на один уровень (С) Свыше 5 000 долларов Повысить на два уровня (D) Свыше 10 000 долларов Повысить на три уровня (Е) Свыше 20 000 долларов Повысить на четыре уровня (F) Свыше 40 000 долларов Повысить на пя гь уровней (G) Свыше 70 000 долларов Повысить на шесть уровней (Н) Свыше 120 000 долларов Повысить на семь уровней (I) Свыше 200 000 долларов 11овысить на восемь уровней (J) Свыше 350 000 долларов Повысить на девять уровней (К) Свыше 500 000 долларов Повысить на десять уровней (L) Свыше 800 000 долларов Повысить на одиннадцать уровней (М) Свыше 1 500 000 долларов Повысить на двенадцать уровней (N) Свыше 2 500 000 долларов Повысить на тринадцать уровней (0) Свыше 5 000 000 долларов Повысить на четырнадцать уровней | (Р) Свыше 10 000 000 долларов Повысить на пятнадцать уровней (Q) Свыше 20 000 000 долларов Повысить на шестнадцать уровней (R) Свыше 40 000 000 долларов Повысить на семнадцать уровней (S) Свыше 5 000 000 долларов Повысить на восемнадцать уровней Эта таблица из Федерального руководства по вынесению наказаний используется во всех федеральных судах Соединенных Штатов. Изложенные в ней рекомендуемые уровни наказания являются попыткой стандартизировать выносимые приговоры, что- бы за преступления одинаковой категории разные суды приговаривали к одинаковому наказанию. Для этого специальная комиссия выработала сложную функцию для оцен- ки тяжести преступления и соотнесения его со сроком заключения. Данная функция представлена посредством табл. 2.2, в которой перечислены соотношения между поне- сенным убытком в долларах и соответствующим повышением базового наказания. Об- ратите внимание, что наказание повышается на один уровень при приблизительном удвоении суммы незаконно присвоенных денег. Это означает, что уровень наказания (который практически линейно связан со сроком заключения) возрастает логарифми- чески по отношению к сумме незаконно присвоенных денег.
68 Часть I. Практическая разработка алгоритмов Задумайтесь на минуту о последствиях этого обстоятельства. Несомненно, многие не- чистые на руку руководители фирм уже задумывались на эту тему. Все вышеизложен- ное означает, что общий срок заключения возрастает чрезвычайно медленно по от- ношению к росту суммы украденных денег. Срок заключения за пять ограблений винно-водочных магазинов на общую сумму 50 000 долларов будет значительно выше, чем за однократное завладение посредством мошеннических операций суммой в 1 000 000 долларов. Соответственно, выгода от обогащения подобным образом на дей- ствительно крупные суммы будет еще больше. Мораль логарифмического роста функ- ций ясна: уж если воровать, так миллионы. Подведение итогов Логарифмические функции возникают при решении задач с повторяющимся делением или удваиванием входных данных. 2.7. Свойства логарифмов Как мы уже видели, выражение Ьх = у эквивалентно выражению х = log/j’. Член b назы- вается основанием логарифма. Особый интерес представляют следующие основания логарифмов: ♦ основание b = 2. Двоичный логарифм, обычно обозначаемый как Igx, является лога- рифмом по основанию 2. Мы уже видели, что логарифмами с этим основанием вы- ражается временная сложность алгоритмов, использующих многократное деление пополам (т. е. двоичный поиск) или умножение на два (т. е. листья деревьев). В большинстве случаев, когда речь идет о применении логарифмов в алгоритмах, подразумеваются двоичные логарифмы; ♦ основание b = е. Натуральный логарифм, обычно обозначаемый как 1пх, является логарифмом по основанию е = 2.71828... Обратной к функции натурального лога- рифма является экспоненциальная функция ехр(х) = ех. Суперпозиция этих функций дает нам формулу exp(lnx) = х; ♦ основание b = 10. Менее распространенными на сегодняшний день являются лога- рифмы по основанию 10, или десятичные логарифмы. До появления карманных калькуляторов логарифмы с этим основанием применялись на логарифмических линейках и таблицах алгоритмов. Мы уже видели одно важное свойство логарифмов, а именно, что log,,(xy) = logn(x) + + log.,0). Другим важным фактом, который нужно запомнить, является то, что логарифм по од- ному основанию легко преобразовать в логарифм по другому основанию. Для этого применяется следующая формула: । к |оё<ь log,. а Таким образом, чтобы изменить основание а логарифма log,Л на с, логарифм просто нужно разделить на log,a. В частности, функцию натурального логарифма можно с легкостью преобразовать в функцию десятичного логарифма и наоборот.
Глава 2 Анализ алгоритмов 69 Из этих свойств логарифмов следуют два важных с арифметической точки зрения следствия. ♦ Основание логарифма не оказывает значительного влияния на скорость роста функции. Сравните следующие три значения: log2(l ООО 000) - 19.9316. log?(l ООО 000) = 12.5754 и logioo(l ООО 000) = 3. Как видите, большое изменение в основании логарифма сопровождается малыми изменениями в значении логарифма. Чтобы изменить у логарифма основание а на с. первоначальный логарифм нужно разделить на log(o. Этот коэффициент теряется в нотации "//-большое", когда а и с являются константами. Таким образом, игнорирование основания логарифма при анализе алгоритма обычно оправдано. ♦ Логарифмы уменьшают значение любой функции. Скорость роста логарифма любой полиномиальной функции определяется как O(lgn). Это вытекает из равенства: loga п = frlogo« Эффективность двоичного поиска в широком диапазоне задач является прямым следствием этого свойства. Обратите внимание, что двоичный поиск в отсортиро- ванном массиве из гб элементов требует всего лишь вдвое больше сравнений, чем в массиве из п элементов. Логарифмы уменьшают значение любой функции. С факториалами трудно выпол- нять какие-либо вычисления, если не пользоваться логарифмами, и тогда формула __ п «! = ]”[”_/ -> log»! = ^logz = &(пlogr?) становится еще одной причиной появления логарифмов в анализе алгоритмов. Остановка для размышлений. Важно ли деление точно пополам ЗАДАЧА. Насколько больше операций сравнения потребуется при двоичном поиске, чтобы найти совпадение в списке из миллиона элементов, если вместо деления точно пополам делить список в отношении 1/3 к 2/3? Решение. Не намного больше, чем при делении списка точно пополам, а именно Iogv2(l ООО 000) ~ 35 операций сравнения в самом худшем случае, что не намного больше, чем log2(l ООО 000) ~ 20 операций сравнения при делении пополам. Эффектив- ность алгоритма двоичного поиска определяется логарифмической временной слож- ностью, а не основанием логарифма. 2.8. История из жизни. Загадка пирамид По выражению его глаз я должен был догадаться, что услышу что-то необычное. - Мы хотим выполнять вычисления до 1 000 000 000 на многопроцессорном супер- компьютере, но для этого нам нужен более быстрый алгоритм. Мне приходилось видеть такой взгляд раньше. Глаза были пусты от того, что этот че- ловек уже давно ни о чем не думал, полагая, что мощные суперкомпьютеры избавляют его от необходимости разрабатывать сложные алгоритмы. Во всяком случае, это было так, пока задача не стала достаточно сложной.
70 Часть I. Практическая разработка алгоритмов — Я работаю с лауреатом Нобелевской премии над решением знаменитой задачи тео- рии чисел с помощью компьютера. Вы знакомы с проблемой Уоринга? Я имел некоторое представление о теории чисел. — Конечно. В проблеме Уоринга ставится вопрос, можно ли выразить каждое целое число, по крайней мере, одним способом, как сумму квадратов, самое большее, четы- рех целых чисел. Например, 78 = 81 2 + З2 + 22 + I2 = 72 + 52 + 22. Я помню, как в инсти- тутском курсе теории чисел мне приходилось доказывать, что для выражения любого целого числа будет достаточно квадратов четырех целых чисел. Да. это знаменитая задача, но она была решена около 200 лет тому назад. — Нет, нас интересует другая версия проблемы Уоринга, с использованием пирами- дальных чисел. Пирамидальным называется число, которое можно представить в виде (т' - т)/6 при т> 2. Примером нескольких первых пирамидальных чисел будут 1, 4, 10, 20, 35, 56, 84, 120, 165. С 1928 года существует гипотеза, что любое целое число можно выразить суммой, самое большее, пяти пирамидальных чисел. Мы хотим с по- мощью суперкомпьютера доказать эту гипотезу для всех чисел от 1 до 1 000 000 000. — Миллиард любых вычислений займет значительное время,— предупредил я.— Критичным будет время, затраченное на вычисление минимального представления каждого числа, т. к. это придется делать миллиард раз. Вы думали, какой тип алгорит- ма использовать? — Мы уже написали свою программу и испытали ее на многопроцессорном суперком- пьютере. На небольших числах она работает очень быстро, но при работе с числами, большими 100 000, время значительно увеличивается. Все ясно, — подумал я. Этот "компьютероман" открыл асимптотический рост. Никаких сомнений, что он использовал алгоритм квадратичной временной сложности и столк- нулся с проблемами, как только число и стало достаточно большим. — Нам нужна более быстрая программа, чтобы дойти до одного миллиарда. Можете ли вы помочь нам с этой задачей? Конечно же, мы будем выполнять ее на нашем много- процессорном суперкомпьютере. Поскольку я люблю такой тип задач — разработку алгоритмов для ускорения времени работы программ — я согласился подумать над этим и принялся за работу. Я начал с просмотра программы, написанной моим посетителем. Он создал массив всех 0(н ) пирамидальных чисел от 1 до п включительно'. Для каждого числа к в этом диапазоне методом полного перебора выполнялась проверка, являлось ли оно суммой двух пирамидальных чисел. При отрицательном результате проверки выполнялась проверка, было ли число суммой трех пирамидальных чисел, потом четырех и, наконец, пяти, пока не находился ответ. Приблизительно 45% целых чисел можно вы- разить как сумму трех пирамидальных чисел. Большинство из оставшихся 55% чисел можно представить в виде суммы четырех пирамидальных чисел и, как правило. 1 Почему п|п? Вспомните, что пирамидальные числа выражаются формулой (и? - /и)/6. Наибольшее число т, такое что результат не превышает я, приблизительно равно >/бл ; поэтому количество таких чисел выра- жается формулой 0(я|/3).
Глава 2 Анализ алгоритмов 71 разными способами. Известно только 241 целое число, представляемое суммой пяти пирамидальных чисел, самое большое из которых равно 343 867. Для примерно поло- вины изд чисел этот алгоритм проверял все трехэлементные комбинации и, по крайней мере, некоторые четырехэлементные. Таким образом, общее время исполнения этого алгоритма было, как минимум, О(п х (и|/3)3) = О(п2), где п = I 000 000 000. Неудиви- тельно. что на больших числах (превышающих 100 000) эта программа замедляла ра- боту. Любое решение, работающее на больших числах значительно быстрее предложенного, должно избегать явной проверки всех трехэлементных комбинаций. Для каждого зна- чения к нам нужно наименьшее множество пирамидальных чисел, сумма которых в точности равна к. Эта задача называется задачей о рюкзаке и рассматривается в разде- ле 13.10. В нашем случае весу предметов соответствует набор пирамидальных чисел, не превышающих и, с тем дополнительным ограничением, что рюкзак вмещает ровно ^предметов (т. е. чисел, в нашем случае). В стандартном подходе к решению задачи о рюкзаке заранее вычисляются суммы меньших подмножеств, которые потом используются в вычислении больших подмно- жеств. Если у нас имеется таблица, содержащая суммы двух чисел, и мы хотим узнать, можно ли выразить число к в виде суммы трех чисел, мы можем перефразировать по- становку задачи и спросить, можно ли выразить число к в виде суммы одного числа и одной из сумм в данной таблице. Поэтому мне нужно было составить таблицу всех целых чисел, меньших, чем п, кото- рые можно выразить в виде суммы двух из 1 818 пирамидальных чисел, меньших чем 1000 000 000. Таких чисел может быть самое большее 1 8 1 82 = 3 3 05 1 24. Более того, если мы уберем повторяющиеся суммы и суммы, большие, чем целевое число, у нас останется меньше чем половина чисел. Создание отсортированного массива этих чисел не составит никакого труда. Назовем эту отсортированную структуру данных таблицей сумм двух пирамидальных чисел. Поиск минимального разложения данного числа к начинается с проверки, является ли оно одним из 1 818 пирамидальных чисел. Если не является, то тогда выполняется про- верка. не находится ли оно в таблице сумм двух пирамидальных чисел. Чтобы прове- рить, можно ли выразить число к в виде суммы трех пирамидальных чисел, нужно бы- ло всего лишь проверить, что к- p[i] находится в таблице сумм двух пирамидальных чисел при 1 < i < 1 818. Эту проверку можно было быстро выполнить посредством дво- ичного поиска. Чтобы проверить, можно ли выразить число к в виде суммы четырех пирамидальных чисел, нужно было всего лишь проверить, что к — /wo[z] находится в таблице сумм двух пирамидальных чисел для любого 1 < i < |Zwo|. Но так как почти любое число к можно выразить как сумму многих комбинаций четырех пирамидаль- ных чисел, эта проверка не займет много времени, и доминирующей составляющей времени общей проверки будет время, затраченное на проверку трехэлементных сумм. Временная сложность проверки, является ли число к суммой трех пирамидальных чи- сел, оценивается как O(«1/3lgn), а для всего множества п целых чисел — как C*(n4/3lg«). По сравнению с временной сложностью С\и2) алгоритма клиента для п = 1 000 000 000. мой алгоритм был в 30 000 раз быстрее. Первый прогон реализации этого алгоритма на моем далеко не новом компьютере PARC ELC занял около 20 минут для п = 1 000 000. После этого я экспериментировал с представлением наборов чисел разными структурами данных и с разными алгоритмами
72 Часть I. Практическая разработка алгоритмов для поиска в этих структурах. В частности, я попробовал использовать вместо отсорти- рованных массивов хэш-таблицы и двоичные векторы, поэкспериментировал с разно- видностями двоичного поиска, такими как интерполяционный поиск (см. раздел 14.2). Наградой за эту работу стало менее чем трехминутное время исполнения программы на множестве чисел п = 1 000 000. что в шесть раз лучше времени исполнения первона- чальной программы. Завершив основную работу, я занялся настройкой программы, чтобы немного повы- сить производительность. В частности, т. к. 1 — это пирамидальное число, то для лю- бого числа к. когда к- 1 было суммой трех пирамидальных чисел, я не вычислял сумму четырех пирамидальных чисел. В результате использования только этого приема об- щее время исполнения программы было сокращено еще на 10%. Наконец, с помощью профилировщика я выполнил несколько низкоуровневых настроек, чтобы еще чуть- чуть повысить производительность. Например, заменив лишь одну вызываемую про- цедуру встроенным кодом, я сбросил еще 10% с времени исполнения. После этого я передал программу заказчику. Он использовал ее самым неподходящим образом, о чем я расскажу в разделе 710. Подготавливая эту историю из жизни более чем десять лет спустя, я достал свою про- грамму из архивов и запустил ее на моем теперешнем настольном компьютере, SunBlade 150. Скомпилированная с помощью компилятора gcc без какой бы то ни бы- ло оптимизации программа обработала 1 000 000 чисел за 27 секунд. А время исполне- ния программы, скомпилированной с четвертым уровнем оптимизации, составило все- го лишь 14 секунд, что заставляет отдать должное качеству оптимизатора. Время ис- полнения на моем настольном компьютере улучшилось приблизительно в три раза за четыре года ко времени публикации первого издания этой книги и еще в 5.3 раза за последние 11 лет. Такое улучшение производительности типично для большинства на- стольных компьютеров. Основная цель моего рассказа состоит в том, чтобы показать громадный потенциал повышения скорости вычислений за счет улучшения эффективности алгоритмов, по сравнению с довольно скромным повышением производительности, получаемым за счет установки более дорогого оборудования. Применив более эффективный алгоритм, я повысил скорость вычислений приблизительно в 30 тысяч раз. Суперкомпьютер мое- го заказчика, стоивший миллион долларов, был оснащен 16 процессорами, каждый из которых мог выполнять целочисленные вычисления в пять раз быстрее, чем мой на- стольный компьютер стоимостью в 3 000 долларов. Но при использовании всей этой техники скорость выполнения моей программы возросла менее чем в 100 раз. Очевид- но, что в данном случае применение более эффективного алгоритма имеет преимуще- ство над использованием более мощного оборудования, что справедливо для любой задачи с достаточно большим вводом. 2.9. Анализ высшего уровня (*) В идеальном случае мы все умели бы свободно обращаться с математическими мето- дами асимптотического анализа. Точно так же, в идеале, мы все были бы богатыми и красивыми. А поскольку жизнь далека от идеала, математические методы асимптоти- ческого анализа требуют определенных знаний и практики.
Глава 2. Анализ алгоритмов 73 В этом разделе выполняется обзор основных методов и функций, применяемых в ана- лизе алгоритмов на высшем уровне. Это факультативный материал — он не использу- ется нигде в первой части этой книги. В то же самое время, знание этого материала будет большим подспорьем в понимании некоторых функций временной сложности, рассматриваемых во второй части. 2.9.1. Малораспространенные функции Основные классы функций временной сложности алгоритмов были представлены в разделе 2.3.1. Но в расширенном анализе алгоритмов также возникают и менее распро- страненные функции. И хотя такие функции нечасто встречаются в этой книге, вам бу- дет полезно знать, что они означают и откуда происходят. Далее приводятся некоторые из таких функций и их краткое описание. ♦ Обратная функция Аккермана /(«) = «(»). Эта функция появляется в подробном анализе нескольких алгоритмов. В данной книге точное определение этой функции и причины ее возникновения не рассматриваются. Будет достаточно воспринимать эту функцию как технический термин для самой медленно возрастающей функции сложности алгоритмов. В отли- чие от функции-константы/(«) = 1. эта функция достигает бесконечности при «—►<», но она определенно не торопится сделать это. Для любого значения п в физической вселенной значение функции будет меньше 5. т. е. а(я) < 5. ♦ /(и) = log log я. Смысл функции "log log" очевиден по ее имени— это логарифм логарифма числа и. Одним из естественных примеров ее возникновения будет дво- ичный поиск в отсортированном массиве из всего лишь lg/j элементов. ♦ f(n) = logn / log logw . Эта функция возрастает немного медленнее, чем функция log«, т. к. она содержит в знаменателе еще более медленно растущую функцию. Чтобы понять, как возникла эта функция, рассмотрим корневое дерево с количест- вом потомков <7, имеющее п листьев. Высота двоичных корневых деревьев, т. е. деревьев, у которых <7=2, определяется следующей формулой: п = 2Л —> h = Ig п получающейся в результате логарифмирования обеих частей равенства. Теперь рас- смотрим высоту дерева, когда количество потомков равно d = log/?. Тогда высота определяется такой формулой: п = (logn)—= logw/log log«. ♦ /(«) = log" п. Это произведение логарифмических функций, т. е. (logA?) х (log/?). Та- кая функция может возникнуть, если мы хотим сосчитать просмотренные биты в процессе двоичного поиска в множестве из п элементов, каждый из которых являет- ся целым числом в диапазоне от 1 до, например, п. Для представления каждого из этих целых чисел требуется lg(n")= 21gn бит, а поскольку количество чисел в мно- жестве поиска равно Ign, то общее количество бит будет равно 21g"w. Функция "логарифм в квадрате" обычно возникает при разработке сложных гнездо- вых структур, где каждый узел в, скажем, двоичном дереве представляет другую структуру данных, возможно, упорядоченную по другому ключу.
74 Часть I. Практическая разработка алгоритмов ♦ f(.n) = yJn . Функция квадратного корня встречается не так уж редко, но представ- ляет класс "сублинейных полиномов", т. к. у[п = п12. Такие функции возникают при построении сАмерных сеток, содержащих п точек. Площадь квадрата размером •Jn у[й равна п, а объем куба размером «l/j xnI/j хп|/3 также составляет п. В об- щем, объем <7-мерного гиперкуба со стороной n'li составляет п. ♦ /(и) = «(1+Е). Греческая буква в обозначает константу, которая может быть сколь угодно малой, но при этом не равняется нулю. Она может возникнуть в таких обстоятельствах. Допустим, время исполнения алго- ритма равно 2‘и(| + 1/0 и мы можем выбирать любое значение для с. При с = 2 время исполнения будет 4и3/2 или О(п3/‘). При с = 3 время исполнения будет 8и4/3 или О(п ), что уже лучше. Действительно, чем больше значение с, тем лучше становит- ся показатель степени. Но с нельзя сделать как угодно большим, до того как член 2‘ станет доминировать. Вместо этого мы обозначаем время исполнения этого алгоритма как О(п Е) и пре- доставляем пользователю определить наилучшее значение для е. 2.9.2. Пределы и отношения доминирования Отношения доминирования между функциями являются следствием теории пределов, изучаемой в курсе высшей математики. Говорят, что функция fin) доминирует над функцией g(n), если 1 im„_mg(«)//(«) = 0. Рассмотрим это определение в действии. Допустим, что/[«)= 2и2 и g(n) = г2. Очевид- но, что/(и) > g(n) для всех п, но не доминирует над ней, т. к. lim g(n) I f(n) = lim n2 / 2n2 = lim 1/2^0 Этого следовало ожидать, т. к. обе функции принадлежат к одному и тому же классу ®(и"). Теперь рассмотрим функции/^) = п и g(ri) = п2. Так как lim g(n) / f(n) = lim n2 / n3 = lim 1 / n = 0 то доминирует многочлен более высокого уровня. Это справедливо для любых двух многочленов, а именно п° доминирует над nh, если а > Ь, т. к. lim nh / па = lim nh~" —> 0 п—>00 П— Таким образом, и1,2 доминирует над л1-|999999< Перейдем к показательным функциям:/(«) = 3" и g(n) = 2". Так как lim g(ri) / f{n) = 2" / 3" = lim(2 / 3)" = 0 п—>оэ п—>ОЭ значит, доминирует функция с большим основанием. Возможность доказать отношение доминирования зависит от возможности доказать пределы. Давайте рассмотрим одну важную пару функций. Любой многочлен (скажем.
Гпава 2 Анализ алгоритмов 75 firi)= п) доминирует над логарифмическими функциями (например, g(n) = lg«). Так как п = 21в", то /(и) = (2lg")‘ = 2 lg". Теперь рассмотрим следующее тождество: limg(w)//(«) = lg??/2tle” n-+m В действительности при п—»оо оно стремится к нулю. Подведение итогов Рассматривая совместно функции, приведенные в этом разделе, и функции, обсуждаемые в разделе 2.3 1, мы видим полную картину порядка доминирования функций: и! » с"'» и2» п «log и » и » sjn » log7 и э» log?) » log/?/loglog« » » loglogw » a(n) » 1 Замечания к главе В других работах по алгоритмам уделяется значительно больше внимания формально- му анализу алгоритмов, чем в этой книге, поэтому читателей со склонностью к теоре- тическим выкладкам я отсылаю к другим изданиям. В частности, анализ алгоритмов рассматривается на более глубоком уровне в книгах [CLRS01] и [КТ06]. В книге [GK.P89] дается интересное и всестороннее изложение математического аппа- рата для анализа алгоритмов. Хорошее введение в теорию чисел, включая проблему Уоринга, упомянутую в разделе 2.8. дается в книге [NZ80]. Понятие доминирования также порождает обозначение о-малое. Говорят, что o(g(«)) тогда и только тогда, когда g(z?) доминирует над Среди прочего, обо- значение о-малое полезно для формулировки задач. Требование представить алгоритм с временной сложностью о(??~) означает, что нам нужен алгоритм с функцией времен- ной сложности лучшей, чем квадратичная, и что будет приемлемой временная слож- ность, выражаемая функцией O(n'’999log3 *n). 2.10. Упражнения Анализ программ I. [3] Какое значение возвращает следующая функция? Ответ должен быть в форме функ- ции числа и. Найдите время исполнения в наихудшем случае, используя обозначение О-бол ьшое. function mystery(n) r:=0 for i := 1 to n — 1 do for j := i + 1 to n do for k := 1 to j do r := r + 1 return(r) 2 [3] Какое значение возвращает следующая функция? Ответ должен быть в форме функ- ции числа и. Найдите время исполнения в наихудшем случае, используя обозначение О-бол ьшое.
76 Часть I. Практическая разработка алгоритмов function pesky(n) r:=Ci for i := 1 to n do for j := 1 to i do for к :=j to i + j do r := r + 1 return(r; 3. [5] Какое значение возвращает следующая функция? Ответ должен быть в форме функ- ции числа п. Найдите время исполнения в наихудшем случае, используя обозначение О-большое. function prestiferous(n) г :=0 for i := 1 to n do for j := 1 to i do for k j to i + j do for 1 := 1 to i + j - k do r := r + 1 return(r) 4. [8] Какое значение возвращает следующая функция? Ответ должен быть в форме функ- ции числа п. Найдите время исполнения в наихудшем случае, используя обозначение О-большое. function conundrum(n) t:=0 for i := 1 to n do for i := i + 1 to n do for k :=i + j — 1 to n do r := r + 1 return(r) 5. [5] Допустим, что для вычисления многочлена р(х) = а,д” + a„ i х" 1 + ... + аре + а0 ис- пользуется следующий алгоритм: р:=а0; xpower := 1; for . := to n do zpower ;= x * xpower; p := p + ai • xpower end • Сколько операций умножения выполняется в наихудшем случае? А сколько операций сложения? • Сколько операций умножения выполняется в среднем? • Можно ли улучшить этот алгоритм? 6. [3] Докажите правильность следующего алгоритма для вычисления максимального зна- чения в массиве Л[1..и]: function max(А) m:=A[l] for i 2 to n do if A[i] > m then m := A[i] return (m)
Гпава 2. Анализ алгоритмов 77 Упражнения по асимптотическим обозначениям 7. [3] Верно или неверно? а) 2"" = 0(2”) б) 22" = 0(2”) 8. [3] Для каждой из следующих пар функций функция /(л) является членом одного из множеств функций O(g(«)), Q(g(»)) или ®(g(«)). Определите, членом какого множества является функция в каждом случае и обоснуйте свой вывод. а)/(л) = logfl’. g(n) = logfl + 5 б)Дл) = Jn : g(«) = logn2 в) Ди) = log2«; g(«) = logn г)Дя) = л; g(») = log2fl Д) f(n) = fllogfl + n\ g(n) = log« е)Ди) = 10; g(n) = log 10 ж)Ди) - 2”; g(«) = 1Ои2 з) Ди) = 2": g(«) = 3" 9 [3] Для каждой из следующих пар функций Ди) и g(«) определите справедливо ли Д«) = O(g(«)) и g(n) = О(Дл)). а) Ди) = (а2 - л)/2; g(n) = 6л б) fin) - л >- 2 д/л ; g(«) = л2 в)/(«) = fllogfl; g(«) = л д/л /2 г) Ди) = л + logfl; g(«) = д/л Д) Ди) = 2(logfl)2; g(«) = log/7 + 1 е)Ди) 4fllogJ7 + л; g(n) = (и2 — и)/2 10. [3] Докажите, что и3 - лгг- л + 1 = ®(л3). II. [3] Докажите, что л2 = 0(2”). 12. [3] Для каждой из следующих пар функций Ди) и g(«) найдите положительную констан- ту с, при которой /(л) < с • g(w) для всех л > 1. а) f(n) - и2 + и г 1, g(n) = 2и3 б)/(л) = и д/л + л2, g(w) = и2 в) Дл) = л2 - л + 1, g(w) = и2/2 13. [3] Докажите, что если/,(л) = O(g,(w)) и /2(л) = O(g2(w)), тогда Д (и) +f2(n) = O(g}(n) + g2(fl)). 14. [3] Докажите, что если /,(А) = Q(gi(w)) и Д2(л) = Q(g2(n)), тогда/!(«)+ /з(«) = ^(gi(«) + gi(n)). 15. [3] Докажите, что если f(n) = O(g,(«)) и Д(и) = O(g2(«)), тогда Д(и) f2{n) = O(g\(n) g2(fl)). 16. [5] Докажите, что для всех к> 1 и всех множеств констант {оА, о* ь ..., аь a0}^R верно а*и* + а* [И* 1 ... + Ojfl + Oq = О(и*). 17. [5] Докажите, что для любых вещественных констант а и b, Ь > 0 верно (и + a)h = ®(лл).
78 Часть I. Практическая разработка алгоритмов 18. [5] Упорядочите указанные в таблице функции в возрастающем порядке. При наличии двух или более функций одинакового порядка укажите их. п 2" rtlgn 1пя п — п' + 7и5 1g/? х/л е" п2 + 1g/? 7 lg lg« (lg«)2 л! п + е, где 0 < £ < 1 19. [5] Упорядочите указанные в таблице функции в возрастающем порядке. При наличии двух или более функций одинакового порядка укажите их. х/й н 2" «logw п — п3 + 7и5 /?2 + log/? и3 log/? п'” + log/? (log/?)2 /?! 1пи «/log/? log log/? (1/3)" (3/2)" 6 20. [5] Найдите две функции//?) и g(«), которые удовлетворяют перечисленным условиям. Если таких/ng нет, укажите на этот факт в ответе. а)/п) = o(g(/?)) и/т?) * ©(g(/?)) б)/«) = ©(g(/?)) и//?) = o(g(/?)) в)/«) = ®(g(«)) и/(и) * O(g(«)) г)/«) = O(g( /?)) и//?) * O(g(/?)) 21. [5] Верно или неверно? а) 2«2 + 1 = О(п2) б) yfn = O(log/i) в) log/? = О( yfn ) г) И2(| + л/Й ) = O(/?2log/?) д) Зи2 + 4п = О(п2) е) х/л log/? = О(п) ж) log/? = О(п ,/2) 22. [5] Для каждой из следующих пар функции/л) и g(n) укажите, какие из перечисленных равенств справедливы://;) = O(g(«)),/«) = ©(g(«)),//?) = ©(g(/?)). а)/л)= и2 + Зл + 4, g(«) = 6н + 7 б)/(л) = л х/л , g(n) = л2 - п e)j(n} = 2n -л2, g(«) = «4 + /?2
Гпава 2. Анализ алгоритмов 79 23. [3] Ответьте на следующие вопросы и обоснуйте свой ответ. а) Если доказано, что время исполнения алгоритма в наихудшем случае определяется как возможно ли, что для некоторых входных экземпляров это время будет опре- деляться как О(н)? б) Если доказано, что время исполнения алгоритма в наихудшем случае определяется как О(п"). возможно ли, что для всех входных экземпляров это время будет определять- ся как ()(п)2 в) Если доказано, что время исполнения алгоритма в наихудшем случае определяется как ©(«'), возможно ли, что на некоторых входных экземплярах это время будет опре- деляться как О(и)? г) Если доказано, что время исполнения алгоритма в наихудшем случае определяется как ©(«'), возможно ли, что для всех входных экземпляров это время будет определять- ся как О(н)? д) Принадлежит ли функциями) множеству ©(/г2) (т. в. fin) = ©(и2)), где /(я) = 100л2 для четных n w fin) = 20л2 — wlog2H для нечетных л? 24. [3] Определите, верны ли следующие тождества, либо укажите, что это определить не- I возможно. Обоснуйте свой ответ. а) 3" = 0(2") б) log3" = O(log2") в) 3" = 0(2") г) log3" = Q(iog2") 25. [5] Для каждой из следующих функцийДл) найдите простую функцию g(n), для которой /«) = ®(g('O)- a) /(«) = Zli7 в) /(H)=£”=|log/ Г) /(л)= log(n!) 26. [5] Расположите следующие функции в возрастающем асимптотическом порядке: /(«) = n2 log2 л , /2(л) = n(Iog2 и)2, /3(и) = £"=о2' , /4(л) = log;(£"=02') 27. [5] Расположите следующие функции в возрастающем асимптотическом порядке. При наличии двух или более функций одинакового порядка укажите их. Л («) = XI1 Л («) = (V«) log л, /3 (я) = w^/logn , (л) = 1+ 4л 28. [5] Для каждой из следующих функций /(л) найдите простую функцию g(n), такую что Дл) = ©(g(n)). (Вообще говоря, вы должны уметь доказывать полученный результат, предоставляя соответствующие параметры, но это не требуется для данного задания.)
80 Часть I. Практическая разработка алгоритмов а) /(«) = XI, 3/4 + 2/’ -19/ + 20 б) Л«) = Х"=|3(4') + 2(3')-/|9+20 в) /(») = XI,5' + 32' 29. [5] Какое из следующих уравнений верно? а) £",3'=©(3"-) б) XI,3' =®(3") в) XI,3' = ®(3"+1) 30- [5] Для каждой из следующих функций Ди) найдите простую функцию g{n), при кото- рой/^) = ®(g(«)) а) Ди) = 0 000)2"+4" б) f> (п) = п + п log п + 4п В) /,(») = log(«20) + (logn)10 г) Д(и) = (0,99)"+и'°° 31- [5] Для каждой пары выражений (Л, В) в таблице укажите, какой именно функцией яв- ляется А для В — функцией О, о, О, w или О. Обратите внимание, что для любой из этих пар возможны несколько, один или ни одного варианта. Укажите все правильные отно- шения. А В а) и100 2" б) (lg»)‘2 x/w в) yfn ^cos(tth/8) г) 10" 100" д) и®' (lg»)'' е) lg(»!) wlgw Суммирование 32. [5] Докажите, что: 12 - 22 + З2 - 42 + ... + (-1 f'k1 = (-1 )* 'k(k + 1 )/2 33. [5] Определите формулу суммы для чисел строки / следующего треугольника и докажи- те ее правильность. Каждый элемент строки является суммой части строки из трех эле- ментов непосредственно над ним. Отсутствующие элементы полагаются равными нулю.
Гпава 2. Анализ алгоритмов 81 I I 3 1 4 10 1 1 1 1 2 3 2 1 6 7 6 3 1 16 19 16 10 4 1 34. [3] Допустим, что рождественские праздники длятся п дней. Сколько точно подарков прислала мне "любовь моя верная"? (Если вы не понимаете, о чем речь, выясните смысл вопроса самостоятельно.) 35. [5] Рассмотрим следующий фрагмент кода. tor 1=1 to n do for j=i to 2*i do output ''foobar11 Пусть T(n) означает, сколько раз слово "foobar” печатается в зависимости от значения н. а) Выразите 7’(и) в виде суммы (точнее, в виде двух вложенных сумм). б) Упростите выражение суммы. Полностью распишите все преобразования. 36. [5] Рассмотрим следующий фрагмент кода. for 1=1 to n/2 do for j=i to n-i do for k=l to ] do output 11foobar11 Допустим, что n — четное. Пусть Т(п) означает, сколько раз слово "foobar" выводится в зависимости от значения п. а) Выразите функцию Т(п) в виде трех вложенных сумм. б) Упростите выражение суммирования. Полностью распишите все преобразования. 37. [6] На уроках арифметики в школе нам говорили, что х * у означает, что число х нужно написать у раз подряд и сосчитать сумму, т. е. 5x4 = 5+ 5 + 54 5 = 20. Выразите в виде функции от п и Ь временную сложность умножения двух чисел, которые в 6-ичной сис- теме счисления состоят из п цифр (люди работают в десятичной системе счисления, а компьютеры— в двоичной) методом многократного сложения. Будем считать, что ум- ножение или сложение однозначных чисел занимает 0(1) времени. (Подсказка: поду- майте, насколько большим может быть число у в зависимости от п и Ь?) 38. [6] На уроках арифметики нас также учили умножать большие числа поразрядно, г. е. 127 х 211 = 127 х 1 + 127 х Ю + 127 х 200 = 26 397. Выразите в виде функции от п вре- менную сложность умножения этим методом двух чисел из и цифр. Будем считать, что умножение или сложение однозначных чисел занимает 0(1) времени. Логарифмы 39. [5] Докажите следующие тождества: а) log„ (ху) = log,, х + log,, у . б) log„xv =ylog„x.
82 Часть I. Практическая разработка алгоритмов в) log,, ,r = '°-' A log. a r) xlos‘ ’ = r 40. [3] Докажите, что pig(/? + lf| = |jgnJ +1 . 41. [3] Докажите, что двоичное представление числа п > 1 содержит llg, пJ +1 битов. 42. [5] В одной из моих научных статей я привел пример алгоритма сортировки методом сравнений с временной сложностью ()(n\og(4n)). Почему это возможно, если нижняя граница сортировки определена как Q(wlogw) ? Задачи, предлагаемые на собеседовании 43. [5] Имеется множество S из п чисел. Из этого множества нужно выбрать такое подмно- жество S' из к чисел, чтобы вероятность вхождения каждого элемента из множества .V в подмножество 5'была одинаковой (т. е., чтобы вероятность выбора каждого элемента была равна kin). Выбор нужно сделать за один проход по числам множества S. Решите задачу также для случая, когда число п неизвестно? 44 [5] Нужно сохранить тысячу элементов данных в тысяче узлов. В каждом узле можно сохранить копии только трех разных элементов. Разработайте схему создания копий, позволяющую минимизировать потерю данных при выходе узлов из строя. Сколько утерянных элементов данных нужно ожидать в случае выхода из строя трех произволь- ных узлов? 45. [5] Имеется следующий алгоритм поиска наименьшего числа в массиве чисел Л[0,. Для хранения текущего минимального числа используется переменная Imp. Начиная с ячейки массива Л[0] значение переменной tmp сравнивается по порядку со значениями ячеек Я[1], Л[2], ..., /4[/У]. Если значение ячейки 4[/] окажется меньше значения imp (/l[zj < imp), то оно записывается в переменную imp (tmp = Я[/]). Сколько нужно ожи- дать таких операций присваивания? 46. [5] Имеется 100-этажное здание и несколько небольших шариков из камня. Нужно оп- ределить самый нижний этаж, чтобы брошенный с него шарик разбился. Сколько вре- мени понадобится для определения этого этажа при неограниченном количестве шари- ков? При наличии только двух шариков? 47. [5] Вам дали 10 кошельков с золотыми монетами. Монеты в девяти из этих кошельков весят по 10 грамм каждая, а в оставшемся кошельке — на 1 грамм меньше. С помощью цифровых весов нужно найти кошелек с легкими монетами, выполнив лишь одно взве- шивание. 48. [5] У вас есть восемь шариков одинакового размера. Семь из них имеют одинаковый вес, восьмой шарик чуть тяжелее остальных. Нужно найти этот шарик, выполнив лишь два взвешивания. 49. [5] Допустим, что планируется слить п компаний в одну. Сколько имеется разных спо- собов для осуществления этого слияния? 50. [5] Число Рамануджана-Харди— это число, представимое в виде суммы двух кубов двумя различными способами. Иными словами, существуют четыре разных числа а, Ь. с
Гпава 2. Анализ алгоритмов 83 и d. для которых о3 + Ь~' = с3 + d\ Сгенерируйте все числа Рамануджана-Харди для а. Л, c.d< п. 51. [7] Шести пиратам нужно поделить между собой 300 долларов следующим способом. Самый старший пират предлагает, как нужно разделить деньги, после чего пираты го- лосуют по его предложению. Если предложение одобрено, по крайней мере половиной пиратов, то деньги распределяются в соответствии с предложенным способом. В про- тивном случае автора предложения убивают, и следующий по старшинству пират пред- лагает свой способ. Процесс повторяется. Ответьте, каков будет результат этого деле- ния, и обоснуйте. То есть, сколько пиратов останется в живых, и каким образом будут распределены деньги? Все пираты обладают хорошим умом, и самая приоритетная за- дача каждого— остаться в живых, а следующая по важности — получить как можно большую долю денег. 52. [7] Вариант предыдущей задачи. Пираты делят только один неделимый доллар. Кто по- лучит этот доллар, и сколько пиратов будет убито? Задачи по программированию Эти задачи доступны на сайтах http://wwvv.programming-challenges.com и http:/ /uva.onlinejudge.org. I. Primary Arithmetic. 110501/10035. 2. A Multiplication Game. 110505/847. 3. Light, More Light. 110701/10110.
ГЛАВА 3 Структуры данных Замен} структуры данных в медленно работающей программе сравнить с пересадкой органа. Такие важные классы абстрактных типов данных, как контейнеры, словари и очереди с приоритетами, могут реализовываться посредством разных, но функцио- нально эквивалентных структур данных. Замена структуры данных одного типа структурой данных другого типа не влияет на правильность программы, т. к. предпола- гается. что одна правильная реализация заменяется другой правильной реализацией. Но в реализации другого типа данных могут применяться операции с иными времен- ными отношениями, в результате чего общая производительность программы может значительно повыситься. Подобно ситуации с больным, нуждающимся в пересадке одного органа, для повышения производительности программы может оказаться доста- точным заменить лишь один ее компонент. Конечно, лучше с рождения иметь здоровое сердце, чем жить в ожидании донорского. То же самое справедливо и в случае со структурами данных. Наибольшую пользу от применения хороших структур данных можно получить, лишь заложив их использова- ние в программу с самого начала. При написании этой книги предполагалось, что ее читатели уже имеют знания об элементарных структурах данных и манипуляциях ука- зателями Но так как в сегодняшних курсах по структурам данных внимание фокусируется боль- ше на абстракции данных и на объектно-ориентированном программировании, чем на деталях представления структур в памяти, то мы повторим этот материал здесь, чтобы убедиться в том, что вы его полностью понимаете. Как и при изучении большинства предметов, при изучении структур данных важнее хорошо освоить основной материал, чем бегло ознакомиться с более сложными поня- тиями. Здесь мы обсудим три фундаментальных абстрактных типа данных, контейне- ры. словари и очереди с приоритетами, и рассмотрим, как они реализуются посредст- вом массивов и списков. Более сложные реализации структур данных рассматриваются в релевантной задаче в каталоге задач. 3.1. Смежные и связные структуры данных В зависимости от реализации (посредством массивов или указателей) структуры дан- ных можно четко разбить на два типа — смежные и связные. ♦ Смежные структуры данных реализованы в виде непрерывных блоков памяти. К ним относятся массивы, матрицы, кучи и хэш-таблицы. ♦ Связные структуры данных реализованы в отдельных блоках памяти, связанных вместе с помощью указателей. К этому виду структур данных относятся списки, деревья и списки смежных вершин графов.
Гпава 3. Структуры данных 85 В этом разделе дается краткий обзор сравнительных характеристик смежных и связных структур данных. Разница между ними более тонкая, чем может показаться с первого взгляда, поэтому я призываю не игнорировать этот материал, даже если вы и знакомы с этими типами структур данных. 3.1.1. Массивы Массив представляет собой основную структуру данных смежного типа. Записи дан- ных в массивах имеют постоянный размер, что позволяет с легкостью найти любой элемент по его индексу (или адресу). Хорошей аналогией массива будет улица с домами, где каждый элемент массива соот- ветствует дому, а индекс элемента— номеру дома. Считая, что все дома одинакового размера и пронумерованы последовательно от 1 до », можно определить точное место- нахождение каждого дома по его адресу1 Перечислим достоинства массивов. ♦ Постоянное время доступа при условии наличия индекса. Так как индекс каждого элемента массива соответствует определенному адресу в памяти, то при наличии соответствующего индекса доступ к произвольному элементу массива осуществля- ется практически мгновенно. ♦ Эффективное использование памяти. Массивы содержат только данные, поэтому память не тратится на указатели и другую форматирующую информацию. Кроме этого, для элементов массива не требуется использовать метку конца записи, т. к. все элементы массива имеют одинаковый размер. ♦ Локальность в памяти. Одна из самых распространенных идиом программирова- ния— обработка элементов структуры данных в цикле. Массивы хорошо подходят для операций такого типа, поскольку обладают отличной локальностью в памяти В современных компьютерных архитектурах физическая непрерывность последова- тельных обращений к данным помогает воспользоваться высокоскоростной кэш- памятью. Недостатком массивов является то. что их размер нельзя изменять в процессе исполне- ния программы. Попытка обращения к (и + 1)-му элементу массива размером п элемен- тов немедленно вызовет аварийное завершение программы. Этот недостаток можно компенсировать объявлением массивов очень больших размеров, но это может повлечь за собой чрезмерные затраты памяти, что опять наложит ограничения на возможности программы. В действительности, размеры массива можно изменять во время исполнения програм- мы посредством приема, называющегося динамическим выделением памяти. Допус- тим, мы начнем с одноэлементного массива размером т и будем удваивать его каждый раз до 2/и, когда предыдущий размер становится недостаточным. Этот процесс состоит из выделения памяти под новый непрерывный массив размером 2ш, копирования со- Данная аналогия неприменима в случае с нумерацией домов в Японии. Там дома нумеруют в по- рядке их возведения, а не физического расположения. Поэтому найти какой-либо дом в Японии по его адресу, не имея карты, очень трудно.
86 Часть I. Практическая разработка алгоритмов держимого старого массива в нижнюю половину нового и возвращения памяти старого массива в систему распределения памяти. Очевидным расточительством в этой процедуре является операция копирования со- держимого старого массива в новый при каждом удвоении размера массива. Это поро- ждает вопрос: сколько раз может возникнуть необходимость перекопировать элемент массива после п вставок? Давайте разберемся. Первый вставленный элемент нужно будет перекопировать при расширении массива после первой, второй, четвертой, вось- мой и т. д. вставок. Для расширения массива до размера в п элементов потребуется login удваиваний. Но большинство элементов не подвергается слишком большому числу перемещений. Более того, элементы с (72 + 1) по и будут перемещены, самое большее, один раз. а могут и вообще не перемещаться. Если половина элементов перемещается один раз, четверть элементов два раза и т. д„ то общее число перемещений определяется следующей формулой: lg w 1g п от Л/ = £ in/2‘ = /7 2' < /7 2'= 2п <=| <=1 i=i Таким образом, каждый из п элементов массива, в среднем, перемещается только два раза, а общая временная сложность управления динамическим массивом определяется той же самой функцией О(п\ какая справедлива для работы с одним статическим мас- сивом достаточного размера. Самой главной проблемой при использовании динамических массивов является отсут- ствие гарантии постоянства времени доступа в наихудшем случае. Теперь все обраще- ния будут быстрыми, за исключением тех относительно нечастых обращений, вызы- вающих удвоение массива. Зато у нас есть уверенность, что п-с обращение к массиву будет выполнено достаточно быстро, чтобы общее затраченное усилие осталось таким же <9(н). 3.1.2. Указатели и связные структуры данных Указатели позволяют удерживать воедино связные структуры. Указатель— это адрес ячейки памяти. Переменная, содержащая указатель на элемент данных, может предос- тавить большую гибкость в работе с этими данными, чем просто их копия. В качестве примера указателя можно привести номер сотового телефона, который позволяет свя- заться с владельцем телефона независимо от его местоположения внутри зоны дейст- вия сети. Так как синтаксис и возможности указателей значительно различаются в разных язы- ках программирования, то мы начнем с краткого обзора указателей в языке С. Пере- менная указателя р содержит адрес памяти, по которому находится определенный блок данных'. В языке С указатель при объявлении получает тип. соответствующий типу данных, на которые этот указатель может ссылаться. Операция, обозначаемая *р, называется разыменованием указателя и возвращает значение элемента данных, рас- ' В языке С разрешается прямая манипуляция адресами такими способами, которые могут привести .lava-программистов в ужас, но в этой книге мы воздержимся от применения подобных методов.
Глава 3 Структуры данных 87 положенного но адресу, содержащемуся в переменной указателя р. А операция, обо- значаемая &х, называется взятием адреса и возвращает адрес (т. е. указатель) данной переменной х. Специальное значение null применяется для обозначения неинициали- зированного указателя или указателя на последний элемент структуры данных. Все связные структуры данных имеют определенные общие свойства, что видно из следующего объявления типа связного списка (листинг 3.1). Листинг 3 1. Объявление структуры связного списка typedef struct list item_type item; /* Данные */ struct list *next; /* Указатель на следующий узел •/ list; В частности: ♦ каждый узел в нашей структуре данных (структура list) содержит одно или не- сколько полей, предназначенных для хранения данных (поле item); ♦ каждый узел также содержит поле указателя на следующий узел (поле next). Это означает, что в связных структурах данных большой объем используемой ими па- мяти должен отдаваться под хранение указателей, а не полезных данных; ♦ наконец, требуется указатель на начало структуры, чтобы мы знали, откуда начи- нать обращение к ней. Простейшей связной структурой является список, пример которого показан на рис. 3.1. Рис. 3.1. Пример связного списка с полями данных и указателями Списки поддерживают три основных операции: поиск (search), вставку (insert) и удале- ние (delete). В двунаправленных или двусвязных списках (doubly-linked list) каждый узел содержит указатель как на следующий, так и на предыдущий узел. Это упрощает опре- деленные операции за счет дополнительного поля указателя в каждом узле. Поиск элемента в связном списке Поиск элемента х в связном списке можно выполнять итеративным или рекурсивным методом. В листинге 3.2 приведен пример реализации рекурсивного поиска. Листинг 3.2. Рекурсивный поиск элемента в связном списке list *search_list(Jist *1, item_type x) if (1 = NULL) return(NULL); if (l->item = x) return(1);
88 Часть I. Практическая разработка алгоритмов : urn arch_list(l->next, х) ); Если список содержит элемент х, то он находится либо в начале списка, либо в мень- шей оставшейся части списка. В конце концов, задача сводится к поиску в пустом спи- ске. который, очевидно, не может содержать элемент х. Вставка элемента в связный список Код вставки элементов в однонаправленный связный список представлен в лис- тинге 3.3. Листинг 3.3. Вставка элемента ч однонаправленный связный список __i insert list (list • 4, Ltem_type x) List *p; /* Временный указатель */ p = mallcc( sizeof(list) ); p- -Item x; p- next »1; *1 = p: Так как нам не требуется содержать элементы списка в каком-либо определенном по- рядке. то мы можем вставлять каждый новый элемент туда, куда его проще всего вста- вить. Вставка элемента в начало списка позволяет избежать необходимости обхода списка, хотя и требует обновления указателя (переменная 1) на начало списка. Обратите внимание на две особенности языка С. Функция maiioc возвращает указатель на блок памяти достаточного размера, выделенный для нового узла, в котором будет храниться элемент х. Двойная звездочка (**1) означает, что переменная 1 является ука- зателем на указатель на узел списка. Таким образом, строчка кода *1=р,- копирует зна- чение р в блок памяти, на который ссылается переменная 1, которая является внешней переменной, обеспечивающей доступ к началу списка. Удаление элемента из связного списка Удаление элемента из связного списка является более сложной операцией. Сначала нам нужно найти указатель на элемент списка, предшествующий удаляемому элементу. Это выполняется рекурсивным способом (листинг 3.4). Листинг 3.4. Поиск указателя на элемент, предшествующий удаляемому list *predecessor_List(list *1, item_type x) if 11 == NULL) I I (l->next == NULL)) { printf("Error: predecessor sought on null list.Xn"); return(NULL);
Гпава 3. Структуры данных 89 i: L-.-ne - ->item == >:.i return(1); returni predecessor_List(l->next, x) ); Найти элемент, предшествующий удаляемому, нужно потому, что он содержит указа- тель г, <t на следующий, в данном случае удаляемый, узел, который нужно обновить после удаления узла. После проверки существования узла, подлежащего удалению, собственно операция удаления не представляет собой ничего сложного. При удалении первого элемента связного списка нужно быть особенно внимательным, чтобы не за- быть обновить указатель (переменную 1) на начало списка (листинг 3.5). Листинг 3.5. Удаление элемента связного списка Oelute_iist (list *-L, item_type x) list *p; /* Указатель на узел*/ list *pred; /* Указатель на предшествующий узел */ ist *search_list'), *predecessor_list(); search_list i *1,x) ; f (p NULL) , pred = predecessor_list(*l,x); if (pred == NULL) /* Соединяем список */ *1 - p->next; else pred->next = p->next; free(p); /* Освобождение памяти узла */ В языке С требуется явное освобождение памяти, поэтому после удаления узла необ- ходимо освободить занимаемую им память, чтобы возвратить ее в систему. 3.1.3. Сравнение Связные списки имеют следующие преимущества над статическими массивами: ♦ переполнение в связных структурах невозможно, если только сама память не пере- полнена; ♦ операции вставки и удаления элементов проще соответствующих операций над не- прерывными списками (т. е. массивами); ♦ при работе с большими записями перемещение указателей происходит легче и бы- стрее, чем перемещение самих записей. Недостатки связных списков таковы: ♦ связным структурам необходимо дополнительное место для хранения указателей; ♦ в связном списке нет эффективного произвольного доступа к элементам; ♦ массивы обладают лучшей локальностью в памяти и более эффективны в использо- вании кэш-памяти, чем связные списки.
90 Часть I. Практическая разработка алгоритмов Подведение итогов Динамическое выделение памяти обеспечивает гибкость в выборе способа и момента ис- пользования этого ограниченного ресурса. Напоследок заметим, что списки и массивы можно рассматривать, как рекурсивные объекты. ♦ Списки. После удаления первого элемента связного списка мы имеем такой же связный список, только меньшего размера. То же самое справедливо для строк, поскольку в результате удаления символов из строки получается более короткая строка. ♦ Массивы. Отделение первых к элементов от массива из п элементов дает нам два массива меньших размеров, а именно размером к и иэлементов соответственно. Знание этого свойства позволяет упростить обработку списков и создавать эффектив- ные алгоритмы типа "разделяй и властвуй", такие как быстрая сортировка (quicksort) и двоичный поиск. 3.2. Стеки и очереди Термин контейнер (container) обозначает структуру данных, позволяющую хранить и извлекать данные независимо от содержимого. В противоположность контейнерам, словари представляют собой абстрактные типы данных, которые извлекаются по клю- чевому значению или содержимому. Словари рассматриваются в разделе 3.3. Контейнеры различаются по поддерживаемому ими типу извлечения данных. В двух наиболее важных типах контейнеров порядок извлечения зависит от порядка помеще- ния: ♦ Стеки. Извлечение данных осуществляется в порядке LIFO ("last in, first out", "по- следним вошел — первым вышел"). Стеки легко реализуются и обладают высокой эффективностью. Ио этой причине стеки удобно применять в случаях, когда поря- док извлечения данных не имеет никакого значения, например, при обработке па- кетных заданий. Операции вставки и извлечения данных для стеков называются push (запись в стек) и pop (снятие со стека): • pushfx.s) — вставить элемент х на верх (в конец) стека s; • pop(s) — извлечь (и удалить) верхний (последний) элемент из стека s. Порядок LIFO возникает во многих реальных ситуациях. Например, из набитого битком вагона метро пассажиры выходят в порядке L1FO. Продукты из холодиль- ника нередко вынимаются в этом же порядке, с игнорированием сроков годности. По крайней мере, такой порядок применяется в моем холодильнике. В алгоритмах порядок LIFO обычно возникает при выполнении рекурсивных операций. ♦ Очереди. Очереди поддерживают порядок извлечения FIFO ("first-in, first-out", "пер- вым вошел — первым вышел"). Использование этого порядка определенно самый справедливый способ управления временем ожидания обслуживания. Компьютер обрабатывает задачи в порядке FIFO, чтобы минимизировать максимальное время ожидания. Обратите внимание, что среднее время ожидания будет одинаковым, не- зависимо от применяемого порядка, будь то FIFO или LIFO. Но так как данные мно-
Гпава 3. Структуры данных 91 гих вычислительных приложений не теряют актуальность бесконечно долго, вопрос максимального времени ожидания становится чисто академическим. Очереди реализовать несколько труднее, чем стеки, и поэтому их применение больше подходит для приложений, в которых порядок извлечения данных является важным, например, эмуляции определенных процессов. Применительно к очередям операции вставки и извлечения данных называются enqueue (поставить в очередь) и dequeue (вывести из очереди) соответственно: • enqueuefx,q) — вставить элемент х в конец очереди q; • dequeue(q) — извлечь (и удалить) элемент в начале очереди q. Далее в книге мы увидим применение очередей в качестве основной структуры данных для управления поиском в ширину в графах. На практике стеки и очереди можно реализовать посредством массивов или связных списков. Ключевой вопрос состоит в том. известна ли верхняя граница размера кон- тейнера заранее, т. е. имеется ли возможность использовать статические массивы. 3.3. Словари Тип данных словарь позволяет доступ к данным по содержимому. Словари применя- ются для хранения данных, которые можно быстро найти в случае надобности. Далее приводится список основных операций, поддерживаемых в словарях: ♦ search(D,k) — возвращает указатель на элемент словаря D с ключом к, если такой элемент существует; ♦ insert(D.x) — добавляет элемент, на который указывает х, в словарь £>; ♦ deleie(D.x) — удаляет из словаря D элемент, на который указывает.г. Некоторые словари также поддерживают другие полезные операции: ♦ max(D) и min(D) — возвращает указатель на элемент множества D. имеющий наи- больший или наименьший ключ. Это позволяет использовать словарь в качестве очереди с приоритетами, которая рассматривается в разделе 3.5\ ♦ predecessor(D,k) и successor(D,k) — возвращает элемент из отсортированного масси- ва D, предшествующий элементу с ключом к или стоящий сразу после него сооответ- ственно. Это позволяет обрабатывать в цикле все элементы этой структуры данных. С помощью только что перечисленных словарных операций можно выполнять многие распространенные задачи обработки данных. Допустим, нам нужно удалить все повто- ряющиеся имена из списка рассылки, затем отсортировать и распечатать получивший- ся список. Для выполнения этой задачи инициализируем пустой словарь D, для которо- го ключом поиска будет служить имя записи. Потом считываем записи из списка рас- сылки и для каждой записи выполняем операцию search в словаре на предмет наличия в нем данного имени. Если имя отсутствует в словаре, то вставляем его с помощью операции insert. Обработав весь список, извлекаем результаты из словаря. Для этого, начиная с наименьшего элемента словаря (операция min(D)). выполняем операцию successor, пока не дойдем до наибольшего элемента (операция max(D)). обойдя таким образом по порядку все элементы словаря.
92 Часть I. Практическая разработка алгоритмов Определяя подобные задачи в терминах абстрактных операций над словарем, мы от- влекаемся от деталей представления структуры данных и концентрируемся на решении задачи. Далее в этом разделе мы внимательно рассмотрим простые реализации словаря на ос- нове массивов и связных списков. Более мощные реализации, такие как использование двоичных деревьев поиска (см. раздел 3 4) и хэш-таблицы (см. раздел 3.7), также явля- ются привлекательными вариантами. Подробно словарные структуры данных рассмат- риваются в разделе 12.1. Читателям настоятельно рекомендуется просмотреть этот раз- дел. чтобы получить лучшее представление об имеющихся возможностях. Остановка для размышлений. Сравнение реализаций словаря (I) ЗАДАЧА. Определите асимптотическое время исполнения в наихудшем случае для каждой из семи основных словарных операций {search, insert, delete, successor, predecessor, minimum и maximum) для структуры данных, реализованной в виде: ♦ неотсортированного массива; ♦ отсортированного массива. Решение. Эта (а также следующая) задача демонстрирует компромиссы, на которые приходится идти при разработке структур данных. Конкретное представление данных может обеспечить эффективную реализацию одних операций за счет снижения эффек- тивности других. В решении этой задачи мы будем полагать, что кроме массивов, упомянутых в форму- лировке. мы имеем доступ к нескольким переменным, в частности, к переменной п, содержащей текущее количество элементов массива. Обратите внимание на необходи- мость обновлять эти переменные в операциях, изменяющих их значения (например, insert и delete), и на то, что стоимость такого сопровождения нужно включать в стои- мость этих операций. Сложность основных словарных операций для неотсортированных и отсортированных массивов показана в табл. 3.1. Таблица 3.1. Сложность основных словарных операций для массивов Словарная операция Неотсортированный массив Отсортированный массив search(L,k) O(n) O(logn) insert(Ljc) 0(1) O(n) delete(Lx) 0(1)* O(h) successor(Lx) О(и) 0(1) predecessor! /...y) O(n) 0(1) Чтобы понять, почему операция имеет данную сложность, необходимо выяснить, ка- ким образом она реализуется. Рассмотрим сначала эти операции для неотсортирован- ного массива/).
Гпава 3. Структуры данных 93 ♦ При выполнении операции search ключ поиска к сравнивается с (возможно) каждым элементом неотсортированного массива. Таким образом, в наихудшем случае, когда ключ к отсутствует в массиве, время поиска будет линейным. ♦ При выполнении операции insert значение переменной п увеличивается на единицу, после чего элементу копируется в п-ю ячейку массиваА[п]. Основная часть массива не затрагивается этой операцией, поэтому время ее исполнения будет постоянным. ♦ Операция delete несколько сложнее, что обозначается звездочкой (*) в табл. 3.1. По определению, в этой операции передается указатель х на элемент, который нужно удалить, поэтому нам не нужно тратить время на поиск этого элемента. Но удаление элемента из массива оставляет в нем промежуток, который нужно заполнить. Этот промежуток можно было бы заполнить, сдвинув все элементы массива, следующие за ним, т. е. элементы с Я[х + 1] по Я[л], вверх на одну позицию, но в случае удале- ния первого элемента эта операция займет время 0(п). Поэтому будет намного луч- ше просто записать в ячейку Л[х] содержимое последней ячейки Л[п] массива и уменьшить значение п на единицу. Время исполнения этой операции будет посто- янным. ♦ Определения операций predecessor и successor даются для отсортированного масси- ва. Но результатом выполнения этих операций в неотсортированном массиве не бу- дет элемент .4[х- 1] (или Л[х + 1]), т. к. физически предыдущий (или следующий элемент) не обязательно будет таковым логически В данном случае элементом, предшествующим элементу Л[х], будет наибольший из элементов, меньших Л[х]. Аналогично, следующим элементом за Л[х] будет наименьший из элементов, больших Л[х]. Чтобы найти предшествующий или следующий элемент в неотсорти- рованном массиве А. необходимо просмотреть все его п элементов, что занимает линейное время. ♦ Операции minimum и maximum также определены только для отсортированного массива. Поэтому, чтобы найти наибольший или наименьший элемент в неотсорти- рованном массиве, необходимо просмотреть все его элементы, что также занимает линейное время. Реализация словаря в отсортированном массиве переворачивает наши представления о трудных и легких операциях. В данном случае нахождение нужного элемент посред- ством двоичного поиска занимает время O(log/z), т. к. мы знаем, что средний элемент находится в ячейке массива А[п/2]. А поскольку верхняя и нижняя половины массива также отсортированы, то поиск может продолжаться рекурсивно в соответствующей половине. Количество делений массива пополам, требуемое для получения половины из одного элемента, равно [Igw]. Отсортированность массива полезна и для других словарных операций. Наименьший и наибольший элементы находятся в ячейках Л[1] и А[п] соответственно, а элементы, идущие непосредственно до и после элемента Л[х],— в ячейках J[x- 1] и Л[х+ 1J со- ответственно. Кооперации вставки и удаления элементов становятся более трудоемкими, т. к. созда- ние места для нового элемента или заполнение промежутка после удаления может по- требовать перемещения многих элементов массива. Вследствие этого обе операции исполняются за линейное время.
94 Часть I. Практическая разработка алгоритмов Подведение итогов При разработке структуры данных должны быть сбалансированы скорости выполнения всех поддерживаемых ею операций. Структура, обеспечивающая максимально быстрое выполнение как операции .1, так и операции В, вполне может оказаться не самой подхо- дящей (в смысле скорости работы) для этих операций, взятых в отдельности. Остановка для размышлений. Сравнение реализаций словаря (II) ЗАДАЧА. Определите асимптотическое время исполнения в наихудшем случае для каждой из семи основных словарных операций для структуры данных, реализованной в виде: ♦ однонаправленного связного неотсортированного списка; ♦ двунаправленного связного неотсортированного списка; ♦ однонаправленного связного отсортированного списка; ♦ двунаправленного связного отсортированного списка. Решение. При оценке производительности необходимо принимать во внимание два момента: тип связности списка (однонаправленная или двунаправленная) и его упоря- доченность (отсортированный или неотсортированный). Производительность этих операций для каждого типа структуры данных показана в табл. 3.2. Операции, оценка производительности которых представляет особые трудности, помечены звездоч- кой (*). Таблица 3.2. Производительность словарных операций в связных структурах данных Словарная операция Односвязный i ^отсортирован- ный список Двусвязный неотсортирован- ный список Односвязный отсортированный список Двусвязный отсортированный список search(L.sk) О(л) О(и) О(п) 0(11) inserl(Lx) 0(1) 0(1) О(п) О(п) delete(Lx) О(л)* 0(1) О(п)* 0(1) successor} L.x) О(и) 0(п) 0(1) 0(1) predecessor} Lx) О(п) О(п) О(п)* 0(1) minimum} L) О(п) О(п) 0(1) 0(1) maximum} L) О(п) О(п) 0(1)* 0(1) Так же. как и в случае с неотсортированными массивами, операция поиска неизбежно будет медленной, а операции модифицирования — быстрыми. ♦ Операции insert/delete. Здесь сложность возникает при удалении элемента из одно- связного списка. По определению операции delete передается указатель х на эле- мент, который нужно удалить. Однако в действительности нам нужен указатель на предшествующий элемент, т. к. именно его поле указателя необходимо обновить после удаления. Следовательно, требуется найти этот элемент, а его поиск в одно-
Гпава 3. Структуры данных 95 связном списке занимает линейное время. Данная проблема не возникает в двусвяз- ном списке, т. к. мы можем сразу же получить предшественника удаляемого элемента. Удаление элемента из отсортированного двусвязного списка выполняется быстрее, чем из отсортированного массива, т. к. связать предшествующий и последующий элементы списка дешевле, чем заполнять промежуток в массиве, перемещая остав- шиеся элементы Но удаление из односвязного отсортированного списка тоже ус- ложняется необходимостью поиска предшествующего элемента. ♦ Операция search. Упорядоченность элементов не дает таких преимуществ при поис- ке в связном списке, как при поиске в массиве. Мы больше не можем выполнять двоичный поиск, т. к. мы не можем получить доступ к среднему элементу, не обра- тившись ко всем предшествующим ему элементам. Однако упорядоченность прино- сит определенную пользу в виде быстрого завершения неуспешного поиска. Дейст- вительно, если мы не нашли запись Ahbott, дойдя до Costello', то можно сделать вы- вод, что такой записи не существует вообще. Тем не менее, поиск в наихудшем случае занимает линейное время. ♦ Операции predecessor и successor. Реализация операции predecessor усложняется уже упомянутой проблемой поиска указателя на предшествующий элемент. В от- сортированных списках обоих типов логически следующий элемент эквивалентен следующему узлу и, поэтому, время обращения к нему постоянно. ♦ Операция maximum. Наибольший элемент находится в конце списка, и чтобы доб- раться до него, обычно потребуется время 0(и) как в односвязном, так и в двусвяз- ном списке. Но можно поддерживать указатель на конец списка при условии, что нас устраива- ют расходы по обновлению этого указателя при каждой вставке и удалении. В дву- связных списках этот указатель на конечный элемент можно обновлять за постоян- ное время: при вставке проверять, имеет ли iast->next значение null, а при удале- нии переводить указатель last на предшественник last в списке, если удален последний элемент. В случае односвязных списков у нас нет эффективного способа найти этот предше- ственник. Как же мы получаем время исполнения операции maximum равным 0(1) в таких списках? Трюк заключается в списывании затрат на каждое удаление, кото- рое уже заняло линейное время. Дополнительный проход для обновления указате- ля, выполняемый за линейное время, не причиняет никакого вреда асимптотической сложности операции delete, но дает нам выигрыш в виде постоянного времени вы- полнения операции maximum как вознаграждение за четкое мышление. 3.4. Двоичные деревья поиска Пока что мы ознакомились со структурами данных, позволяющими выполнять быст- рый поиск или гибкое обновление, но не быстрый поиск и гибкое обновление одновре- менно. В неотсортированных двусвязных списках для операций вставки и удаления Abbott и Costello — знаменитый американский комический дун. — Прим перев.
96 Часть I. Практическая разработка алгоритмов записей требуется время 69(1)- а для операций поиска — линейное время в худшем слу- чае. Отсортированные массивы обеспечивают логарифмическое время выполнения двоичного поиска и линейное время выполнения вставок и удалений. Для двоичного поиска необходимо иметь быстрый доступ к двум элементам, а именно средним элементам верхней и нижней частей массива по отношению к данному эле- менту. Иными словами, нужен связный список, в котором каждый узел содержит два указателя. Так мы приходим к идее двоичных деревьев поиска. Корневое двоичное дерево определяется рекурсивно либо как пустое, либо как состоя- щее из узла, называющегося корнем, из которого выходят два корневых двоичных де- рева. называющиеся левым и правым поддеревом. В корневых деревьях порядок "род- ственных" узлов имеет значение, поэтому левое поддерево отличается от правого. На рис. 3.2 показаны пять разных двоичных деревьев, которые можно создать из трех узлов. Рис. 3.2. 11ять разных двоичных деревьев поиска с тремя узлами Каждый узел двоичного дерева поиска помечается ключом х таким образом, что для любого такого узла ключи всех узлов в его левом поддереве меньше х, а ключи всех узлов в его правом поддереве больше х. Это достаточно необычный способ постановки меток на деревьях поиска. Для любого двоичного дерева с количеством узлов п и лю- бого множества из п ключей существует только один способ постановки меток, кото- рый делает его двоичным деревом поиска. Допустимые способы постановки меток для двоичного дерева из трех узлов показаны на рис. 3.2. 3.4.1. Реализация двоичных деревьев Кроме поля ключа каждый узел двоичного дерева имеет поля left, right и parent, ко- торые содержат указатели на левый и правый дочерние узлы и на родительский узел соответственно. (Единственным узлом, указатель parent которого равен null, является корневой узел всего дерева.) Эти отношения показаны на рис. 3.3. В листинге 3.6 приводится объявление типа для структуры дерева. Листинг 3.6. Объявление типа для структуры дерева typedef struct tree item_type item; struct tree ‘parent; struct tree ‘left; struct tree ‘right; } tree; /* Элемент данных */ /* Указатель на родительский узел */ /★ Указатель на левый дочерний узел */ /* Указатель на правый дочерний узел */
Гпава 3. Структуры данных 97 Рис. 3.3. Отношения и двоичном дереве (а), поиск наименьшего (б) и наибольшего (в) элементов в двоичном дереве Двоичные деревья поддерживают три основные операции: поиск, обход, вставку и уда- ление. Поиск в дереве Схема маркировки двоичного дерева поиска однозначно определяет расположение каждого ключа. Поиск начинается с корневого узла дерева. Если узел не содержит ис- комый ключ а', то в зависимости от того, меньше или больше значение искомого ключа значения ключа корневого узла, поиск продолжается в левом или правом дочернем узле соответственно. Этот алгоритм основан на том, что как левое, так и правое подде- рево двоичного дерева сами являются двоичными деревьями. Реализация такого рекур- сивного алгоритма поиска в двоичном дереве показана в листинге 3.7. Листинг 3.7. Алгоритм рекурсивного поиска произвольного элемента в двоичном дереве tree * sea rcht гее (tree *1, item_type x) if 1 =- NULL) return(NULL); if (l->item == x) return(l); if l->item) return; search_tree(l->left, x) ); else return! search_tree(l->right, x) ); Время исполнения этого алгоритма равно где h обозначает высоту дерева. Поиск наименьшего и наибольшего элементов дерева Реализация операции поиска наименьшего элемента требует понимания, каким обра- зом этот элемент размещается в дереве. Согласно определению двоичного дерева, наи- меньший ключ должен находиться в левом поддереве корневого узла, т. к. значения всех ключей в этом дереве меньше, чем значение ключа корневого узла. Поэтому, как показано на рис. 3.3, б, наименьший элемент должен быть самым левым потомком корневого узла. Алгоритм поиска наименьшего элемента двоичного дерева показан в листинге 3.8. 4 Зак 3741
98 Часть I. Практическая разработка алгоритмов Листинг 3.8. Поиск наименьшего элемента ь двоичном дереве L ......... -..............Л..;..;....;......!......;;;..;...,. ......................................... tree *find_minimum(tree *t) { tree *min; /» Указатель на наименьший элемент */ if (t == NULL) return(NULL); min = t; while (min->left != NULL) min = min->left; return(min); Аналогично, наибольший элемент должен быть самым правым потомком корневого узла (см. рис. 3.3, б). Обход дерева Посещение всех узлов двоичного дерева является важной составляющей многих алго- ритмов. Эта операция является частным случаем задачи обхода всех узлов и ребер гра- фа, обсуждению которой посвящена глава 5. Основным применением алгоритма обхода дерева является перечисление ключей всех узлов дерева. Природа двоичного дерева позволяет с легкостью перечислить ключи всех его узлов в отсортированном порядке. Согласно определению двоичного дерева, все ключи со значением меньшим, чем значение ключа корневого узла, должны нахо- диться в левом поддереве корневого узла, а все ключи с большим значением — в пра- вом. Таким образом, результатом рекурсивного посещения узлов согласно данному принципу является симметричный обход (in-order traversal) двоичного дерева. Реализа- ция соответствующего алгоритма приведена в листинге 3.9. void traverse_tree(tree *1) { if (1 != NULL) { traverse_tree(l->left); process_item(l->item); traversetree(l->right); Вставка элементов в дерево В двоичном дереве Т имеется только одно место, в которое можно вставить новый элемент х, где его потом можно будет найти в случае необходимости. Для этого нужно заменить указателем на вставляемый элемент указатель null, возвращенный неуспеш- ным запросом поиска ключам в дереве. В реализации этой операции этапы поиска и вставки узла совмещаются с помощью рекурсии. Соответствующий алгоритм показан в листинге 3.10.
Гпава 3. Структуры данных 99 Листинг 3.10. Вставка узла в двоичное дерево insert_tree (tree **1, item_type х, tree *parent) tree *p; /* Временный указатель ★/ if <4 == NULL) ( p = malloc(sizeof(tree)); /* Выделение памяти для нового узла */ p->item = х; p->left = p->right = NULL; p->parent = parent; *1 = p; /★ Указатель на родительский узел */ return; if (x < (*l)->item) inserttree(&((*1)->left), x, *1); else insert_tree(s((*1)->right), x, *1); Процедура insert tree принимает три аргумента: ♦ указатель 1 на указатель, связывающий поддерево с остальной частью дерева; ♦ вставляемый ключ х; ♦ указатель parent на родительский узел, содержащий 1. Вставляемому узлу выделяется память и он интегрируется в структуру дерева, когда алгоритм находит указатель со значением null. Обратите внимание, что во время поис- ка соответствующему левому или правому указателю передается указатель, так что операция присваивания *1 = р,- интегрирует новый узел в структуру дерева. После выполнения поиска за время O(h) выделение памяти новому узлу и интегриро- вание его в структуру дерева выполняется за постоянное время. Удаление элемента из дерева Операция удаления элемента несколько сложнее, чем вставка, т. к. удаление узла озна- чает, что нужно должным образом связать получившиеся поддеревья с общим деревом. Существуют три типа удаления, показанные на рис. 3.4. Так как листовые узлы не имеют потомков, то их можно удалить, просто обнулив ука- затель на такой узел. Удаление узла с одним потомком также не вызывает проблем. Такой узел имеет один родительский и один дочерний узел, и последний можно связать непосредственно с родительским узлом, не нарушая при этом симметричную схему размещения ключей дерева. Но как удалить узел с двумя потомками? Нужно присвоить этому узлу ключ его непо- средственного потомка в отсортированном порядке. Этот дочерний узел должен иметь наименьшее значение ключа в правом поддереве, а именно быть самым левым узлом в правом поддереве родительского дерева (р). Перемещение этого узла на место удаляе- мого узла дает в результате двоичное дерево с должным расположением ключей, а
100 Часть I. Практическая разработка алгоритмов также сводит нашу задачу к физическому удалению узла с, самое большее, одним по- томком, а это задача уже была решена ранее. Исходное дерево Удаление узла без потомков (3) Удаление узла с одним потомком (6) Удаление узла с двумя потомками (4) Рис. 3.4. Удаление узла дерева без потомков, с одним потомком и с двумя потомками Полная реализация не приводится здесь по той причине, что она выглядит несколько устрашающе, хотя ее код логически следует из вышеизложенного описания. Как определить сложность наихудшего случая? Для каждого удаления требуется, самое большее, две операции поиска, каждая из которых выполняется за время б?(Л), где Л — высота дерева, плюс постоянные затраты времени на манипуляцию указателями. 3.4.2. Эффективность двоичных деревьев поиска Все три словарные операции, реализованные посредством двоичных деревьев поиска, исполняются за время O(hY где h — высота дерева. Дерево имеет самую меньшую вы- соту, на которую мы можем надеяться, когда оно полностью сбалансированно; тогда высота равна h = |~logn~|. Это очень хорошо, но дерево должно быть идеально сбалан- сированным. Наш алгоритм вставки помещает каждый новый элемент в лист дерева, в котором он должен находиться. Вследствие этого форма (и, что более важно, высота) дерева зави- сит от порядка вставки ключей. К сожалению, при построении дерева методом вставок могут происходить неприятные события, т. к. структура данных не контролирует порядок вставок. Например, посмот- рим, что случится, если вставлять ключи в отсортированном порядке. Здесь операции inserlfa), insertfb), inserl(c), insertfd) и т. д. дадут нам тонкое длинное дерево, в котором используются только правые узлы. Таким образом, высота двоичных деревьев может лежать в диапазоне от lg/7 до п. Но какова средняя высота дерева? Что именно мы считаем средней высотой? Вопрос бу- дет четко определен, если мы будем считать равновероятными каждое из л! возмож- ных упорядочиваний вставки и возьмем среднее их значение. В таком случае нам по- везло. т. к. с высокой вероятностью высота получившегося дерева будет O(logH). Это будет показано в разделе 4.6.
Глава 3. Структуры данных 101 Этот пример наглядно демонстрирует мощь рандомизации. Часто можно разработать простые алгоритмы, с высокой вероятностью имеющие хорошую производительность. Далее мы увидим, что подобная идея лежит в основе алгоритма быстрой сортировки quicksort. 3.4.3. Сбалансированные деревья поиска Произвольные деревья поиска обычно дают хорошие результаты. Но если нам не пове- зет с порядком вставок, то в наихудшем случае мы может получить дерево линейной высоты. Мы не можем прямо контролировать этот наихудший случай, т. к. мы должны создавать дерево в ответ на запросы, исходящие от пользователя. Но ситуацию можно улучшить с помощью процедуры вставки/удаления. которая после каждой вставки слегка корректирует дерево, удерживая его как можно более сбаланси- рованным, так что максимальная высота дерева будет логарифмической. Существуют сложные структуры данных в виде сбалансированных двоичных деревьев поиска, ко- торые гарантируют, что высота дерева всегда будет O(log/z). Вследствие этого время исполнения любой из словарных операций (вставка, удаление, запрос) будет равняться O(logn). Реализация структур данных сбалансированных деревьев, таких как красно- черные и косые деревья, обсуждается в разделе 12.1. С точки зрения разработки алгоритмов важно знать, что такие деревья существуют и что их можно использовать в качестве черного ящика, чтобы реализовать эффективный словарь. При расчете трудоемкости словарных операций для анализа алгоритма можно предположить, что сложность сбалансированного двоичного дерева в наихудшем слу- чае будет подходящей мерой. Подведение итогов Выбор неправильной структуры данных для поставленной задачи может иметь катастро- фические последствия для производительности. Определение самой лучшей структуры данных не настолько критично, т. к. возможно существование нескольких вариантов структур, имеющих примерно одинаковую производительность. Остановка для размышлений. Использование сбалансированных деревьев поиска ЗАДАЧА. Нужно прочитать п чисел и вывести их в отсортированном порядке Допус- тим. что у нас имеется сбалансированный словарь, который поддерживает операции search, insert, delete, minimum, maximum, successor и predecessor, время исполнения каждой из которых равно 6>(log/7). 1. Отсортируйте данные за время O(nlogn). используя только операции вставки и сим- метричного обхода. 2. Отсортируйте данные за время O(»log/7), используя только операции minimum, successor и insert. 3. Отсортируйте данные за время O(/zlogn), используя только операции minimum, insert, delete и search.
102 Часть I. Практическая разработка алгоритмов Решение. В первой задаче мы можем выполнять операции вставки и симметричного обхода. Это позволяет нам создать дерево поиска, вставив все п элементов, после чего выполнить обход дерева, чтобы отсортировать элементы. Sortl() initialize-tree(t) while (not EOF) read(x); insert(x, t) traverse(t) Sort2() initialize-tree(t) while (not EOF) read(x); insert(x,t); у = minimum(t) while (y#NULL) do print(y—item) у = successor(у,t) Sort3() initialize-tree(t) while (not EOF) read(x); insert(x,t); у = minimum(t) while (y#MULL) do print(y—item) delete(y,t) у = minimum(t) Во второй задаче, после создания дерева, мы можем использовать операции minimum и maximum. Для сортировки мы можем выполнить обход дерева, начав с наименьшего элемента и последовательно выполняя операцию поиска следующего элемента. В третьей задаче мы не имеем операции поиска следующего элемента, но можем вы- полнять удаление. Для сортировки мы можем выполнить обход дерева, опять начав с наименьшего элемента и последовательно выполняя операцию удаления следующего наименьшего элемента. Каждый из этих алгоритмов выполняет линейное количество операций с логарифмиче- ским временем исполнения, что дает общее время его исполнения O(nlogn). Основной подход к использованию сбалансированных двоичных деревьев заключается в том, чтобы рассматривать их как "черные ящики". 3.5. Очереди с приоритетами Многие алгоритмы обрабатывают элементы в определенном порядке. Допустим, на- пример, что нужно запланировать выполнение работ согласно их относительной важ- ности. Для этого работы нужно сначала отсортировать по важности, после чего оце- нить их в этом отсортированном порядке. Очереди с приоритетами предоставляют разработчику больше гибкости, чем обычная сортировка, т. к. они позволяют вводить новые элементы в систему в произвольном месте. Намного эффективнее вставить новый элемент в нужное место в очереди с при- оритетами, чем сортировать заново все элементы при каждой вставке. Базовая очередь с приоритетами поддерживает три основные операции: ♦ insert(Q,x) — вставляет в очередь с приоритетами Q элемент х с ключом к; ♦ find-minimum(Q) и find-maximum(Q) — возвращает указатель на элемент с наимень- шим (наибольшим) значением ключа в очереди с приоритетами Q\ ♦ delete-minimum(Q) и delete-maximum(Q) — удаляет из очереди Q элемент с наи- меньшим (наибольшим) значением ключа. Очереди с приоритетами точно моделируют многие природные процессы. Например, люди с неустроенной личной жизнью мысленно (или открыто) поддерживают очередь
Гпава 3. Структуры данных 103 с приоритетами из возможных кандидатов на роль постоянного партнера. Впечатления о новом знакомстве отображаются непосредственно на шкале привлекательности. Привлекательность здесь играет роль ключа для создания нового элемента в очереди с приоритетами. Выбор партнера — это процесс, включающий в себя извлечение наибо- лее привлекательного элемента из очереди (выполняется операция find-maximum}, сви- дание с ним для уточнения степени привлекательности и вставку обратно в очередь с приоритетами, возможно, в другое место с новым значением привлекательности. Подведение итогов Словари и очереди с приоритетами позволяют создавать алгоритмы с аккуратной струк- турой и хорошей производительностью. Остановка для размышлений. Построение базовых очередей с приоритетами ЗАДАЧА. Определить временную сложность в наихудшем случае трех основных опе- раций очереди с приоритетами (вставка элемента, поиск наименьшего элемента и по- иск наибольшего элемента) для следующих базовых структур данных: ♦ неотсортированного массива; ♦ отсортированного массива; ♦ сбалансированного двоичного дерева. Решение. В реализации этих трех операций присутствуют некоторые тонкости, даже при использовании такой простой структуры данных, как неотсортированный массив. В словаре на основе неотсортированного массива (см. раздел 3.3) операции вставки и удаления выполняются за постоянное время, а операции поиска произвольного и наи- меньшего элементов — за линейное время. Операцию удаления наименьшего элемента (delete-minimum), выполняемую за линейное время, можно составить из последова- тельности операций find-minimum, search и delete. В отсортированном массиве операции вставки и удаления можно выполнить за линей- ное время, а найти наименьший элемент— за постоянное. Но любые удаления из оче- реди с приоритетами затрагивают только наименьший элемент. После сортировки мас- сива в обратном порядке (наибольший элемент вверху) наименьший элемент будет са- мым последним элементом массива. Чтобы удалить последний элемент, не нужно перемещать никакие элементы, а только уменьшить на единицу количество оставшихся элементов и. вследствие чего операция удаления наименьшего элемента может выпол- няться за постоянное время. Все это правильно, но. как можно видеть из следующей таблицы, операцию поиска наименьшего значения, выполняемую за постоянное время, можно реализовать для каждой структуры данных. Операция Неотсортированный массив Отсортированный массив Сбалансированное дерево inserl(Ox) 0(1) О(л) O(logn) find-mininiuin(O) 0(1) 0(1) 0(1) delete-minimum( О) О(и) 0(1) O(logn)
104 Часть I. Практическая разработка алгоритмов Вся хитрость заключается в использовании дополнительной переменной для хранения указателя на наименьший элемент в каждой из этих структур, что позволяет просто возвратить это значение в любое время при получении запроса на поиск этого элемен- та. Обновление этого указателя при каждой вставке не составляет труда — он обновля- ется тогда и только тогда, когда вставленный элемент меньше, чем текущий наимень- ший элемент. Но что будет в случае удаления наименьшего элемента? Мы можем уда- лить текущий наименьший элемент, после чего найти новый наименьший элемент и зафиксировать его в этом качестве до тех пор, пока он тоже не будет удален. Настоя- щая операция поиска наименьшего элемента занимает линейное время для неотсорти- рованных массивов и логарифмическое время для деревьев и поэтому затраты на ее выполнение можно включить в затраты на выполнение каждого удаления. Очереди с приоритетами являются очень полезными структурами данных. Особенно изящная реализация очереди с приоритетами— пирамида рассматривается в кон- тексте задачи сортировки в разделе 4.3. Кроме этого, в разделе 12.2 дается полный на- бор реализаций очереди с приоритетами. 3.6. История из жизни. Триангуляция Применяемые в компьютерной графике геометрические модели обычно разбивают на множество небольших треугольников, как показано на рис. 3.5. Рис. 3.5.1 риангуляционная модель динозавра (а), использование нескольких полос треугольников (6) Для прорисовывания и закрашивания треугольников применяются высокопроизводи- тельные движки визуализации (rendering engine), работающие на специализированном аппаратном обеспечении. Скорость работы этого аппаратного обеспечения такая высо- кая, что единственным узким местом в системе являются затраты на ввод в него триан- гулированной структуры. Хотя каждый треугольник можно описать, указав три его вершины, альтернативное представление более эффективно. Вместо того чтобы определять каждый треугольник по отдельности, мы соединяем их в полосы смежных треугольников и обрабатываем их, перемещаясь вдоль этой полосы. Поскольку в этом случае каждый треугольник имеет две общие вершины со своим соседом, то мы экономим на расходах по передаче данных о двух вершинах и другой сопутствующей информации. Для однозначного описания треугольников в рендерере треугольной сетки OpenGL принимается согла- шение, что все повороты чередуются влево-вправо, как показано на рис. 3.6.
Гпава 3 Структуры данных 105 Рис. 3.6. Разбиение треугольной сетки на полосы: с чередующимися влево-вправо поворотами (и). с произвольными поворотами (б) Задачу определения наименьшего количества полос, которые покрывают все треуголь- ники в сетке, можно рассматривать как задачу графов. В таком графе каждый тре- угольник сетки представляется вершиной, а смежные треугольники представляются ребром между соответствующими вершинами. Такое представление в виде двойствен- ного грифа содержит всю информацию, необходимую для разбиения треугольной сет- ки на полосы треугольников (см. раздел 15.12). После получения двойственного графа можно было приступать к работе. Мы хотели разбить множество вершин на минимально возможное количество путей (или полос). Если бы мы смогли объединить их в один путь, то это бы означало, что мы нашли гамильтонов путь, т. е. путь, содержащий каждую вершину графа ровно один раз. Так как поиск гамильтонова пути является NP-полной задачей (см, раздел 16 5), мы знали, что нам не стоит искать алгоритм точного решения, а концентрироваться на эвристиче- ском алгоритме приблизительного решения. Самый простой эвристический алгоритм для полосового покрытия начинает работу с произвольного треугольника и двигается направо, пока не дойдет до границы объекта или до уже посещенного треугольника. Достоинством этого эвристического алгоритма являются его быстрота и простота, но при этом нет гарантии, что удастся найти наи- меньший набор ориентированных слева направо полос для данной триангуляции. А вот получение такого набора с помощью "жадного" эвристического алгоритма более веро- ятно. "Жадные" алгоритмы выбирают оптимальные возможные решения на каждом этапе в надежде, что и конечное решение также будет оптимальным. При выполнении триангуляции "жадный" алгоритм определяет начальный треугольник самой длинной полосы слева направо и отделяет ее. Носам факт использования "жадного" алгоритма не гарантирует наилучшее возможное решение, т. к. первая отделенная полоса может разбить много потенциальных полос, которые можно было бы использовать в дальнейшем. Тем не менее, хорошим практи- ческим правилом для получения оптимального решения будет использование "жадно- го" алгоритма. Так как отделение самой длинной полосы оставляет самое меньшее ко- личество треугольников для создания последующих полос, то "жадный" эвристический алгоритм является более производительным, чем простой эвристический алгоритм. Но сколько времени понадобится этому алгоритму, чтобы найти следующую самую длинную полосу треугольников? Пусть к— длина прохода, начинающегося в средней вершине. Используя самую простую реализацию алгоритма, можно выполнять проход от каждой из п вершин, что позволит найти самую длинную оставшуюся полосу за время О(кп). Повторение такого прохода для каждой из почти п/к извлекаемых полос
106 Часть I. Практическая разработка алгоритмов дает нам время исполнения О(п), что неприемлемо медленно для типичной модели, состоящей из 20 тысяч треугольников. Как можно улучшить это время? Интуитивно кажется, что неэкономично повторять проход от каждого треугольника после удаления лишь одной полосы. Можно просто хранить информацию о длине всех возможных будущих полос в какой-либо структуре данных. Но при удалении каждой полосы необходимо обновить информацию о длине всех других затронутых этим удалением полос. Эти полосы будут укорочены, т. к. они проходят через треугольник, которого больше нет. Такая структура данных будет иметь характеристики двух структур: ♦ Очередь с приоритетами. Так как мы многократно повторяем процесс определения самой длинной оставшейся полосы, то нам нужна очередь с приоритетами для хра- нения полос, упорядоченных по длине. Следующая удаляемая полоса всегда нахо- дится в начале очереди. Наша очередь с приоритетами должна была разрешать по- нижение приоритета произвольных элементов в очереди при каждом обновлении информации о длине полос, чтобы мы знали, какие треугольники были удалены. Так как длина всех полос ограничивалась довольно небольшим целым числом (ап- паратные возможности не позволяли иметь в полосе больше 256 вершин), мы ис- пользовали ограниченную по высоте очередь с приоритетами (массив корзин, пока- занный на рис. 3.7 и рассматриваемый в разделе 12.2). Обычная куча также была бы вполне приемлемой. верхняя 1раница 1 2 3 4 • • • 253 254 255 256 I I । и iym X Т I, Г г Рис. 3.7. Ограниченная по высоте очередь с приоритетами для полос треугольников Для обновления элемента очереди, связанного с треугольником, требовалось быстро находить его. Это означало, что нам нужен словарь. ♦ Словарь. Для каждого треугольника в сетке нам нужно было знать его расположе- ние в очереди. Это означало, что в словаре нужно было иметь указатель на каждый треугольник. Объединив этот словарь с очередью с приоритетами, мы создали структуру данных, способную поддерживать широкий диапазон операций. Хотя возникали и другие трудности, например, необходимость найти способ быстрого вычисления новой длины у полос, затронутых удалением полосы, главный метод по- вышения производительности состоял в применении очереди с приоритетами. Исполь- зование этой структуры данных улучшило время исполнения на несколько порядков.
Глава 3. Структуры данных 107 Из табл. 3.3 видно, насколько производительность "жадного" алгоритма выше, чем у простого. Таблица 3.3. Сравнение производительности "жадного" и простого алгоритмов для нескольких типов сетки треугольников Название модели Количество треугольников Затраты простого алгоритма Затраты "жадного" алгоритма Время "жадного" алгоритма Ныряльщик 3 798 8 460 4 650 6,4 сек Головы 4 157 10 588 4 749 9,9 сек Каркас 5 602 9 274 7210 9,7 сек Барт Симпсон 9 654 24 934 И 676 20,5 сек Космический корабль Enterprise 12710 29016 13 738 26,2 сек Тор 20 000 40 000 20 200 272,7 сек Челюсть 75 842 104 203 95 020 136,2 сек Во всех случаях "жадный" алгоритм выдавал набор полос меньшей стоимости в смыс- ле суммарной длины полос. Экономия составляла от 10 до 50%, что было просто заме- чательно, т. к. максимально возможное улучшение (уменьшение количества вершин каждого треугольника с трех до одной) позволяет сэкономить не более 66,6%. Результатом реализации "жадного" алгоритма со структурой данных в виде очереди с приоритетами было время исполнения программы О(и«А), где п— количество тре- угольников, к— длина средней полосы. Вследствие этого обработка тора, состоящего из небольшого количества очень длинных полос, занимала больше времени, что обра- ботка челюсти, хотя количество треугольников в последней фигуре было втрое больше. Из этой истории можно извлечь несколько уроков. Во-первых, при работе с достаточно большим набором данных только алгоритмы с линейным или почти линейным време- нем исполнения (скажем, O(n\ogn)) могут быть достаточно быстрыми. Во-вторых, вы- бор правильной структуры данных часто является ключевым фактором для уменьше- ния временной сложности алгоритма в такой степени. И в-третьих, применение "жад- ного" эвристического алгоритма может значительно улучшить производительность по сравнению с применением простого алгоритма. Насколько лучше будет производи- тельность, можно определить только экспериментальным путем. 3.7. Хэширование и строки Хэш-таблицы предоставляют очень практичный способ реализации словарей. В них используется то обстоятельство, что поиск элемента массива по индексу выполняется за постоянное время. Хэш-функция соотносит набор ключей с целочисленными значе- ниями. Мы будем использовать значение нашей хэш-функции в качестве индекса мас- сива и записывать элемент в этой позиции.
108 Часть I Практическая разработка алгоритмов Сначала хэш-функция соотносит каждый ключ с большим целым числом. Пусть значе- ние а представляет размер алфавита, используемого для создания строки 5. Пусть char(c) будет функцией, которая однозначно отображает каждый символ алфавита в ЬН целое число в диапазоне от 0 до а - 1. Функция H(S) = а < +l> х char(S,) однозначно /=0 отображает каждую строку в (большое) целое число, рассматривая символы строки как "цифры" системы счисления с основанием а. В результате получатся уникальные идентификационные числа, но они будут настоль- ко большими, что очень быстро превысят количество ячеек в нашей хэш-таблице (обо- значаемое т). Это значение необходимо уменьшить до целого числа в диапазоне от О до т- 1, для чего выполняется операция получения остатка от деления H(S) mod т. Здесь применяется тот же самый принцип, что и в колесе рулетки. Шарик проходит долгий путь, обходя раз колесо окружностью т, пока не остановится в произвольной ячейке, размер которой составляет незначительную часть пройденного пути. Если выбрать размер таблицы с достаточной тщательностью (в идеале, число т должно быть большим простым числом, не слишком близким к 2' — 1), то распределе- ние полученных хэш-значений будет довольно равномерным. 3.7.1. Коллизии Независимо от того, насколько хороша наша хэш-функция, время от времени она будет отображать два разных ключа в одно хэш-значение, и нужно быть готовым к подобной ситуации. Самым легким способом разрешения таких коллизий является применение цепочек. Для этого хэш-таблица реализуется в виде массива из т связных списков, как показано на рис. 3.8. Рис. 3.8. Разрешение коллизий при помощи цепочек Список с порядковым номером i содержит все элементы, хэшированные в одно и то же значение /. Таким образом, операции поиска, вставки и удаления сводятся к соответст- вующим операциям в связных списках. Если п ключей распределены в таблице равно- мерно, то каждый список будет содержать приблизительно п/т элементов, делая их размер постоянным при т = п. Метод цепочек — самый естественный, но он требует значительных объемов памяти для хранения указателей. Эту память можно было использовать, чтобы сделать таблицу больше, а "списки", соответственно, меньше.
Гпава 3 Структуры данных 109 Другой способ разрешения коллизий хэш-значений называется открытой адресацией. При использовании этого метода хэш-таблица реализуется в виде массива ключей (а не корзин), каждый из которых инициализирован значением null, как показано на рис. 3.9. I 23 4 5 6 7 89 ЮН I I 7 X I I X I X I I X I X I I 1 I Рис. 3.9. Разрешение коллизий методом открытой адресации При вставке ключа выполняется проверка, свободна ли требуемая ячейка. Если сво- бодна, то вставка выполняется. В противном случае нам нужно найти другое место, куда вставить данный ключ. Самый простой подход к определению альтернативного места для вставки называется последовательным исследованием (sequential probing). Этот метод заключается в том, что последовательно исследуются следующие ячейки таблицы, пока не будет найдена свободная, в которую и выполняется вставка. Если таблица не слишком заполнена, то последовательность смежных занятых ячеек должна быть довольно небольшой, и свободная ячейка должна находиться лишь в нескольких позициях от требуемой. Теперь для поиска определенного ключа нужно взять соответствующее хэш-значение и проверить, является ли его значение тем, какое нам нужно. Если да. то поиск заканчи- вается. В противном случае поиск продолжается в последовательности смежных ячеек, пока не будет найдено требуемое значение. Удаление из хэш-таблицы с открытой адресацией довольно сложная процедура, т. к. удаление одного элемента может разорвать цепочку вставок, вследствие чего некото- рые элементы последовательности больше не будут доступны. Не остается иного вы- хода, кроме как выполнить повторную вставку всех элементов, следующих за удален- ным. В обоих методах разрешения коллизий требуется время О(т) для инициализации ну- лями хэш-таблицы из т элементов до того, как можно будет выполнять первую встав- ку. Обход всех элементов в таблице цепочек занимает время О(п + /и), т. к. нам нужно просмотреть все т корзин в процессе поиска, даже если в действительности таблица содержит небольшое количество вставленных элементов. Для таблицы с открытой ад- ресацией это время равно О(т), т. к. значение п может быть равным, самое большее, значению т. При использовании метода цепочек для разрешения коллизий в хэш-таблице из т эле- ментов словарные операции для п элементов можно реализовать со следующими пока- зателями времени исполнения в ожидаемом и наихудшем случае: Операция Ожидаемое время исполнения Время исполнения в наихудшем случае search(I.k) О(п/т) О(п) insert(Ljc) (W 0(1)
110 Часть I. Практическая разработка алгоритмов (окончание) Операция Ожидаемое время исполнения Время исполнения в наихудшем случае deleted, л) O(l) 0(1) successor^ Loe) O(n + in) О(и + т) predecessor) Ljc) O(n + m) О(п + т) minimumlL) O(n + in) О(п + т) maximum(L) O(n + in) О(п + т) С практической точки зрения, хэш-таблица часто является самым лучшим типом структуры данных для реализации словаря. Но область применения хэширования го- раздо шире, чем просто создание словарей, в чем мы скоро убедимся. 3,7.2. Эффективный метод поиска строк посредством хэширования Строки представляют собой последовательности символов, порядок размещения кото- рых имеет значение, т. к. строка АЛГОРИФМ (устаревший вариант слова "алгоритм") отличается от строки ЛОГАРИФМ. Текстовые строки являются основным типом дан- ных для многих компьютерных приложений, от синтаксического разбора и компили- рования в языках программирования до поисковых систем Интернета и анализаторов биологических рядов. Основной структурой данных для представления строк является массив символов, что обеспечивает постоянное время доступа к /-му символу строки. Для обозначения конца строки требуется хранить вместе с ней определенную вспомогательноую информа- цию— специальный символ "конец строки" или (что, возможно, полезнее) количество символов в строке. Основной строковой операцией является поиск подстроки в строке. В следующей зада- че и ее решении приводится демонстрация такой операции. ЗАДАЧА. Найти подстроку в строке. Вход. Текстовая строка t и строка для поискар. Выход. Определить, содержит ли строка t подстроку р, и если содержит, то в каком месте? Самый простой алгоритм поиска подстроки в строке циклически накладывает подстро- ку р на строку /, перемещаясь вправо на одну позицию в строке t, пока последний сим- вол подстроки р не совмещается с последним символом строки /, и проверяет на совпа- дение каждого символа подстроки р с соответствующим символом строки t. Как было показано в разделе 2.5.3, время исполнения такого алгоритма равно О(щн), где п = |/| и т = |р|. Это квадратичный предел времени исполнения в наихудшем случае. Но существуют алгоритмы и с линейным временем исполнения в наихудшем случае, но более слож-
Глава 3. Структуры данных 111 ные. Такие алгоритмы подробно рассматриваются в разделе 18.3. Здесь же мы рас- смотрим алгоритм поиска подстроки с ожидаемым линейным временем исполнения, называющийся алгоритмом Рабина-Карпа и основывающийся на использовании хэши- рования. Допустим, что мы вычислим определенную хэш-функцию от строки-образца р и от подстроки длиной т символов, начинающейся в позиции i текста /. Очевидно, что если эти две строки одинаковы, то и их хэш-значения должны быть одинаковыми. Если же строки разные, то и их хэш-значения почти наверняка будут разными. Одина- ковые хэш-значения для разных строк должны встречаться настолько редко, что мы вполне можем позволить себе потратить время О(т) на явную проверку идентичности строк при совпадении хэш-значений. Это сводит сложность поиска подстроки к п-т + 2 вычислениям хэш-значений (п-т + I хэш-значений для окон-подстрок в тексте /, плюс одно хэш-значение для строки-образца /?), плюс несколько операций проверки с временем исполнения О(т). количество которых должно быть очень небольшим. Но проблема состоит в том, что вычисление хэш-функции для строки из т символов занимает время О(т), а О(п) таких вычислений, по-видимому, опять дают нам общее время исполнения алгоритма О(тп) в качестве оценки сложности. Давайте подробнее рассмотрим применение ранее определенной хэш-функции к т М-1 символам строки S. начиная с позицииj: P(S,j) = а"' ('+,) х charts,+7). /=| Что изменится, если сейчас мы попробуем вычислить значение H(S,j + 1), т. е. — хэш- значение следующей подстроки длиной в т символов? Обратите внимание, что т - 1 символов одинаковы в обеих подстроках, хотя количество умножений на а различается у них на единицу. Выполнив некоторые алгебраические преобразования, видим, что: H(S, j +1) = а(Р(5, J) - charts t)) + charts ;+n,) Иными словами, когда нам известно хэш-значение подстроки, начинающейся в пози- ции у, то мы можем узнать хэш-значение подстроки с позиции (/+ I), выполнив две операции умножения, одну операцию сложения и одну операцию вычитания. Для этого требуется постоянное время (значение а" 1 можно вычислить один раз, а затем исполь- зовать его для вычисления всех хэш-значений). Такой подход годится даже для вычис- ления H(Sj) mod М, где Мявляется достаточно большим простым числом, таким обра- зом ограничивая величину наших хэш-значений (самое большее М) даже для длинных строк-образцов. Алгоритм Рабина-Карпа является хорошим примером рандомизированного алгоритма (когда значение М выбирается каким-либо произвольным способом). Время исполне- ния алгоритма не обязательно будет равно О(п + т), т. к. нам может не повезти и мы будем регулярно получать конфликтные хэш-значения для ложных совпадений. Тем не менее, у нас неплохие шансы на удачу. Если хэш-функция возвращает значения равно- мерно в диапазоне от 0 до М- 1, то вероятность ложной коллизии хэш-значений долж- на быть ММ. Это вполне приемлемо: если М~ п, то должна быть только одна ложная коллизия на каждую строку, а если М~ п при к> 2, то существует очень высокая веро- ятность, что мы никогда не получим ложных коллизий.
112 Часть I. Практическая разработка алгоритмов 3.7.3. Выявление дубликатов с помощью хэширования Основополагающей идеей хэширования является представление большого объекта (будь то ключ, строка или подстрока) посредством одного числа. Цель состоит в том, чтобы представлять большой объект другим объектом, которым можно манипулиро- вать за постоянное время, при этом вероятность представления двух разных больших объектов одним и тем же числовым значением должна быть сравнительно невысокой. Кроме ускорения поиска, хэширование находит многие другие разнообразные хитро- умные применения. Я однажды слышал, как Уди Манбер (Udi Manber). в то время ру- ководитель исследовательских работ в Yahoo, рассказывал об алгоритмах, используе- мых в его компании. По его словам, наиболее важными алгоритмами, используемыми в Yahoo, были хэширование, хэширование и снова хэширование. Рассмотрим следующие задачи, имеющие красивые решения посредством хэширо- вания. ♦ Отличается ли данный документ от других в большом собрании документов? По- исковый механизм, который уже накопил громаднейшую базу данных из п доку- ментов, обработал еще одну веб-страницу. Как он определит, добавит ли она нечто новое в базу данных или только продублирует уже имеющуюся информацию? Для большого собрания документов будет весьма неэффективно сравнивать новый документ D со всеми имеющимися. По мы можем преобразовать документ D в це- лое число с помощью хэш-функции и сравнить его с хэш-кодами документов, уже имеющихся в базе данных. Только в случае коллизии хэш-значений документ D может быть возможным дубликатом. Так как вероятность ложных коллизий низка, то мы без больших затрат можем явно сравнить те несколько документов с одина- ковыми хэш-кодами. ♦ Не является ли данный документ плагиатом? Ленивый студент копирует документ из Интернета как свою курсовую работу, слегка изменив его в некоторых местах. "Интернет большой, — ухмыляется он. — Никто ничего не узнает". Это гораздо более трудная задача, чем предыдущая. Ехли в документ добавлен, уда- лен или изменен всего лишь один символ, то его хэш-код будет совершенно другим. Таким образом подход с использованием хэш-кода, применяемый для решения пре- дыдущей задачи, не годится для решения этой более общей задачи. Но мы можем создать хэш-таблицу всех перекрывающихся окон (подстрок) длиной и’ символов для всех документов в базе данных. Любое совпадение хэш-кодов озна- чает, что оба документа, вероятно, содержат одинаковую подстроку длиной w сим- волов, что можно исследовать более подробно. Значение w нужно выбрать доста- точно длинным, чтобы минимизировать вероятность случайного совпадения хэш- кодов. Самым большим недостатком этой схемы является то обстоятельство, что объем хэш-таблицы становится таким же большим, как и объем самих документов. Оста- вив для каждого документа небольшое, но удачно определенное подмножество хэш- кодов (состоящее, например, из чисел, кратных 100), мы с большой вероятностью сможем обнаруживать копии достаточно длинных строк.
Гпава 3 Структуры данных 113 ♦ Как убедиться в том, что файл не был изменен? На закрытых торгах каждая сторо- на подает свое предложение цены, которое неизвестно никакой другой стороне. В назначенное время цены оглашаются и сторона, предложившая наивысшую цену, объявляется выигравшей торги. Если какой-либо недобросовестный участник тор- гов знает предложения других сторон, то он может предложить цену, немного пре- вышающую максимальную цену, предложенную оппонентами, и выиграть торги. Одним из способов получить информацию о предложениях других сторон будет взлом компьютера, на котором хранится эта информация. Для предотвращения такого развития событий можно потребовать предоставление хэш-кода предложения до даты открытия предложений, с тем, чтобы само предло- жение было представлено после этой даты. После определения победителя его предложение хэшируется и полученное хэш-значение сравнивается с хэш-зна- чением, предоставленным ранее. Такие методы криптографического хэширования позволяют удостовериться, что предоставленный сегодня файл такой же, как и пер- воначальный. т. к. любые изменения в файле отразятся в измененном хэш-коде. Хотя наихудшие случаи любого алгоритма с хэшированием способны привести в смя- тение, при правильно подобранной хэш-функции мы можем с уверенностью ожидать хорошие результаты. Хэширование является основополагающей идеей рандомизиро- ванных алгоритмов, позволяющей получить линейное время исполнения алгоритма, по сравнению с 0(7zlog/?) или даже О(п") в наихудшем случае. 3.8. Специализированные структуры данных Все рассматриваемые до этого времени основные структуры данных представляют обобщенные наборы элементов, облегчающие доступ к данным. Эти структуры данных хорошо известны большинству программистов. Менее известны структуры данных для представления специализированных типов объектов, таких как точки в пространстве, строки и графы. Принципы создания этих структур данных те же, что и для основных типов структур. Имеется набор основных многократно исполняемых операций. Нам нужна структура данных, поддерживающая достаточно эффективное исполнение этих операций. Эти специализированные структуры данных являются важными для создания высокопро- изводительных алгоритмов для работы с графами и геометрическими объектами, по- этому нужно знать об их существовании. Далее приводится краткое описание несколь- ких таких специализированных структур данных. Подробно они обсуждаются при ис- пользовании в решениях в каталоге задач. ♦ Строки. Строки обычно представляются в виде массивов символов, возможно, с использованием специального символа для обозначения конца строки. Примером таких структур данных является суффиксное дерево/массив, в котором выполняется предварительная обработка строк для ускорения операции поиска подстрок. Под- робности см. в разделе 12.3. ♦ Геометрические структуры данных. Геометрические данные обычно представляют собой коллекцию точек и областей данных. Плоскость можно представить в виде многоугольника с границей, состоящей из цепочки отрезков. Многоугольник можно
114 Часть I. Практическая разработка алгоритмов представить с помощью массива точек (vb v,„ vj), где (v„ v,+ i) является сегмен- том границы многоугольника. В пространственных структурах данных, таких как kd-деревья, точки и области организованы по их геометрическому расположению для ускорения поиска. Подробности см. в разделе I2-.6. ♦ Графы. Графы обычно представляются посредством матриц или списков смежных вершин графа. Выбор конкретного типа представления может существенно повли- ять на структуру конечных алгоритмов для работы с графом, как рассказывается в главе бив разделе 12.4. ♦ Множества. Подмножества элементов обычно представляются посредством слова- ря для обеспечения быстроты поиска. Кроме того, могут использоваться двоичные векторы, которые представляют собой булевы массивы, в которых установленный z-й бит означает, что 1 является членом подмножества. Структуры данных для мани- пулирования множествами представлены в разделе 12.5. Структура данных, исполь- зуемая при работе с разбиениями множества, рассматривается в разделе 6.1.3. 3.9. История из жизни. Геном человека В геноме человека закодирована вся необходимая для создания человека информация. Проект по его расшифровке, называемый геномом человека, уже оказал огромное влияние на развитие современной медицины и молекулярной биологии. Алгористы также заинтересовались проектом генома человека, и на то были свои причины. ♦ Последовательности ДНК можно точно представить в виде строк алфавита из четы- рех символов— А, С, Т и G. Нужды биологов возродили интерес к старым алго- ритмическим задачам, таким как поиск подстрок (см. раздел 18.3), а также создали новые задачи, такие как поиск самой короткой общей подстроки (см. раздел 18.9). ♦ Последовательности ДНК являются очень длинными строками. Длина генома чело- века составляет приблизительно три миллиарда базовых пар (или символов). Такой большой размер входного экземпляра задачи означает, что асимптотический анализ * сложности алгоритма совершенно необходим в биологических приложениях. ♦ В геномику вливается достаточно большой объем денег, что вызывает у специали- стов в компьютерной области желание получить свою долю. Лично меня в вычислительной биологии заинтересовал метод, предложенный для сек- венирования ДНК и названный секвенированием путем гибридизации (sequencing by hybridization, SBH). Для его реализации набор проб (коротких нуклеотидных последо- вательностей) организуется в массив, создавая секвенирующий чип. Каждая из этих проб определяет наличие строки пробы в виде подстроки в целевой ДНК. Теперь мож- но выполнить секвенирование целевой ДНК на основе ограничений, определяющих, какие строки являются подстроками целевой ДНК. По данному набору всех подстрок длиной к строки 5 мы должны были идентифициро- вать все строки длиной 2к. возможные подстроки S. Предположим, нам известно, что строки АС, СА и СС являются двухсимвольными подстроками строки S. Возможно, что строка АССА является подстрокой строки 5, т. к. средняя подстрока является одним из вариантов. Но строка СААС не может быть подстрокой S, т. к. ее часть АА не является
Гпава 3 Структуры данных 115 подстрокой S. Так как строка S могла быть очень длинной, нам нужно было найти бы- стрый алгоритм для построения всех допустимых строк длиной 2k. Самым простым решением была бы конкатенация всех ()(н2) пар строк длиной к с по- следующей проверкой, что все к - I строк длиной к. переходящих границу конкатена- ции, действительно являются подстроками (рис. З.Ю). Т А Т С С Т Т А Т С G Т Т А Т С G Т Т А А С G Т Т А Т С С А Рис. 3.10. Конкатенация двух строк может входить в строку S. только если в нее входят все объединяемые строки Например, для подстрок АС, СА и СС возможны девять вариантов объединения — АСАС, АССА, АССС, СААС, САСА. САСС, ССАС. СССА и СССС. Из них можно ис- ключить только последовательность СААС. т. к. АА не является первоначальной под- строкой. Нам нужно было найти быстрый способ проверить, что к- 1 подстрок, переходящих границу конкатенации, были членами нашего словаря разрешенных строк длиной к. Время выполнения такого поиска зависит от структуры используемого словаря. Дво- ичное дерево поиска позволило бы найти правильную строку за 6>(log«) сравнений, состоящих в проверке, какая из двух строк длиной к символов встречается первой (по алфавиту). Общее время исполнения с применением такого двоичного дерева было бы равно O(k\ogti). Все это выглядело довольно хорошо, и мой аспирант. Димитрис Маргаритке (Dimitris Margaritis), создал алгоритм поиска с использованием двоичного дерева. Алгоритм вы- глядел прекрасно, пока мы не испытали его в действии. — Я выполнил программу на самом мощном компьютере в нашем отделе, но она слишком медленная. — пожаловался мне Димитрис. — Работает бесконечно долго на строках длиной всего лишь в 2 000 символов. Мы никогда не сможем дойти до 50 000. Мы исследовали нашу программу на профилировщике и обнаружили, что почти все время уходило на поиск в структуре данных. Это нас не удивило, т. к. эта операция вы- полнялась к - 1 раз для каждой из О(п) возможных последовательностей. Нам была нужна более быстрая структура данных, т. к. поиск был самой внутренней операцией из множества вложенных циклов. — А если попробовать хэш-таблицу? — предложил я. — Хэширование строки длиной в А: символов и поиск ее в нашей таблице должны занять время О(к). Это сократит вре- мя исполнения до (9(log/i), что будет довольно важно при п ~ 2 000. Димитрис принялся опять за работу и реализовал хэш-таблицу для нашего словаря. Как и в первый раз, программа выглядела прекрасно, пока мы не испытали ее в действии.
116 Часть I. Практическая разработка алгоритмов — Программа все еще работает слишком медленно.— опять пожаловался Димит- рис. — Конечно же, сейчас на строках длиной в 2 000 символов она выполняется раз в десять быстрее, так что мы можем дойти до строк длиной около 4 000 символов. Но мы все равно никогда не сможем дойти до 50 000. — Нам следовало ожидать этого.— размышлял я.— В конце концов, lg3(2000) = 11. Нам нужна более быстрая структура данных для поиска строк в словаре. — Но что может быть быстрее, чем хэш-таблица? — возразил Димитрис. — Чтобы найти строку длиной в к символов, нам нужно считать все эти к символов. Наша хэш- таблица уже дает нам время поиска О(к). — Конечно же. чтобы проверить первую подстроку, нужно выполнить к сравнений. Но. возможно, мы можем улучшить производительность на последующих проверках. Вспомни, как получаются наши запросы. Для конкатенации подстрок ABCD и EFGH мы сначала ищем в словаре части BCDE, а потом CDEF. Эти две подстроки отличают- ся всего лишь одним символом. Мы должны использовать это обстоятельство, чтобы каждая последующая проверка выполнялась за постоянное время. — С хэш-таблицей этого нельзя сделать, — заметил Димитрис. — Второй ключ не бу- дет находиться рядом с первым в таблице. Двоичное дерево поиска тоже не поможет. Так как ключи ABCD и BCDE различаются в первом символе, то они будут в разных частях дерева. — Но мы можем использовать для этого суффиксное дерево.— возразил я.— Суф- фиксное дерево содержит все суффиксы данного набора строк. Например, суффиксами АСАС строки будут {АСАС, САС, АС, С}. Вместе с суффиксами строки САСТ это даст нам суффиксное дерево, показанное на рис. 3.11. Рис. 3.11. Суффиксное дерево строк 1САС и САСТс указателем на суффикс строки АСАС Следуя по указателю из строки АСАС на ее самый длинный суффикс САС, мы попадем в правильное место для выполнения проверки строки САСТ на вхождение в наше мно- жество строк. Значит, нам нужно выполнить только одно сравнение символов. Суффиксные деревья являются удивительными структурами данных: более подробно они рассматриваются в разделе 12.3. Димитрис некоторое время изучал литературу
Гпава 3 Структуры данных 117 о них, после чего создал элегантный словарь на их основе. Как и прежде, программа выглядела прекрасно, пока мы не испытали ее в действии. — Теперь скорость работы программы нормальная, но ей не хватает памяти. — пожа- ловался Димитрис. — Программа создает путь длиной к для каждого суффикса длиной к, так что в дереве получается ©(»') узлов. Когда количество символов превышает 2000, происходит аварийное завершение программы. Мы никогда не сможем дойти до строки из 50 000 символов. Но я еще не был готов сдаваться. — Проблему с памятью можно решить, используя сжатые суффиксные деревья,— вспомнил я.— Вместо явного представления длинных путей мы можем ссылаться на- зад на исходную строку.— Сжатые суффиксные деревья всегда занимают линейный объем памяти (см. раздел 12.3). Димитрис пошел снова переделывать программу и реализовал структуру данных сжа- того суффиксного дерева. Вот теперь программа работала отличнейшим образом! Как видно из табл. 3.4, мы смогли без проблем выполнить нашу эмуляцию для строк дли- ной п = 65 536. Таблица 3.4. Время исполнения эмуляции секвенирования SBH с применением разных структур данных Длина СфОКИ Двоичное дерево Хэш-таблица Суффиксное дерево Сжатое суффиксi юс дерево 8 0.0 0.0 0.0 0.0 16 0.0 0,0 0,0 0.0 32 0,1 0.0 0.0 0.0 64 0.3 0.4 0,3 0.0 128 2.4 1.1 0.5 0 0 256 17.1 9.4 3.8 0.2 512 31.6 67,0 6.9 1.3 1,024 1 828.9 96,6 31.5 2.7 2.048 11 441.7 941.7 553.6 39.0 4.096 > 2 дней 5 246.7 нехватка 45.4 8.192 > 2 дней памяти 642.0 16.384 1 614.0 32,768 13 657.8 65.536 39 776.9 Полученные результаты показали, что интерактивное секвенирование SBH может быть очень эффективным. Это позволило нам заинтересовать биологов нашим методом. Но обеспечение реальных экспериментов в лаборатории стало очередной проверкой наше-
118 Часть I Практическая разработка алгоритмов го знания вычислительных алгоритмов. Описание нашей работы по решению этой за- дачи приводится в разделе 7 7 Выводы, которые можно сделать на основе информации из табл. 3.4, очевидны. Мы выделили одну многократно выполняемую операцию (поиск строки в словаре) и оптимизировали используемую в ней структуру данных. Мы начали с простой структу- ры (двоичное дерево поиска) в надежде, что этого будет достаточно, а когда оказалось, что нет, то выполнили профилирование программы, чтобы локализовать проблему. Когда и улучшенная структура словаря оказалась недостаточно эффективной, мы про- вели более глубокий анализ выполняемых запросов, чтобы можно было определить, какие дополнительные улучшения можно было сделать. Наконец, мы не сдались после нескольких неудачных попыток, а продолжали поиски подходящей структуры данных, пока не нашли такую, которая обеспечивала бы требуемый уровень производительно- сти. В разработке алгоритмов, как и в жизни, настойчивость обычно приносит резуль- таты. Замечания к главе Оптимизация производительности хэш-таблицы оказывается неожиданно сложной за- дачей для такой концептуально простой структуры данных. Подробности см. в книге [Кпи98]. Программа для оптимизации полосы треугольников, stripe, описывается в книге [ESV96], Использование методов хэширования для выявления плагиата рассматрива- ется в книге [SWA03]. Обзор алгоритмических вопросов в секвенировании ДНК методом гибридизации при- водится в книгах [СК94] и [PL94]. Доклад о нашей работе над интерактивным методом SBH, описанной в разделе 3.9, представлен в [MS95a]. 3.10. Упражнения Стеки, очереди и списки 1. [3] Распространенной задачей в компиляторах и текстовых редакторах является обеспе- чение правильного соотношения и вложенности открывающих и закрывающих скобок в строке. Например, в строке ((())())() скобки вложены правильно, а в строках )()( и ()) — нет. Разработайте алгоритм для проверки размещения скобок, который возвращает зна- чение ИСТИНА при правильном вложении скобок и ЛОЖЬ в противном случае. Чтобы решение было полностью засчитано, алгоритм должен определять положение первой не- правильной скобки в случае непарных или неправильно вложенных скобок. 2. [3] Напишите программу для изменения направления односвязного списка на противо- положное. Иными словами, после обработки программой все указатели должны быть пе- ревернуты в обратном направлении Алгоритм должен исполняться за линейное время. 3. [5] Мы видели, как использование динамических массивов позволяет увеличивать раз- мер массива, сохранив при этом постоянное амортизированное время исполнения. В этой задаче рассматривается вопрос произвольного расширения и уменьшения разме- ров динамических массивов.
Глава 3. Структуры данных 119 а) Разработайте стратегию экономии памяти, при которой размер массива, заполненного меньше чем на половину, уменьшается вдвое. Приведите пример последовательности вставок и удалений, при которой эта стратегия дает неприемлемое амортизированное время исполнения. б) Разработайте лучшую стратегию экономии памяти, чем предложенная выше, в кото- рой достигается постоянное амортизированное время исполнения для каждой операции удаления. Деревья и другие словарные структуры 4. [3] Разработайте словарную структуру данных с временем исполнения (7(1) в наихудшем случае для операций поиска, вставки и удаления. В данной задаче допускается, что эле- ментами множества являются целые числа из конечного множества I, 2, .... п. а инициа- лизация выполняется за время О(п). 5. [3] Определите долю накладных расходов (отношение объема памяти, занимаемого дан- ными, к общему объему памяти, отведенному под структуру) для каж дой из следующих реализаций двоичного дерева с п узлами: а) Все узлы содержат данные, два указателя на дочерние узлы и указатель на родитель- ский узел. Как поле данных, так и каждый указатель занимает четыре байта. б) Только листья содержат данные: внутренние узлы содержат два у казателя на дочерние узлы Поле данных занимает четыре байта, а каждый указатель - два байта. 6. [5] Опишите, как можно модифицировать любую сбалансированную структуру данных таким образом, чтобы время исполнения операций поиска, вставки, удаления, определе- ния минимума и максимума оставалось равным O(logn), а операции поиска предшест- вующего и следующего узла выполнялись за время 0(1)- Какие операции нужно моди- фицировать для этого? 7 [5] Допустим, имеется сбалансированный словарь, который поддерживает операции search, insert, delete, minimum, maximum, successor и predecessor, время исполнения каж- дой из которых равно (7(log/7). Как можно модифицировать операции вставки и удаления, чтобы их время исполнения оставалось равным O(log/?), но время исполнения операций определения максимума и минимума было 0(1). (Подсказка: думайте в терминах абст- рактных словарных операций и не теряйте время на указатели и т. п.) 8. [6] Разработайте структуру данных, поддерживающую следующие операции: • insertfx, Т) — вставляет элемент v в множество Т; • delete(k, Т) — удаляет наименьший k-й элемент из множества Т\ • member(x,T) — возвращает значение ИСТИНА тогда и только тогда, когда .г является членом Т. Время исполнения всех операций для набора входных данных из п элементов должно быть равным (7(log«). 9. [8] Операция конкатенации получает на входе два множества S, и .S2. где каждый элемент из S| меньше, чем любой элемент из S2, и соединяет их в одно множество. Разработайте алгоритм для конкатенации двух двоичных деревьев в одно. Время исполнения в наи- худшем случае должно быть равно О(1Г), где h — максимальная высота обоих деревьев.
120 Часть I. Практическая разработка алгоритмов Применение древовидных структур Ю. [5] В задаче разложения по контейнерам нам нужно разложить п объектов, каждый ве- сом от нуля до одного килограмма, в наименьшее количество контейнеров, максималь- ная емкость каждого из которых не больше одного килограмма. • Эвристический алгоритм типа "первый лучший" выглядит таким образом. Объекты рассматриваются в порядке, в котором они представлены. Каждый рассматриваемый объект помещается в частично заполненный контейнер, в котором после помещения данного объекта останется наименьший свободный объем. Если такого контейнера нет. объект помещается в новый (пустой) контейнер. Реализуйте алгоритм "первый лучший" с временем исполнения O(/7log«). Алгоритм принимает в качестве входа множество из п объектов и’ь w2, .... и возвращает количество требуемых контей- неров. • Разработайте и реализуйте алгоритм типа "первый худший”, в котором следующий объект помещается в такой частично заполненный контейнер, в котором после его помещения останется наибольший свободный объем. II. [5] Допустим, что для последовательности из п значений хь х2,—, х„ нам нужно быстро выполнять запросы, в которых, зная i и j, нужно найти наименьшее значение в подмно- жестве х,.....х;. а) Разработайте структуру данных объемом О(п~) и временем исполнения запросов 0(1). б) Разработайте структуру данных объемом О(н) и временем исполнения запросов O(log»). Чтобы решение было зачтено частично, структура данных может иметь объем O(/?logH) и обеспечивать выполнение запросов за время O(log/?). 12. [5] Допустим, что есть множество .S' из п чисел и черный ящик, который при вводе в не- го любой последовательности действительных чисел и целого числа к немедленно вы- дает ответ о наличии (или отсутствии) в данной последовательности подмножества, сумма которого равна точно к. Покажите, как можно использовать черный ящик О(п) раз. чтобы найти подмножество множества S, сумма членов которого равна точно к 13. [5] Пусть Л[1..н] — массив действительных чисел. Разработайте алгоритм для выполне- ния последовательности следующих операций: • add(i.y) — складывает значение у и z-й элемент массива; • partial-sum(i) — возвращает сумму первых i элементов массива, т. е. Л[у]. Нет никаких вставок или удалений, только изменяются значения чисел. Каждая опера- ция должна выполняться за <9(log«) шагов. Можно использовать дополнительный мас- сив размером п в качестве буфера. 14. [8] Расширьте структуру данных в предыдущей задаче, чтобы обеспечить поддержку операций вставки и удаления. Каждый элемент теперь имеет ключ и значение. Доступ к элементу осуществляется по его ключу. Операция сложения выполняется над значения- ми элементов. Операция partial-sum отличается от аналогичной операции из предыду- щей задачи. Используемые операции таковы: • add(k.y) — складывает значение у со значением элемента с ключом к; • insert(k.y) — вставляет новый элемент с ключом к и значением у;
Гпава 3 Структуры данных 121 • delete(k) — удаляет элемент с ключом А; • partial-sitm(k) - возвращает сумму всех текущих элементов множества, у которых значение ключа меньше чем у, т. е. £ х, . Время исполнения в наихудшем случае для любой последовательности О(л) операций должно оставаться таким же — (?(/7logn). 15. [8] Разработайте структуру данных, поддерживающую операции поиска, вставки и уда- ления целого числа А за время 0(1) (т е. за постоянное время, независимо от общего количества хранящихся в структуре целых чисел). Допустим, что 1 < X < п, и что суще- ствует т + и ячеек для хранения целых чисел, где т — наибольшее количество целых чисел, которые могут одновременно храниться в структуре. (Подсказка: используйте два массива ,4[1 и] и В|1..от].) Массивы нельзя инициализировать, т. к. для этого понадо- бится ()(т) или О(п) операций. Отсюда следует, что массивы изначально содержат не- предсказуемые значения, поэтому вам нужно проявлять особую аккуратность. Проекты по реализации 16. [5J Реализуйте разные варианты словарных структур данных, такие как связные списки, двоичные деревья, сбалансированные двоичные деревья поиска и хэш-таблицы. Экспе- риментальным путем сравните производительность этих структур данных в простом приложении, считывающим текстовый файл большого объема и отмечающим только один раз каждое встречающееся в нем слово. Это приложение можно эффективно реа- лизовать, создав словарь и вставляя в него каждое новое обнаруженное в тексте слово. Напишите краткий доклад с вашими выводами. 17. [5] Шифр Цезаря (см разде:/ 18.6) относится к классу простейших шифров. К сожале- нию, зашифрованные таким способом сообщения можно расшифровать, используя ста- тистические данные о частотном распределении букв латинского алфавита. Разработай- те программу для расшифровки достаточно длинных текстов, зашифрованных шифром Цезаря. Задачи, предлагаемые на собеседовании 18. [3] Каким методом вы бы воспользовались, чтобы найти слово в словаре? 19. [3] Представьте, что у вас полный шкаф рубашек. Как можно организовать рубашки, чтобы упростить их извлечение из шкафа? 20. [4] Напишите функцию поиска среднего узла односвязного списка. 21. [4] Напишите функцию сравнения двух двоичных деревьев. Идентичные двоичные де- ревья имеют одинаковую структуру и одинаковые значения в соответствующих узлах. 22. [4] Напишите программу для преобразования двоичного дерева поиска в связный спи- сок. 23. [4] Реализуйте алгоритм для изменения направления связного списка на обратное. Раз- работайте такой же алгоритм, но без использования рекурсии. 24. [5] Какой тип структуры данных будет наилучшим для хранения URL-адресов, посе- щенных поисковым механизмом? Создайте алгоритм для проверки, был ли ранее посе- щен данный URL-адрес: оптимизируйте алгоритм по времени и памяти.
122 Часть I. Практическая разработка алгоритмов 25. [4] У вас имеется строка поиска и журнал. Вам нужно сгенерировать все символы в строке поиска, выбрав их из журнала. Создайте эффективный алгоритм для определе- ния. содержит ли журнал все символы в строке поиска. 26. [4] Создайте алгоршм для изменения порядка слов в предложении на обратный, т. е. фраза "Меня зовут Крис" должна превратиться в "Крис зовут меня". Оптимизируйте ал- горитм по времени и памяти. 27. [5] Разработайте как можно быстрый алгоритм, использующий минимальный объем памяти, для определения, содержит ли связный список петлю. В положительном случае определите местонахождение петли. _8. [5] Имеется неотсортированный массив .V, содержащий п целых чисел. Определите множество М, содержащее п элементов, где Mt является произведением всех целых чи- сел из V. за исключением X,. Операцию деления применять нельзя, но можно использо- вать дополнительную память. (Подсказка: существуют решения с временем исполнения быстрее, чем О(п2).) 29. [6] Разработайте алгоритм, позволяющий найти на веб-странице наиболее часто встре- чающуюся упорядоченную пару слов (например. "New York"). Какой тип структуры данных вы бы использовали? Оптимизируйте алгоритм по времени и памяти. Задачи по программированию Эти задачи доступны на сайтах http://www.programming-challenges.com и http:// uva.onlinejudge.org. I. Jolly Jumpers. I Ю201/10038. 2. Crypt Kicker. 110204/843. 3. Where's Waldorf? 110302/10010. 4. Crypt Kicker II. 110304/850.
ГЛАВА 4 Сортировка и поиск Студенты, получающие специальность, связанную с вычислительными системами, изучают основные алгоритмы сортировки по крайней мере три раза: сначала во введе- нии в программирование, затем в курсе по структурам данных и, наконец, в курсе раз- работки и анализа алгоритмов. Почему сортировке уделяется так много внимания? На то есть ряд причин. ♦ Сортировка является базовым строительным блоком, на котором основаны многие алгоритмы. Понимание сортировки расширяет наши возможности при решении других задач. ♦ Большинство интересных идей, которые используются в разработке алгоритмов (в частности, метод "разделяй и властвуй", структуры данных и рандомизированные алгоритмы), возникло в контексте сортировки. ♦ Исторически сложилось так. что на сортировку уходит больше компьютерного вре- мени. чем на остальные задачи. Четверть циклов всех мэйнфреймов была использо- ваны для сортировки данных [Кпи98]. Сортировка по-прежнему остается самой распространенной практической алгоритмической задачей. ♦ Сортировка — самая изученная задача в теории вычислительных систем. Известны буквально десятки алгоритмов сортировки, большинство из которых имеют опреде- ленное преимущество над другими алгоритмами в определенных ситуациях. В этой главе мы обсудим сортировку, уделяя особое внимание тому, как ее можно применить для решения других задач. Мы рассмотрим подробное описание нескольких фундаментальных алгоритмов,— пирамидальной сортировки' (heapsort), сортировки слиянием (mergesort), быстрой сортировки (quicksort) и сортировки распределением (distribution sort), — представляющих собой примеры важных парадигм разработки алгоритмов. Задача сортировки также представлена в разделе 14.1. 4.1. Применение сортировки В этой главе мы рассмотрим несколько алгоритмов сортировки и оценим их времен- ную сложность. Важнейшая идея, которую я хочу донести до читателя, заключается в том, что существуют алгоритмы сортировки со временем исполнения (?(nlog/z). Это намного лучше, чем производительность О(п~), которую демонстрируют простые алго- ритмы сортировки на больших значениях п 'Другое на шаппе "сортировка кучей". —Прим, перев.
124 Часть I. Практическая разработка алгоритмов Рассмотрим следующую таблицу: п И-74 nlg/i 10 25 33 100 2 500 664 1 000 250 000 9 965 10 000 25 000 000 132 877 100 000 2 500 000 000 1 660 960 Алгоритмы квадратичной сложности могут быть приемлемыми еще при п = Ю ООО. но когда п > 100 000. сортировка за квадратичное время становится неприемлемой. Большинство важных задач можно свести к сортировке; в результате задачу, на первый взгляд требующую квадратичного алгоритма, можно решить с помощью интеллекту- альных алгоритмов с временной сложностью O(nlogM) Одним из важных методов раз- работки алгоритмов является использование сортировки в качестве базового конструк- тивного блока, т. к. после сортировки набора входных данных многие другие задачи становятся легко решаемыми. Рассмотрим несколько задач. ♦ Поиск. При условии, что входные данные отсортированы, двоичный поиск элемента в словаре занимает время 69(logn). Предварительная обработка данных для поиска, пожалуй, является наиболее важным применением сортировки. ♦ Поиск ближайшей пары. Как найти в множестве из п чисел два числа с наименьшей разностью между ними? После сортировки входного набора такие числа должны находиться рядом в упорядоченной последовательности. Поиск этих чисел занимает линейное время, а общее время, включая сортировку, составляет 6?(wlog/?). ♦ Определение уникальности элементов. Как определить, имеются ли дубликаты в множестве из п элементов? Это частный случай общей задачи поиска ближайшей пары элементов, в котором нам нужно найти два элемента, разность между которы- ми равна нулю. Данная задача решается так же, как и предыдущая — сначала вы- полняется сортировка входного множества, после чего отсортированная последова- тельность перебирается за линейное время, которое требуется для проверки всех смежных пар. ♦ Частотное распределение. Задача состоит в определении самого часто встречаю- щегося элемента в множестве из п элементов. Если элементы множества отсортиро- ваны, то их можно просто перебрать слева направо, подсчитывая, сколько раз встречается каждый элемент, т. к. при сортировке все одинаковые элементы будут размещены рядом друг с другом. Для подсчета количества вхождений произвольного элемента к в некоторое множе- ство выполняется двоичный поиск этого элемента в отсортированном массиве клю- чей. После нахождения требуемого элемента в массиве выполняется сканирование влево от этого элемента, пока не будет найден первый элемент, отличный от к. По- том эта же процедура повторяется вправо от первоначальной точки. Общее время
Гпава 4 Сортировка и поиск 125 определения вхождений этим методом равно O(log« + с), где с — количество вхож- дений элемента к. Еще лучшее время — O(log«)— можно получить, определив по- средством двоичного поиска местонахождение как элемента к- г, так и элемента к+ е (где Е — сколь угодно малое) и затем вычислив разницу между этими двумя позициями. ♦ Выбор элемента. Как найти к-й по величине элемент массива? Если элементы мас- сива отсортированы, то А-й по величине элемент можно найти за линейное время, просто прочитав Л-ю ячейку массива. В частности, средний элемент находится в от- сортированном массиве в (и/2)-й позиции. ♦ Выпуклая оболочка. Как определить многоугольник с наименьшей поверхностью, содержащий множество точек п в двух измерениях? Выпуклую оболочку можно сравнить с резиновой нитью, натянутой вокруг точек в плоскости. Резиновая нить сжимается вокруг самых выступающих точек, образуя многоугольник (рис. 4.1, а). Рис. 4.1. Создание выпуклой оболочки: резиновой нитью (а), вставкой точек слева направо (б) Выпуклая оболочка дает хорошее представление о размещении точек и является одним из важных конструктивных блоков для создания более сложных геометриче- ских алгоритмов, рассматриваемых в разделе 17.2. Но каким образом можно использовать сортировку для создания выпуклой оболоч- ки? После того как точки отсортированы по х-координате, их можно вставлять в оболочку слева направо. Так как самая правая точка всегда расположена на пери- метре. то мы знаем, что она войдет в оболочку. После добавления этой новой самой правой точки остальные могут оказаться удаленными, но мы легко идентифицируем эти точки, т. к. они находятся внутри многоугольника, полученного в результате до- бавления новой точки (рис. 4.1, о). Эти точки будут соседями предыдущей встав- ленной точки, так что их можно будет с легкостью найти и удалить. После сорти- ровки общее время исполнения — линейное. Хотя некоторые из этих задач (например, выбор элемента) можно решить за линейное время с помощью более сложных алгоритмов, сортировка предоставляет самые про- стые решения. Время исполнения сортировки оказывается узким местом лишь в не- многих приложениях, к тому же в этом случае ее почти всегда можно заменить на бо- лее интеллектуальные алгоритмы. Поэтому никогда не бойтесь потратить время на сортировку, при условии, что используется эффективная процедура сортировки.
126 Часть I. Практическая разработка алгоритмов Подведение итогов Сортировка является центральной частью многих алгоритмов. Сортировка данных долж- на быть одним из первых шагов, предпринимаемых любым разработчиком алгоритмов с целью повышения эффективности разрабатываемого решения. Остановка для размышлений. Поиск пересечения множеств ЗАДАЧА. Предоставить эффективный алгоритм для определения, являются ли два множества (мощностью т и п соответственно) непересекающимися. Проанализировать сложность алгоритма в наихудшем случае относительно т и п. рассматривая случай, когда т значительно меньше, чем п. Решение. В голову приходят, по крайней мере, три решения, каждое из которых явля- ется разновидностью сортировки и поиска. ♦ Сортируется только большое множество. Большое множество можно отсортиро- вать за время OMIogn). Теперь для каждого из т элементов меньшего множества выполняется поиск на его наличие в большом множестве. Общее время исполнения будет равно О((н + z»)logH)- ♦ Сортируется только малое множество. Меньшее множество сортируется за время (J(m\ogm). Теперь для каждого из п элементов большего множества выполняется поиск на его наличие в меньшем множестве. Общее время исполнения будет равно О((п + /n)log«). ♦ Сортируются оба множества. Когда оба множества отсортированы, то для опре- деления общего элемента больше не требуется выполнять двоичный поиск. Вместо этого мы сравниваем наименьшие элементы в обеих отсортированных последова- тельностях и. если они различаются, удаляем меньший элемент. Операция повторя- ется рекурсивно на все уменьшающихся последовательностях, занимая линейное время, а общее время исполнения (включая сортировку) равно O(»Iogn + wzlogw + Т п + «?). Какой же из этих методов самый быстрый? Ясно, что сортировка меньшего множества выполняется быстрее, чем сортировка большего множества, т. к. logm < log/i при т < и. Аналогично, (п + /»)logw должно быть асимптотически меньше, чем »log п, т. к. п + т < < 2п при т < п. Следовательно, метод сортировки меньшего множества является са- мым лучшим из этих трех. Обратите внимание, что при константном значении т время исполнения будет линейным. Кроме этого, ожидаемое линейное время можно получить посредством хэширова- ния — создается хэш-таблица, содержащая элементы обоих множеств, после чего вы- полняется проверка, что конфликты в корзине являются результатом хэширования идентичных элементов. На практике это решение может быть самым лучшим. 4.2. Практические аспекты сортировки Мы уже видели, что сортировка часто применяется в различных алгоритмах. Теперь обсудим несколько эффективных алгоритмов сортировки. Это делает актуальным во- прос — в каком порядке нужно сортировать элементы?
Гпава 4. Сортировка и поиск 127 Ответ на этот вопрос зависит от конкретной задачи. Более того, во внимание должен приниматься ряд соображений. ♦ Сортировать в возрастающем или убывающем порядке? Набор ключей 5 отсорти- рован в возрастающем порядке, если S, < S, i для всех 1 < / < п, и в убывающем по- рядке, если S, > S, । для всех 1 < i < п. Для разных приложений требуется разный порядок сортировки. ♦ Сортировать только ключ wiu все поля записи? При сортировке набора данных необходимо сохранять целостность сложных записей. Например, элементы списка рассылки, содержащего имена, адреса и телефонные номера, можно отсортировать по полю имен, но при этом необходимо сохранять связь между полем имен и дру- гими полями. Таким образом, для сложной записи нам нужно указать, какое поле является ключевым, а также понимать полную структуру записей. ♦ Что делать в случае совпадения ключей? Элементы с одинаковыми значениями ключей будут сгруппированы вместе при любом порядке сортировки, но иногда имеет значение порядок размещения этих элементов относительно друг друга. До- пустим. что энциклопедия содержит записи, как для Майкла Джордана баскетболи- ста, так и для Майкла Джордана специалиста в области статистики. Какая из этих записей должна быть первой? Для решения вопроса одинаковых первичных ключей придется прибегнуть к использованию вторичного ключа, которым может оказать- ся, например, размер статьи. Иногда при сортировке необходимо сохранить первоначальный порядок элементов с одинаковыми ключами. Алгоритмы сортировки, которые автоматически выпол- няют это требование, называются устойчивыми (stable). К сожалению, очень немно- гие быстрые алгоритмы являются устойчивыми. Стабильность в любых алгоритмах сортировки можно обеспечить, указав первоначальное положение записи в качестве вторичного ключа. Конечно же. в случае совпадения ключей можно ничего не делать, а просто позво- лить алгоритму разместить их там, где он сочтет нужным. Но имейте в виду, что производительность некоторых эффективных алгоритмов сортировки (например, быстрой сортировки) может ухудшиться до квадратичной, если в них явно не пре- дусмотреть возможность обработки большого количества одинаковых значений ключей. ♦ Как поступать с нечисловыми данными? Сортировка текстовых строк называется упорядочиванием по алфавиту. В библиотеках применяются очень подробные и сложные правила касательно последовательности упорядочения букв и знаков пунктуации. Это вызвано необходимостью принимать такие решения, как, напри- мер. являются ли ключи Skiena и skiena одинаковыми, или как упорядочить записи Brown-Williams, Brown America и Brown, John? Правильным решением таких вопросов в алгоритме сортировки будет использование функции попарного сравнения элементов, специфической для конкретного приложения. Такая функция принимает в качестве входа указатели на элементы а и b и возвращает "<", если а < Ь, ">". если а > Ь, и " = ”, если а = Ь. Сводя попарное упорядочивание к подобной функции, мы можем реализовывать алго- ритмы сортировки, не обращая внимания на такие детали. Мы просто передаем функ-
128 Часть I. Практическая разработка алгоритмов цию сравнения процедуре сортировки в качестве аргумента. Любой серьезный язык программирования содержит встроенную в виде библиотечной функции процедуру сортировки. Почти во всех случаях использование этой функции предпочтительнее написания своей собственной процедуры сортировки. Например, стандартная библио- тека языка С содержит функцию сортировки qsort: ((include stdlib.h> void qs< rt(void *base, size_t nel, size_t width, jnt (*compare) (const void +, const void *)); При использовании функции qsort важно понимать назначение ее аргументов. Функ- ция сортирует первые nel элементов массива, длина которых составляет width байт (на сам массив указывает переменная base). Таким образом, мы можем сортировать масси- вы однобайтовых символов, четырехбайтовых целых чисел или 100-байтовых записей, изменяя лишь значение переменной width. Конечный требуемый порядок отсортированной последовательности определяется функцией compare. Эта функция принимает в качестве аргументов указатели на два элемента размером width и возвращает отрицательное число, если в отсортированной последовательности первый элемент должен быть перед вторым, положительное чис- ло, если наоборот, и ноль, если элементы одинаковые. Код функции для сравнения це- лых чисел в возрастающей последовательности показан в листинге 4.1. Листинг 4 1. Реализация функции сравнения int intcump are ( int "i, int *j) < : f • return (1) , if ”. *j) return (-1) ; return Эту функцию сравнения можно использовать при сортировке массива а, в котором за- няты первые п ячеек. Функция сортировки вызывается таким образом: qsort(a, n, sizeof(int), intcompare); Название функции сортировки qsort дает основания полагать, что в ней реализован алгоритм быстрой сортировки quicksort, но это обстоятельство обычно не имеет ника- кого значения для пользователя. 4.3. Пирамидальная сортировка Тема сортировки представляет собой естественную лабораторию для изучения прин- ципов разработки алгоритмов, т. к. использование различных методов влечет за собой создание интересных алгоритмов сортировки. В следующих нескольких разделах представляются методы разработки алгоритмов, порожденные определенными алго- ритмами сортировки. Внимательный читатель должен здесь спросить, почему мы рассматриваем стандарт- ные методы сортировки, когда в конце предыдущего раздела была дана рекомендация
Гпава 4 Сортировка и поиск 129 не реализовывать их, а вместо этого использовать встроенные библиотечные функции соргировки конкретного языка программирования. Ответ на этот вопрос состоит в том, что применяемые для разработки этих алгоритмов методы также являются очень важ- ными для разработки алгоритмов решения других задач, с которыми вам, скорее всего, придется столкнуться. Мы начнем с разработки структур данных, т. к. одним из методов существенного улучшения производительности алгоритмов сортировки является использование соот- ветствующих задаче структур данных. Сортировка методом выбора представляет собой простой алгоритм, который в цикле извлекает наименьший оставшийся элемент из не- отсортированной части набора входных данных. На псевдокоде этот алгоритм можно выразить таким образом: Selectionsort (А) for 1 1 to n do Sort[r] = Find-Minimum from A Delete-Minimum from A Return(Sorti Реализация алгоритма на языке С приводится в листинге 2.1. Здесь массив входных данных разделяется на отсортированную и неотсортированную части. Поиск наи- меньшего элемента выполняется сканированием элементов в неотсортированной части массива, что занимает линейное время. Найденный наименьший элемент меняется мес- тами с элементом / массива, после чего цикл повторяется. Всего выполняется п итера- ций,' в каждой из которых, в среднем, и/2 шагов, а общее время исполнения равно Может быть, существует лучшая структура данных? После того как местонахождение элемента в неотсортированном массиве определено, его удаление занимает время 0(1), но чтобы найти наименьший элемент, нужно время О(п). Эти операции могут исполь- зовать очереди с приоритетами. Что будет, если заменить текущую структуру данных структурой с реализацией очереди с приоритетами, такой как пирамида или сбаланси- рованное двоичное дерево? Теперь, вместо О(п). операции внутри цикла исполняются за время O(Iog/?). Использование такой реализации очереди с приоритетами ускоряет сортировку методом выбора с О(п2) до <9(»log/j). Распространенное название этого алгоритма— пирамидальная сортировка— не от- ражает его механизма, но пирамидальная сортировка в действительности есть нечто иное, как реализация сортировки методом выбора с применением удачной структуры данных. 4.3.1. Пирамиды Пирамида представляет собой простую и элегантную структуру данных, поддержи- вающую такие операции очередей с приоритетами, как вставка и поиск наименьшего элемента. Принцип работы пирамиды основан на поддержании порядка "частичной отсортированности" в заданном наборе элементов. Такой порядок слабее, чем состоя- ние полностью отсортированного набора (что обеспечивает эффективность сопровож- дения), но сильнее, чем произвольный порядок (что обеспечивает быстрый поиск наи- меньшего элемента). 5 Зак 3741
130 Часть I. Практическая разработка алгоритмов Властные отношения в любой организации с иерархической структурой отображаются деревом, каждый узел которого представляет сотрудника, а ребро (х, jr) означает, чтох непосредственно управляет у (доминирует над ним). Сотрудник, отображаемый корне- вым узлом, находится наверху организационной пирамиды. Подобным образом пирамида определяется как двоичное дерево, в котором значение ключа каждого узла доминирует над значением ключа каждого из его потомков. В не- убывающей бинарной пирамиде (min-heap) доминирующим над своими потомками яв- ляется узел с меньшим значением ключа, чем значения ключей его потомков; а в не- возрастающей бинарной пирамиде (max-heap) доминирующим является узел со значе- нием ключа большим, чем значения ключей его потомков. На рис. 4.2, а показана неубывающая пирамида дат знаменательных событий в истории Соединенных Штатов. i 2 3 4 5 6 7 8 9 10 1492 1783 1776 1804 1865 1945 1963 1918 2001 1941 Рис. 4.2. Пирамидальное дерево дат важных событий в американской истории (а) и соответствующий массив (6) В наиболее естественной реализации этого двоичного дерева каждый ключ сохранялся бы в узле с указателями на двух его потомков. Как и в случае с двоичными деревьями поиска, объем памяти, занимаемый указателями, может быстро превысить объем памя- ти, занимаемый ключами, которые интересуют нас в первую очередь. Однако пирамида является настолько удачной структурой данных, что позволяет нам реализовать двоичные деревья без помощи указателей. Данные в ней сохраняются в виде массива ключей, а позиции ключей используются в роли неявных указателей. Корневой узел дерева сохраняется в первой ячейке массива, а его левый и правый по- томки — во второй и третьей соответственно. В общем, мы сохраняем 2Z ключей уров- ня / полного двоичного дерева в направлении слева направо в позициях 2/-1 до 2;-1, как показано на рис. 4.2, б. Для простоты мы полагаем, что нумерация ячеек массива начинается с единицы. typedef struct { item_type q[PQ_SIZE+l]; /* Тело очереди ★/ int n; /* Количество элементов очереди */ } priority_queue; Это представление весьма удачно благодаря легкости, с которой определяется место- нахождение родителя и потомка ключа, расположенного в позиции к. Левый потомок
Гпава 4. Сортировка и поиск 131 ключа к находится в позиции 2к, правый— в позиции 2к + I, а родительский ключ находится в позиции |_и/2_|. Таким образом, по дереву можно перемещаться без ис- пользования указателей (листинг 4.2). Листинг 4.2. Код для работы с пирамидой pq_parent (int n) I if (n == 1) return(-l); else return!(int) n/2); /* Явно вычисляем floor(n/2) */ pq_young_child(int n) return (2 * n); Итак, мы можем сохранить любое двоичное дерево в массиве, не используя указатели. Но нет ли здесь какого-то подвоха? Да, есть. Допустим, что наше дерево высотой h разреженное, т. е. количество его узлов n < 2". Тем не менее, нам нужно выделить ме- сто в структуре для всех отсутствующих узлов, т. к. необходимо представить полное двоичное дерево, чтобы сохранить позиционные отношения между узлами. Таким образом, чтобы не расходовать понапрасну память, мы не можем допустить на- личие пробелов в нашем дереве, иными словами, каждый уровень должен быть запол- нен до предела. Только последний уровень может быть заполнен частично. Упаковы- вая элементы последнего уровня как можно плотнее влево, мы можем представить де- рево из п ключей, используя точно и элементов массива. Если не придерживаться этих структурных ограничений, то для представления этого же количества элементов нам может потребоваться массив размером 2л. Так как в пирамидальном дереве всегда за- полняются все уровни, кроме последнего, высота пирамиды из п элементов логариф- h мическая: h = | 1gиJ, т. к. ^2' = 2/,+| -1 > и. <=о Хотя такое неявное представление двоичных деревьев позволяет сэкономить память, оно менее гибкое, чем представление с использованием указателей. Сохранение де- ревьев с произвольной топологией влечет за собой большой расход памяти. Кроме это- го. размещение поддеревьев можно изменять, только явно перемещая все элементы поддерева, а не обновляя один-единственный указатель, как делается в случае деревьев с указателями. Такое отсутствие гибкости объясняет, почему эту идею нельзя исполь- зовать для представления двоичных деревьев поиска. Впрочем, для пирамид она впол- не подходит. Остановка для размышлений. Поиск в пирамиде ЗАДАЧА. Возможен ли эффективный поиск определенного ключа в пирамиде? Решение. Невозможен. В пирамиде нельзя использовать двоичный поиск, т. к. пира- мида не является двоичным деревом. Нам почти ничего не известно об относительном
132 Часть I. Практическая разработка алгоритмов размещении п/2 листьев пирамиды, во всяком случае, ничего, что помогло бы избежать линейного поиска в этих листьях. 4.3.2. Создание пирамиды Пирамиду можно создать пошагово, вставляя каждый новый элемент в самую левую свободную ячейку массива, а именно в (п + 1)-ю ячейку предыдущей пирамиды из п элементов. Таким образом обеспечивается сбалансированная форма пирамидального дерева, но необязательно соблюдается порядок доминирования ключей. Новый ключ может быть меньшим, чем его предшественник в неубывающей бинарной пирамиде, или большим, чем его предшественник в невозрастающей бинарной пирамиде. Эта задача решается за счет обмена местами такого элемента с его родителем. Таким образом, прежний родитель теперь занимает должное место в иерархии доминирова- ния. а порядок доминирования другого дочернего узла прежнего родителя продолжает оставаться правильным, т. к. теперь над ним находится элемент с большим уровнем доминирования, чем у его предыдущего родителя. Новый элемент теперь в несколько лучшем положении, но может продолжать доминировать над своим новым родителем. Тогда мы повторяем процедуру необходимое количество раз на более высоком уровне, перемещая новый ключ пузырьковым методом на должное место в иерархии. Так как на каждом шаге корень поддерева заменяется элементом с большим уровнем домини- рования, то порядок доминирования сохраняется во всех других частях пирамиды. Код для вставки нового элемента в пирамиду показан в листинге 4.3. Листинг 4.3. Вставка элемента в пирамиду pq_insert(priority_queue *q, item_type x) { if (q->n >= PQ_SIZE) printf("Warning: priority queue overflow insert x=%d\n",x); else { q->n = (q->n) + 1; q->q[ q->n ] = x; bubble_up(q, q->n); } } bubble_up(priority_queue *q, int p) { if (pq_parent(p) == -1) return; /* Корень */ if (q->q[pq_parent(p)] >q->q[p]) { pq_swap(q,p,pq_parent(p)); bubble_up(q, pq_parent(p)); } } На каждом уровне процесс обмена элементов местами занимает постоянное время. Так- как высота пирамиды из п элементов равна [_lgnj, то каждая вставка занимает, самое большее, время O(log«). Таким образом, первоначальную пирамиду из п элементов
Гпава 4. Сортировка и поиск 133 можно создать посредством и таких вставок за время (9(«logn). Соответствующий код приведен в листинге 4.4. Листинг 4.4. Создание пирамиды повторяющимися вставками pq_init 'priority_queue *q) q->n = 0; ”iake_heap(priority_queue *q, item_type s[], int n) int i; /* Счетчик */ pq_init (q) ; for (i=0; i<n;i++) pq_insert(q, s[i] ) ; 4.3.3. Наименьший элемент пирамиды Осталось рассмотреть такие операции, как поиск и удаление корневого элемента пира- миды. Поиск не составляет никакого труда, т. к. верхний элемент пирамиды находится в первой ячейке массива. После удаления элемента остается пустая ячейка в массиве, которую можно заполнить, перемещая в нее элемент из самого правого листа (который размещен в и-й ячейке массива). Таким образом восстанавливается форма дерева, но (как и в случае со вставкой) значе- ние корневого узла может больше не удовлетворять иерархическим требованиям пира- миды. Более того, над этим новым корнем могут доминировать оба его потомка. Ко- рень данной неубывающей бинарной пирамиды должен быть наименьшим из этих трех элементов, т. е. данного корня и его двух потомков. Если текущий корень доминирует над своими потомками, значит, порядок пирамиды восстановлен. В противном случае доминирующий потомок меняется местами с корнем и проблема передается на один уровень вниз. Проблемный элемент продвигается вниз пузырьковым методом до тех пор, пока он не начнет доминировать над всеми своими потомками, возможно, став листом. Данная операция "просачивания" элемента вниз также называется восстановлением пирамиды (heapify), т. к. она сливает вместе две пирамиды (поддеревья под первоначальным кор- нем) с новым ключом. Код для удаления наименьшего элемента пирамиды показан в листинге 4 5. Листинг 4.5 Удаление наименьшего элемента пирамиды .tem_type extract_min (priority_queue *q) int min = -1; /* Минимальное значение */ if (q->n <=• 0) printf("Warning: empty priority queue.\n");
134 Часть I Практическая разработка алгоритмов else { min = q—>q[l] ; q->q[l] = q->q[q->n] ; q->n = q->n — 1; bubble_down(q,1); } return(min); ) bubble_down(priorityqueue *q, int p) { int с; /* Индекс потомка */ int i; /* Счетчик */ int min_index; /* Индекс наименьшего потомка */ с = pq_young_child(p); min_index = p; for (i=0; i<=l; i++) if ((c+i) <= q->n) { if (q->q[min index] > q->q[c+i]) min_index = c+i; 1 if (min_index != p) ( pq_swap(q,p,min index); bubbledown(q, min_index); } ) Для достижения позиции листа требуется [jg^J исполнений процедуры bubbie_down, каждое из которых исполняется за линейное время. Таким образом, удаление корня занимает время (9(log«). Обмен местами наибольшего элемента с последним элементом и многократный вызов процедуры восстановления пирамиды дает нам алгоритм пирамидальной сортировки с временем исполнения O(nlog«) (листинг 4.6). Листинг 4.6. Алгоритм пирамидальной сортировки .А,*....:.....-..V.... ...............___ heapsort(item_type s[],int n) ( int i; /★ Счетик */ priority_queue q; /* Память для пирамидальной сортировки */ make_heap(sq,s,n); for (i=0; icn; i++) s[i] = extract_min(&q); Пирамидальная сортировка является замечательным алгоритмом. Его легко реализо- вать, что подтверждает полный код, представленный в листингах 4 4-4.6. Время ис- полнения этого алгоритма в наихудшем случае равно O(Hlog/r), что является наилуч- шим временем исполнения, которое можно ожидать от любого алгоритма сортировки. Сортировка выполняется "на месте", что означает, что используется только память, содержащая массив с сортируемыми элементами. Хотя на практике другие алгоритмы
Гпава 4. Сортировка и поиск 135 сортировки могут оказаться немного быстрее, вы не ошибетесь, если предпочтете этот для сортировки данных в оперативной памяти компьютера. Очереди с приоритетами являются очень полезными структурами данных. Вспомните, как мы их использовали на практике (см. раздел 3.6). Кроме этого, в разделе 12.2 дает- ся полный набор реализаций очереди с приоритетами. 4.3.4. Быстрый способ создания пирамиды (*) Как мы видели, пирамиду из п элементов можно создать за время O(«log«) методом пошаговой вставки элементов. Удивительно, но пирамиду можно создать еще быстрее, используя нашу процедуру bubble down (см. листинг 4.5) и выполнив некоторый анализ. Допустим, что мы упакуем п ключей, предназначенных для построения пирамиды, в первые п элементов массива очереди с приоритетами. Форма пирамиды будет пра- вильной, но иерархия доминирования будет полностью нарушена. Как можно испра- вить это положение? Рассмотрим массив в обратном порядке, начиная с последней (n-й) ячейки. Эта ячейка является листом дерева и поэтому доминирует над своими несуществующими потом- ками. То же самое верно и для последних и/2 ячеек массива, т. к. все они являются листьями. Продолжая двигаться по массиву в обратном направлении, мы, наконец, дойдем до внутреннего узла, имеющего потомков. Этот элемент может не доминиро- вать над своими потомками, но эти потомки представляют правильно построенные (хоть и небольшого размера) пирамиды. Это как раз та ситуация, для исправления которой предназначена процедура bubble_down,— восстановление правильности иерархии произвольного элемента пира- миды, находящегося сверху двух меньших пирамид. Таким образом мы можем создать пирамиду, выполнив н/2 вызовов процедуры bubble down, как показано в листинге 4.7. Листинг 4.7. Алгоритм быстрого создания пирамиды ........ ................?...... ................------------.... make_heap (priority_queue *q, item_type s[], int n) ( int i; /* Счетчик */ q->n = n; for (i=0; i<n; i++) q->q[i+l] = s[i]; for (i=q->n; i>=l; i—) bubble_down(q,i); Умножив количество вызовов (и) процедуры bubble_down на верхний предел сложности каждой операции (67(logw)), мы получим время исполнения O(»logn). Но это ничуть не быстрее, чем время исполнения алгоритма пошаговой вставки, описанного ранее. Но обратите внимание, что это действительно верхний предел, т. к. фактически только последняя вставка требует [igwj шагов. Вспомните, что время исполнения процедуры oubbie_down пропорционально высоте пирамид, слияние которых она осуществляет. Большинство из этих пирамид очень небольшого размера. В полном двоичном дереве из п узлов п/2 узлов являются листьями (т. е. имеют высоту 0), и/4 узлов имеют высо-
136 Часть I. Практическая разработка алгоритмов ту I. /?/8 узлов имеют высоту 2. и т. д.). В общем, имеется самое большее | п ' 2/,+| | уз- лов высотой /?. соответственно, затраты на создание пирамиды составляют: L'8"J г L’b"J [и / 2Л+| | /? < п У /г / 2;' < 2п Л=0 h=0 Так как эта сумма, строго говоря, не является геометрической прогрессией, мы не мо- жем выполнить обычную операцию тождества. Но можно быть уверенным, что несу- щественный вклад числителя (/г) полностью перекроезся значением знаменателя (2Л). Таким образом, функция быстро приближается к линейной. Имеет ли значение тот факт, что мы можем создать пирамиду за линейное время вместо времени O(wlogn)? Как правило, нет. Время создания не доминирует над слож- ностью пирамидальной сортировки, поэтому улучшение времени создания не улучшает его производительность в наихудшем случае. Тем не менее, перед нами убедительная демонстрация важности внимательного анализа и возможности получения дополни- тельных бонусов от сходимости геометрической прогрессии. Остановка для размышлений. Расположение элемента в пирамиде ЗАДАЧА. Для данной пирамиды на основе массива из п элементов и действительного числах разработайте эффективный метод определения, является ли А-й элемент пира- миды большим или равным х. Независимо от размера пирамиды время исполнения ал- горитма в наихудшем случае должно быть равным О(к). Подсказка: находить £-й наи- меньший элемент не нужно; требуется только определить его взаимосвязь сх. Решение. Существуют, по крайней мере, два разных подхода, дающих правильные, но неэффективные алгоритмы для решения этой задачи. 1. Процедура выборки наименьшего значения вызывается к раз, и каждое из выбран- ных значений проверяется, меньше ли оно, чем значение х. Таким образом явно сортируются первые к элементов, что предоставляет нам больше информации, чем требует постановка задачи, но для этого требуется время О(Иоця). 2. Наименьший к-й элемент не может находиться глубже, чем на к-м уровне пирами- ды. т. к. путь от него до корня должен проходить через элементы с убывающими значениями. Таким образом, мы можем просмотреть все элементы первых к уров- ней пирамиды и сосчитать, сколько из них меньше, чем х; просмотр прекращается, когда мы либо нашли к таких элементов, либо исчерпали все элементы. В то время как этот алгоритм дает правильное решение задачи, он исполняется за время O(min(«, 2 )), поскольку к верхних уровней насчитывают 2А элементов. Решение с временем исполнения О(Аг) должно проверить только к элементов мень- ших. чем х, плюс не более О(к) элементов больших, чем х. Такое решение в виде рекурсивной процедуры, вызывающейся для корневого элемента с параметрами i = 1 и count = к, приводится в листинге 4.8. Листинг 4.8. Сравнение k-го элемента с числом х .. ......... ................. . ...... ...... ..... kWU.viuWUU/l .kW/а. ..'.J.lkV....... ......I ... int heap_compare (priority_queue *q, int i, int count, int x) { if ((count <= 0) || (i > q->n) return(count):
Глава 4. Сортировка и поиск 137 if (q-'q[i] х) { count = heap_compare(q, pq_young_child(i), count-1, x) ; count - heap_compare(q, pq_young_child(i)+1, count, x); return (count) ; Если корневой элемент неубывающей бинарной пирамиды меньше чем х, тогда в ос- тавшейся пирамиде не может быть элементов меньше чем х, т. к. по определению этого типа пирамиды корневой элемент должен быть наименьшим. В противном случае про- цедура просматривает потомки всех узлов со значением меньшим, чем х до тех пор, пока либо не найдет Этаких узлов (и в этом случае возвращается 0), либо не исчерпает все узлы (и тогда возвращается положительное значение). Таким образом, процедура найдет достаточное количест во подходящих элементов, при условии, что они имеются в пирамиде. Несколько времени потребуется? Процедура проверяет потомков только тех узлов, чье значение меньше, чем х, и общее количество таких узлов равно, самое большее, к. Для каждого из этих узлов проверяются, самое большее, два потомка, следовательно, количество проверяемых узлов равно, самое большее, 2к, а общее время исполнения равно О(к). 4.3.5. Сортировка вставками Теперь рассмотрим другой подход к сортировке, опирающийся на использование эф- фективных структур данных. Выбираем произвольный элемент из неотсортированного списка и вставляем его в должную позицию в отсортированном списке. Псевдокод это- го метода сортировки показан в листинге 4.9. Листинг 4.9. Сортировка вставками [nsertionSort (А) А[0] = -~ for , to n do J = 1 while [A[j] < A[j — 1]) do swap(A[j] ,A[j-l]) j = j - 1 Реализация алгоритма сортировки вставками на языке С приводится в листинге 2.2. Хотя время исполнения алгоритма сортировки вставками в наихудшем случае равно 0(н), его производительность значительно выше, если данные почти отсортированные, т. к. для помещения элемента в должное место достаточно лишь небольшого числа итераций внутреннего цикла. Сортировка вставками является, возможно, самым простым примером метода сорти- ровки поэтапными вставками, где мы создаем сложную структуру из п элементов, сначала создав ее из п — 1. элементов, после чего осуществляя необходимые изменения.
138 Часть I. Практическая разработка алгоритмов чтобы добавить последний элемент. Метод поэтапной вставки оказывается особенно полезным в геометрических алгоритмах. Обратите внимание, что быстрые алгоритмы сортировки на основе поэтапных вставок обеспечены эффективными структурами дан- ных. Операция вставки в сбалансированное дерево поиска занимает время O(logn). а общее время создания дерева равно (?(«Iog/7). При симметричном обходе такого дерева элементы считываются в отсортированном порядке, что занимает линейное время. 4.4. История из жизни. Билет на самолет Я взялся за эту работу в поисках справедливости. Меня наняла одна туристическая фирма, чтобы помочь им разработать алгоритм поиска самого дешевого маршрута для перелета из города х в город у. Сумасшедшие колебания цен на авиабилеты, устанав- ливаемые согласно современным методам "управления доходами", повергали меня в полное недоумение. Возникало впечатление, что цены на авиарейсы взлетают намного эффективнее, чем сами самолеты. Мне казалось, проблема в том, что авиакомпании скрывают действительно низкие цены на их рейсы. Я рассчитывал, что если я справ- люсь с поставленной задачей, это позволит мне в будущем покупать билеты поде- шевле. — Смотрите, — начал я свою речь на первом совещании. — Все не так уж сложно. Создаем граф, в котором вершины представляют аэропорты, и соединяем ребром каж- дую пару аэропортов и и v, между которыми есть прямой рейс. Устанавливаем вес это- го ребра равным стоимости самого дешевого имеющегося в наличии билета из и в v. Теперь самая низкая цена перелета из городах в городу будет соответствовать самому короткому пути между соответствующими точками графа. Этот путь можно найти с помощью алгоритма Дейкстры для поиска кратчайшего пути. Проблема решена! — провозгласил я, эффектно взмахнув руками. Члены собрания задумчиво покивали головами, потом разразились смехом. Мне пред- стояло кое-что узнать о чрезвычайно сложном процессе формирования цен на билеты пассажирских авиарейсов. В любой момент времени существует буквально миллион разных цен. при этом в течение дня они меняются несколько раз. Цена определяется сложным набором правил, которые являются общими для всей отрасли пассажирских авиаперевозок и представляют собой запутанную систему, не основанную на каких- либо логических принципах. Именно поэтому нам и требовался эффективный способ поиска минимальной цены перелета. Исключением из общего правила являлась только восточноафриканская страна Малави. Обладая населением в 12 миллионов и доходом на душу населения в 596 долларов (179 место в мире), она неожиданно оказалась серь- езным фактором, формирующим политику ценообразования мировых пассажирских авиаперевозок. Чтобы определить точную стоимость любого авиамаршрута, требова- лось убедиться, что он не проходит через Малави. Задачу усложнял тот факт, что для первого отрезка маршрута, скажем, от Лос- Анджелеса (аэропорт LAX) до Чикаго (аэропорт ORD) существует сотня разных тари- фов. и не меньшее количество тарифов может существовать для каждого последующе- го отрезка маршрута, например, от Чикаго до Нью-Йорка (аэропорт JFK). Самый деше-
Гпава 4. Сортировка и поиск 139 вый билет для отрезка LAX-ORD (скажем, для детей членов Американской ассоциации пенсионеров) может быть несовместим с самым дешевым билетом для отрезка ORD- JFK. (если, например, он является специальным предложением для мусульман, которое можно использовать только с последующей пересадкой на рейс в Мекку). Признав справедливость критики за чрезмерное упрощение задачи, я принялся за рабо- ту. Я начал со сведения задачи к наиболее простому случаю. — Хорошо. Скажем, вам нужно найти самый дешевый билет для рейса с одной пере- садкой (т. е. двухэтапного), который проходит вашу проверку соответствия правилам. Есть ли какой-либо способ заранее определить, какие пары этапов маршрута пройдут проверку, и не выполнять при этом саму проверку? — Нет никакой возможности знать это наперед,— заверили меня.— Мы только мо- жем выполнить процедуру "черного ящика", чтобы решить, существует ли конкретная цена на билет для данного маршрута и для данного пассажира. — Значит, наша цель заключается в том. чтобы вызывать эту процедуру для мини- мального количества комбинаций билетов. Это означает, что нужно оценивать все воз- можные комбинации билетов, начиная с самой дешевой до самой дорогой, до тех пор, пока мы не найдем первую удовлетворяющую правилам комбинацию. — Совершенно верно. — Тогда почему бы нам не составить набор всех возможных т х п пар билетов, отсор- тировать их по стоимости, после чего оценить их в отсортированной последовательно- сти? Безусловно, это можно сделать за время О(пт\о^(птУ)'. — Это мы сейчас и делаем, но составление полного набора т * п пар обходится до- вольно дорого. притом, что подходящей может оказаться первая пара. Я понял, что задача действительно интересная. — Что вам действительно нужно, так это эффективная структура данных, которая многократно возвращает следующую наи- более дорогую пару билетов, при этом не составляя все пары наперед. Это в самом деле было интересно. Поиск наибольшего элемента в наборе значений, подвергающемся вставкам и удалениям, является как раз той задачей, для решения ко- торой замечательно подходят очереди с приоритетами. Но проблема в данном случае заключалась в том, что мы не могли заранее заполнить ключами очередь с приорите- тами. Новые пары нужно было вставлять в очередь после каждой оценки. Я составил несколько примеров, как показано на рис. 4.3. Каждую возможную цену билета двухэтапного рейса можно было представлять спи- ском индексов ее компонентов (т. е. цен билетов каждого отрезка маршрута). Безус- ловно. самый дешевый билет для всего маршрута будет получен сложением самых де- шевых билетов для каждого отрезка маршрута. В нашем представлении это будет би- лет (I, I). Следующий самый дешевый билет получается сложением первого билета ' Вопрос, можно ли отсортировать все такие суммы быстрее, чем пт произвольных целых чисел, представляет собой нерешенную задачу теории алгоритмов. Дополнительную информацию по сорти- ровке А'+ Y (так называется эта задача) можно найти в книгах [Fre76] и [Lam92],
140 Часть I. Практическая разработка алгоритмов X Y X+Y $100 $50 $150 (1,1) $160 (2,1) $110 $75 $175 (1.2) $130 $125 $180 (3,1) $185 (2.2) $205 (2,3) $225 (1,3) $235 (2,3) $255 (3,3) Рис. 4.3. Сортировка в возрастающем порядке сумм А' и Y одного отрезка маршрута и второго билета другого маршрута, т. е. это будет или (1,2) или (2. 1). Дальше становится сложнее. Третьим самым дешевым билетом могла бы быть неиспользованная пара из двух, приведенных выше, или же пара (1.3) или (3. 1). На самом деле, это могла бы быть пара (3. 1), если бы третья по величине цена билета на отрезок X маршрута была 120 долларов. — Скажите, — спросил я, — у нас есть время, чтобы отсортировать эти оба списка цен билетов в возрастающем порядке? — В этом нет надобности. — ответил руководитель группы.— Они подаются из базы данных в отсортированном порядке. Это была хорошая новость. Нам не нужно было исследовать пары цен (/ + 1, /) и (/, j + 1) до рассмотрения пары (/’./). т. к. они очевидно были более высокими. — Есть, — сказал я. — Мы будем отслеживать пары индексов в очереди с приоритета- ми, а ключом пары будет сумма цен билетов. Вначале в очередь вставляется только одна пара цен — (1, 1). Если эта пара оказывается неподходящей, мы вставляем в оче- редь две следующие пары — (1, 2) и (2, 1). В общем, после исследования и выбраковки пары (Лу) мы последовательно вставляем в очередь пары (/ + I, /) и (i.J + 1). Таким об- разом, мы исследуем все пары в правильном порядке. Компания быстро уловила суть решения. — Конечно. Но как насчет дубликатов? Мы будем создавать пару (х,у) двумя разными способами — при расширении пар (х- 1,у) и (х.у- 1). — Вы правы. Нам нужна дополнительная структура данных, чтобы предотвратить по- явление дубликатов. Самым простым решением будет использование хэш-таблицы для проверки существования данной пары в очереди с приоритетами до ее вставки. В дей- ствительности. структура данных никогда не будет содержать больше, чем л активных пар, т. к. с каждым отдельным значением цены первого отрезка маршрута можно соз- дать только одну пару цен. Решение было принято. Наш подход естественным образом обобщался для маршрутов с более чем двумя отрезками (при этом сложность возрастала с увеличением количест-
Глава 4. Сортировка и поиск 141 ва этапов маршрута). Метод выбора первого оптимального варианта, свойственный нашей очереди с приоритетами, позволил системе прекращать поиск, как только она находила действительно самый дешевый билет. Такой подход оказался достаточно бы- стрым, чтобы предоставить интерактивный ответ пользователю. Однако я не заметил, чтобы мои авиабилеты хоть насколько подешевели. 4.5. Сортировка слиянием. Метод "разделяй и властвуй" Рекурсивные алгоритмы разбивают большую задачу на несколько подзадач. При ре- курсивном подходе сортируемые элементы разбиваются на две группы, каждая из этих меньших групп сортируется рекурсивно, после чего два отсортированных списка сли- ваются воедино, и их элементы чередуются, образуя полностью отсортированный об- щий список. Этот алгоритм называется сортировкой слиянием (mergesort). Mergesort (A[l,n]) Merge ( MergeSort(A[l, [n/2j]) , MergeSort (A[ [n/2] + l,n]) ) Базовый случай рекурсивной сортировки имеет место, когда исходный массив состоит из одного элемента, вследствие чего перестановки в нем невозможны. На рис. 4.4 пока- зана графическая иллюстрация работы алгоритма сортировки слиянием. Сортировку слиянием можно представлять себе как симметричный обход верхнего дерева, при ко- тором преобразования представлены в нижнем (перевернутом) дереве. MERGESORT Е Е G М R Е М R EEGMORRST Рис. 4.4. Работа алгоритма сортировки слиянием Эффективность сортировки слиянием зависит от эффективности слияния двух отсор- тированных частей в один отсортированный список. Одним из способов было бы объ- единить обе части и выполнить сортировку получившегося списка методом пирами- дальной сортировки или каким-либо другим способом, но это свело бы на нет все наши усилия, потраченные на сортировку этих частей по отдельности. Поэтому списки сливаются в один следующим образом. Оба списка отсортированы в возрастающем порядке, значит, наименьший элемент должен находиться в самом на-
142 Часть I. Практическая разработка алгоритмов чале одного из них. Этот наименьший элемент перемещается в общий список, при этом один из списков укорачивается на один элемент. Следующий наименьший элемент опять же должен быть самым первым в одном из оставшихся двух списков. Повторяя эту операцию до тех пор, пока не опустеют оба отсортированных списка, мы сливаем эти два списка (с общим количеством элементов и) в один отсортированный список, выполняя, самое большее, п - I сравнений за время О(п). Какое же общее время исполнения сортировки слиянием? Чтобы ответить на этот во- прос, следует принять во внимание, какой объем работы выполняется на каждом уров- не дерева сортировки. Если мы допустим для простоты, что и является степенью двой- ки, то k-й уровень содержит все 2* вызовов процедуры сортировки слиянием, обраба- тывающих поддиапазоны из и/2* элементов. Работа, выполняемая на нулевом уровне (к = 0), состоит в слиянии двух отсортирован- ных списков, каждый размером и/2, и при этом происходит, самое большее, п- 1 срав- нений. Работа, выполняемая на первом уровне (к = I), состоит в слиянии двух отсорти- рованных списков, каждый размером и/4. и при этом происходит, самое большее, и —2 сравнений. В общем, работа, выполняемая на к-м уровне, состоит в слиянии 2* пар от- сортированных списков, каждый размером п/2к k', и при этом происходит, самое боль- шее, п — 2 сравнений. Слияние всех элементов на каждом уровне выполняется за ли- нейное время. Каждый из п элементов фигурирует только в одной подзадаче на каждом уровне. Наиболее трудоемким случаем (по количеству сравнений) является самый верхний уровень. На каждом уровне количество элементов в подзадаче делится пополам. Количество делений пополам числа п до тех пор, пока не будет получена единица, равно pg, Так как глубина рекурсии составляет lg/v уровней, а каждый уровень обрабатывается за линейное время, то в наихудшем случае время исполнения алгоритма сортировки слиянием равно О(и1оц«). Алгоритм сортировки слиянием хорошо подходит для сортировки связных списков, т. к. он, в отличие от алгоритмов пирамидальной и быстрой сортировки, не основыва- ется на произвольном доступе к элементам. Его основным недостатком является необ- ходимость во вспомогательном буфере при сортировке массивов. Отсортированные связные списки можно легко слить вместе, не требуя дополнительной памяти, а просто упорядочивая указатели. Но для слияния двух отсортированных массивов (или частей массива) требуется третий массив для хранения результатов слияния, чтобы не поте- рять содержимое сливаемых массивов. Допустим, что мы сливаем множества {4, 5, 6} и {1,2, 3}, записанные слева направо в одном массиве. Без использования буфера нам придется записывать отсортированные элементы поверх элементов первой половины массива, вследствие чего последние будут утеряны. Сортировка слиянием является классическим примером алгоритмов типа "разделяй и властвуй". Мы всегда в выигрыше, когда можем разбить одну большую задачу на две подзадачи, т. к. меньшие задачи легче решить. Секрет заключается в том, чтобы вос- пользоваться двумя частичными решениями для создания решения всей задачи, как это было сделано с операцией слияния.
Гпава 4. Сортировка и поиск 143 Реализация Псевдокод алгоритма сортировки слиянием представлен в листинге 4.10. ........ ........................................................... . Листинг 4.10. Алгоритм сортировки слиянием j mergesort (item_type s[], int low, int high) { int i; /* Счетчик */ int middle; /* Индекс среднего элемента */ if (low < high) { middle = (low+high)/2; mergesort(s,low,middle); mergesort(s,middle+l,high); merge(s, low, middle, high); Но реализация части алгоритма, в которой осуществляется слияние отсортированных списков, оказывается более сложной. Проблема состоит в том, что нам нужно где-то хранить слитый массив. Чтобы избежать потери элементов в процессе слияния, мы сначала создаем копии исходных массивов (листинг 4.11). . Листинга 11. Процедура слияния массивов merge(item_type s[], int low, int middle, int high) ( int i; /* Счетчики */ queue bufferl, buffer2; /* Буфера для хранения элементов*/ init_queue(&bufferl); init_queue(&buffer2); for (i=low; i<=middle; i++) enqueue(abufferl,s[i]); for (i=middle+l; i<=high; i++) enqueue(&buffer2,s[i]); i = low; while (!(empty_queue(bbufferl) || empty_queue(&buffer2))) { if (headq(&bufferl) <= headq(&buffer2)) s[i++] = dequeue(abufferl); else s[i++] = dequeue(&buffer2); ) while (!empty_queue(&bufferl)) s[i++] = dequeue(abufferl); while (!empty_queue(&buffer2)) s[i++] = dequeue(&buffer2); 4.6. Быстрая сортировка. Рандомизированная версия Алгоритм быстрой сортировки (quicksort) работает таким образом. Из массива разме- ром п выбирается произвольный элемент р. Оставшиеся n— 1 элементов массива раз-
144 Часть I. Практическая разработка алгоритмов деляются на две части: левую (или нижнюю), содержащую все элементы, меньшие элемента/?, и правую (или верхнюю), содержащую все элементы, большие р. Графическая иллюстрация работы алгоритма быстрой сортировки представлена на рис. 4.5. Q U I С К s О R[T Q I С К S cTr) Т JJ Q I С K0R STU I с]<]о QRSTU l[cj К О Q R S Т и С I К О О R S Т и Рис. 4.5. Графическая иллюстрация работы алгоритма быстрой сортировки Элементр занимает отдельную ячейку между этими двумя частями. Такое разбиение массива на две части преследует две цели. Во-первых, опорный эле- мент р находится в точно той же позиции массива, в которой он будет находиться в конечном отсортированном массиве. Во-вторых, после разделения массива на части, в конечной отсортированной последовательности элементы не перемещаются из одной части в другую. Таким образом, сортировку элементов в левой и правой части массива можно выполнять независимо друг от друга. Это дает нам рекурсивный алгоритм сор- тировки, т. к. мы можем использовать подход с разбиением массива на две половины для сортировки каждой первоначальной половины. Такой алгоритм должен быть пра- вильным, поскольку, в конечном счете, каждый элемент оказывается в правильном месте. В листинге 4.12 приведен код алгоритма на языке С. Листинг 4.12. Алгоритм быстрой сортировки quicksort(item_type s[], int 1, int h) { int p; /* Индекс элемента-разделителя */ if ((h-l)>0) ( p = partition(s,l,h); quicksort(s,1,p-1); quicksort(s,p+l,h); 1 } Массив можно разбить на части за один проход с линейным временем исполнения для определенного опорного элемента посредством постоянного сопровождения трех час- тей массива: содержащей элементы меньшие, чем опорный элемент (слева от firsthigh), содержащей элементы равные или большие, чем опорный (между firsthigh и i) и содержащей непроверенные элементы (справа от i). Реализация процедуры раз- биения показана в листинге 4.13.
гпава 4. Сортировка и поиск 145 Листинг 4.13. Процедура разбиения массива на части int partition(item_type s[], int 1, int h) int i; /* Счетчик */ int p; /★ Индекс элемента-разделителя ”/ int firsthigh; /* Позиция для элемента-разделителя */ Г = h; firsthigh = 1; for (i=l; i<h; i++) if Cs[i] <s[p]) { swap(ss[i],ss[firsthigh]); firsthigh ++; I swap(&s[p],&s[firsthigh]); return(firsthigh); Так как процедура разделения содержит, самое большее, п операций обмена местами двух элементов, то разбиение массива выполняется за линейное время. Но каково об- щее время исполнения алгоритма быстрой сортировки? Подобно алгоритму сортиров- ки слиянием, алгоритм быстрой сортировки создает рекурсивное дерево вложенных поддеревьев массива из п элементов. Подобно алгоритму сортировки слиянием алго- ритм быстрой сортировки обрабатывает (не слиянием частей в один массив, а наобо- рот, разбиением массива на части) элементы каждого подмассива на каждом уровне за линейное время. Опять же, подобно алгоритму сортировки слиянием, общее время ис- полнения алгоритма быстрой сортировки равно O(rrh\ где h — высота рекурсивного дерева. Но здесь трудность состоит в том, что высота дерева зависит от конечного местонахо- ждения опорного элемента в каждой части массива. Если нам очень повезет, и опор- ным каждый раз будет элемент, находящийся посередине массива, то размер получен- ных вследствие такого деления подзадач всегда будет равен половине размера задачи предыдущего уровня. Высота представляет количество делений, которым подвергается массив и последующие подмассивы до тех пор, пока полученный подмассив не будет состоять из одной ячейки. Количество таких делений равно, самое большее, pg, п~|, Такая благоприятная ситуация показана на рис. 4.6, а и представляет наилучший слу- чай алгоритма быстрой сортировки. Теперь допустим, что нам постоянно не везет, и что наш выбор опорного элемента все время разбивает массив самым неравномерным образом. Это означает, что в качестве опорного всегда выбирается наибольший или наименьший элемент текущего массива. После установки этого опорного элемента в должную позицию у нас остается одна подзадача размером п — 1 элементов. Таким образом, мы тратим линейное время на ничтожно малое уменьшение задачи — всего лишь на один элемент (рис. 4.6. 6). Чтобы разбить массив так, что на каждом уровне будет находиться один элемент, требуется дерево высотой п - 1 и время ()(п).
146 Часть I Практическая разработка алгоритмов Рис. 4.6. Рекурсивные деревья алгоритма быстрой сортировки: наилучший случай (а) и наихудший случай (б) Таким образом, время исполнения алгоритма быстрой сортировки в наихудшем случае хуже, чем для пирамидальной сортировки или сортировки слиянием. Чтобы оправдать свое название, алгоритму быстрой сортировки следовало бы работать лучше в среднем случае. Для понимания этого требуется почувствовать произвольную выборку на ин- туитивном уровне. 4.6.1. Ожидаемое время исполнения алгоритма быстрой сортировки Ожидаемое время исполнения алгоритма быстрой сортировки зависит от высоты дере- ва разбивки первоначального массива, создаваемого произвольными опорными эле- ментами на каждом шаге разбивки. Время исполнения алгоритма слиянием равно O(nlogz7), потому что мы рекурсивно разбиваем общее количество элементов на две равные части, после чего сливаем в требуемом порядке за линейное время. Таким об- разом. всякий раз, когда опорный элемент находится возле центра сортируемого мас- сива (т. е. разделение проходит возле среднего элемента), мы получаем хорошее раз- биение и такую же производительность, как и для алгоритма сортировки слиянием. Я дам не строгое, а опирающееся на интуицию объяснение, почему время исполнения алгоритма быстрой сортировки в среднем случае равно O(nlogn). Какова вероятность того, что выбранный в произвольном порядке разделяющий элемент окажется хоро- шим? Самым лучшим разделяющим элементом был бы средний элемент массива, т. к. по каждую его сторону оказалось бы ровно по половине элементов исходного массива. К сожалению, вероятность выбрать наудачу точно средний элемент довольно низка, а именно равна \/п. Но допустим, что разделитель является достаточно хорошим, если он находится в центральной половине массива, т. е. в диапазоне элементов для сортировки от и/4 до Зя/4. Таких достаточно хороших разделительных элементов имеется довольно много, т. к. половина всех элементов расположена ближе к центру массива, чем к его краям (рис. 4.7). Таким образом, вероятность выбора достаточно хорошего разделительного элемента при каждом выборе равна 1/2. Возможно ли. чтобы при бросании монеты постоянно выпадала решка? При бросании честным образом нет. Если бросить монету п раз. то примерно в половине случаев вы- падет орел. Вероятность выбора достаточно хорошего разделителя можно рассматри- вать как выпадение орла при бросании монеты.
Гпава 4. Сортировка и поиск 147 I п/4 п/2 Зп/4 п Рис. 4.7. В половине случаев разделительный элемент расположен ближе к середине массива. чем к его краям При выборе самого худшего возможного достаточно хорошего разделителя большая часть разделенного массива содержит 3/?/4 элемента. Какой будет высота /?х дерева ал- горитма быстрой сортировки, созданного в результате последовательного выбора наи- худших достаточно хороших разделителей? Самый длинный путь по этому дереву проходит через части размером п, (3/4)/?, (3/4)‘и, и т. д. до I элемента. Сколько раз можно умножить п на 3/4, пока мы не дойдем до 1 ? Так как (3 / 4)/'li п = 1 => п = (43)Лк, то /?g = logi/;,/?. Но только половина выбираемых произвольно разделителей являются достаточно хорошими, а другую половину мы назовем плохими. Наихудшие из плохих разделите- лей по существу не уменьшают размера раздела вдоль самого глубокого пути. Самый глубокий путь от корня вниз по типичному дереву разделов быстрой сортировки, соз- данного посредством выбора произвольных разделителей, проходит через приблизи- тельно одинаковое количество достаточно хороших и плохих разделителей. Так как ожидаемое количество достаточно хороших и плохих разделителей одинаково, то пло- хие разделители могут увеличить высоту дерева, самое большее, вдвое, поэтому h~2hK = 21og4,3«. что сводится к ©(log/?). В среднем, деревья разделов алгоритма быстрой сортировки, созданные посредством произвольного выбора (и. аналогично, двоичные деревья поиска, созданные произ- вольными вставками), дают очень хорошие результаты. Более подробный анализ пока- зывает. что после п вставок средняя высота дерева составляет приблизительно 21пи. Так как 21п/? = 1,3861g?/?, то это всего лишь на 39% выше, чем высота идеально сбалан- сированного двоичного дерева. Поскольку время обработки каждого уровня составляет 0(н), то среднее время исполнения алгоритма быстрой сортировки равно <)(/?log/?). Если нам чрезвычайно не повезет и наши произвольно выбранные разделители всегда будут оказываться наибольшими или наименьшими элементами массива, то алгоритм быстрой сортировки превратится в сортировку методом выбора с временем исполне- ния О(п'). Но вероятность такого развития событий очень мала. 4.6.2. Рандомизированные алгоритмы Следует отметить одну важную тонкость в ожидаемом времени исполнения O(/?log??) алгоритма быстрой сортировки. В нашей реализации алгоритма в предыдущем разделе мы выбирали в качестве разделителя последний элемент каждого подмассива (части предыдущего массива). Допустим, мы используем этот алгоритм на отсортированном массиве. В таком случае при каждой разбивке будет выбираться наихудший изо всех возможных элемент-разделитель, а время исполнения будет квадратичным. Для любо- го детерминистического способа выбора элемента-разделителя существует наихудший входной экземпляр с квадратичным временем исполнения. В представленном ранее анализе утверждается лишь следующее: существует высокая вероятность, что время
148 Часть I. Практическая разработка алгоритмов исполнении алгоритма быстрой сортировки будет равно 0(»iog/z), при условии, что подлежащие сортировке данные идут в произвольном порядке. Теперь допустим, что прежде чем приступать к сортировке и элементов, мы переупо- рядочим их произвольным образом. Эту операцию перестановки можно выполнить за время ()(н) (подробности см. в разделе 13.7). Эта кажущаяся расточительность гаран- тирует ожидаемое время исполнения O(wlogn) для любого входного экземпляра задачи. Хотя наихудший случай времени исполнения продолжает оставаться возможным, он зависит исключительно от того, насколько нам повезет или не повезет. Но явно опре- деленного наихудшего входного экземпляра больше нет. Следовательно, теперь мы можем сказать: существует высокая вероятность, что время исполнения рандомизиро- ванного алгоритма быстрой сортировки будет равно ®(»logn) для любого входного экземпляра задачи. В качестве альтернативного варианта мы могли бы прийти к такому же утверждению, выбирая произвольный элемент-разделитель на каждом шаге. Рандомизация является мощным инструментом для улучшения алгоритмов с плохой временной сложностью в наихудшем случае, но с хорошей сложностью в среднем слу- чае. С ее помощью алгоритмы можно сделать более устойчивыми в граничных случаях и более эффективными на высоко структурированных вводных экземплярах, которые делают неэффективными эвристические механизмы принятия решений (как в случае с отсортированным входом для алгоритма быстрой сортировки). Рандомизацию часто можно применять с простыми алгоритмами, обеспечивая таким образом производи- тельность. которую в противном случае можно получить, только используя сложные детерминистические алгоритмы. Чтобы должным образом анализировать рандомизированные алгоритмы, необходимо иметь определенные познания в области теории вероятностей, обсуждение которой выходит за рамки данной книги. Но некоторые подходы к разработке эффективных рандомизированных алгоритмов можно с легкостью объяснить. ♦ Рандомизированная выборка. Допустим, мы хотим получить общее представление о среднем значении п элементов, но у нас нет ни достаточного времени, ни места, чтобы просмотреть все значения. В таком случае мы делаем выборку произвольных элементов из всего множества и исследуем эти элементы. Полученный результат должен быть репрезентативным для всего множества. Эта идея лежит в основе исследований путем опроса. Но если не обеспечить дейст- вительно произвольную выборку, а опросить х первых встречных, то возможны ис- кажения в ту или иную сторону. Во избежание таких искажений выполняющие оп- рос агентства обычно звонят по произвольным телефонным номерам и надеются, что кто-либо ответит. ♦ Рандомизированное хэширование. Мы уже говорили, что посредством хэширования можно реализовать словарные операции с ожидаемым временем исполнения 0(1). Но для любой хэш-функции имеется наихудший случай в виде набора ключей, ко- торые хэшируются в одну корзину. Но допустим, что первым шагом алгоритма мы выбираем произвольную функцию хэширования из большого семейства подходя- щих функций. Таким образом, мы получаем такую же улучшенную гарантию, как и для рандомизированного алгоритма быстрой сортировки.
Гпава 4. Сортировка и поиск 149 ♦ Рандомизированный поиск. Рандомизацию можно также применять для орга- низации методов поиска, таких как имитация отжига. Подробности см. в раз- деле 7.5.3. Остановка для размышлений. Болты и гайки ЗАДАЧА. Задача болтов и гаек определяется таким образом. Есть набор разных п гаек и такое же количество соответствующих болтов. Вам нужно найти для каждого болта подходящую гайку. Сравнивать можно только болты и гайки, т. е. нельзя сравнивать гайки с гайками или болты с болтами. Разработайте алгоритм для решения этой задачи за время О(/Г), а потом разработайте рандомизированный алгоритм для решения этой задачи за ожидаемое время O(/?log»). Решение. Алгоритм полного перебора решает эту задачу, сравнивая первый болт со всеми гайками, пока не найдет подходящую, после чего переходит к следующему бол- ту и сравнивает его со всеми оставшимися гайками, и т. д. В худшем случае для перво- го болта потребуется п сравнений. Повторение этой процедуры для всех последующих болтов со всеми остающимися гайками дает нам квадратичное число сравнений. А если вместо последовательного перебора болтов, начиная с первого, мы каждый раз будем брать произвольный болт? В среднем, мы можем ожидать перебора около поло- вины гаек, пока не найдем подходящую для данного болта, поэтому этот рандомизиро- ванный алгоритм будет иметь ожидаемое время исполнения наполовину меньшее, чем в худшем случае. Это можно рассматривать как определенное улучшение, хотя и не асимптотического типа. Так как рандомизированный алгоритм быстрой сортировки обеспечивает ожидаемое время исполнения, то будет естественной идея эмулировать его для решения данной задачи. Действительно, в отсортированных последовательностях для Его болта подой- дет /-я гайка. Основной операцией алгоритма быстрой сортировки является разбиение массива эле- ментов на две части по элементу-разделителю. Можем ли мы разбить множества гаек и болтов по произвольно выбранному болту Ы Определенно, что, сравнивая размер гаек с размером произвольно выбранного болтай, мы можем разбить множество гаек на две части — меньших и больших чем гайка, подходящая для болта Ь. Но нам также нужно разбить на части множество болтов по этому же произвольно выбранному болту раз- мером й, а мы не можем сравнивать болты друг с другом. Но ведь когда мы найдем гайку подходящего размера для болта размером й, то мы можем использовать ее для разбиения болтов, точно так же, как мы использовали болт для разбиения на части множества гаек. Разбиение на части гаек и болтов происходит за 2п-2 сравнения, а время исполнения оставшихся операций следует непосредственно из анализа рандоми- зированного алгоритма быстрой сортировки. Эта задача интересна тем, что для нее не существует простого детерминистического алгоритма. Это хорошая иллюстрация, как использование рандомизации позволяет из- бавиться от плохих входных экземпляров задачи посредством простого и изящного алгоритма.
150 Часть I. Практическая разработка алгоритмов 4.6.3. Действительно ли алгоритм быстрой сортировки работает быстро? Существует четкое, асимптотическое различие между алгоритмом с временем испол- нения C-)(nlogn) и алгоритмом с временем исполнения &(п~). Поэтому только самый не- доверчивый читатель будет сомневаться в моем утверждении, что алгоритмы сорти- ровки слиянием, пирамидальной сортировки и быстрой сортировки покажут лучшую производительность на достаточно больших входных экземплярах задачи, чем алго- ритмы сортировки вставками или сортировки методом выбора. Но как можно сравнить два алгоритма с временной сложностью ©(nlog/r), чтобы ре- шить, который из них быстрее? Как можно доказать, что алгоритм быстрой сортировки действительно быстрый? К сожалению, модель RAM и асимптотический анализ явля- ются слишком грубыми инструментами для сравнений такого типа. В случае алгорит- мов с одинаковой асимптотической сложностью, детали их реализации и особенности программной и аппаратной платформы, на которой они исполняются, такие как объем оперативной памяти и производительность кэша, могут вполне оказаться решающим фактором. Что можно сказать, так это то. что в процессе экспериментирования было установлено, что должным образом реализованный алгоритм быстрой сортировки обычно в 2-3 раза быстрее, чем алгоритм сортировки слиянием или пирамидальной сортировки. Основ- ной причиной этому является тот факт, что в алгоритме быстрой сортировки операции внутреннего цикла менее сложные. Но если вы не верите, что алгоритм быстрой сорти- ровки быстрее, я не смогу вам этого доказать. Ответ на этот вопрос лежит за рамками применения аналитических инструментов, рассматриваемых в этой книге. Самым луч- шим способом будет реализовать оба алгоритма и определить их эффективность экс- периментальным способом. 4.7. Сортировка распределением. Метод блочной сортировки Имена в зелефонной книге можно отсортировать по первой букве фамилии. Таким об- разом у нас получится 26 разных корзин для имен (в английском алфавите 26 букв — прим. перев.). Обратите внимание, что любая фамилия в корзине J должна находиться после всех фамилий из корзины /, но перед любой фамилией из корзины К. Вследствие этого обстоятельства фамилии можно отсортировать в каждой отдельной корзине, по- сле чего просто объединить отсортированные корзины. Если имена распределены среди корзин равномерно, то каждая из получившихся 26 подзадач сортировки должна быть значительно меньшего размера, чем первона- чальная задача. Далее, разделяя каждую корзину по второй букве фамилии, потом третьей и т. д„ мы создаем корзины все меньшего и меньшего размера (т. е. содержа- щие все меньше и меньше элементов-фамилий). Список фамилий будет отсортирован, поскольку вследствие такого деления каждая корзина содержит только одну фамилию. Только что описанный алгоритм сортировки называется блочной сортировкой (bucket sort) или сортировкой распределением (distribution sort).
Гпава 4. Сортировка и поиск 151 Применение корзин является очень эффективным подходом, когда мы уверены, что данные распределены приблизительно равномерно. Эта же идея лежит в основе хэш- таблиц, kd-деревьсв и многих других практических структур данных. Обратной сторо- ной этой медали является то, что производительность может быть ужасной, если рас- пределение данных окажется не таким, на какое мы рассчитывали. Хотя для таких структур данных, как сбалансированные двоичные деревья, гарантируется производи- тельность худшего случая для входных данных с любым распределением, такая гаран- тия отсутствует для эвристических структур данных с неожиданным распределением ввода. Неравномерное распределение данных встречается и в реальной жизни. Возьмем, на- пример, такую необычную американскую фамилию, как Shifflett. Когда я последний раз смотрел телефонный справочник Манхэттена (в котором свыше миллиона фами- лий), то в нем было пять человек с такой фамилией. Как вы думаете, сколько людей по фамилии Shiffiett проживает в небольшом городке с населением в 50 000? На рис. 4.8 показан фрагмент телефонного справочника для города Шарлотсвилл в штате Вирд- жиния. Фамилия Shiffiett занимает в справочнике более двух с половиной страниц. Shifflett DabMt к Rvcunvuia - Shifflett Debra S SR «12 Quinque Shifflett Dalma SR60».............. Shifflett Dalma* Crane ............ Shifflett Dempiay A Maritynn 100 Greenbrier Tar.............. Shifflett DaNM Rt Ы7 Pyb* — SNfflatt Dannta StanardnWe • • Shifflett Danota H St*nara*v4lv SNfflatt Dewey E RW................ SNfflatt Dewey 0 Dyka.............. W-ГУЛ 9AS-MU 823-5901 swmwt jama» zuv wiiMtrawn ru SNfflatt JamfflSAOl StorwNne» Av- Shifflett JamecC ..... Shifflett Jam* E EMtytwta .......... SNfflatt Jama* E Jr SSI CtavatanJ M SNfflatt Jama* P - lota LorhA**iaov Shifflett JamaiF А УсгпаНЫП-- Shifflett Jama* J 14Э0 Ruaby Av--- Shifflett Jame* К St Gaorga Av ----- Shifflett Jama* L 8R33 SWWfflvi* Shifflett ( k Patricia it SNfflatt Dontaa М Ml P73-719S 94S-8W7 9U-2M4 MS-S57S ____________________ 98S-72M SNfflatt Jama* 0 EartrrNta ... W9-7W5 Shlfflatt Jama*0. SUnanMM .... 2M-4227 SNfflatt Jama* ROM LynchberaAe- 974-7443 ShfflUJ Juma» 7331 or — Рис. 4.8. Фрагмент телефонного справочника Клан Shiffiett присутствует в этом регионе много лет, и это обстоятельство приведет в расстройство любую программу сортировки распределением, т. к. при последователь- ном разбиении корзины S на Sh на Shi на Shif на... на Shifflett в действительности не происходит никакого значительного разбиения. Подведение итогов Сортировку можно использовать для иллюстрации большинства парадигм разработки ал- горитмов. Методы структур данных, принцип "разделяй и властвуй”, рандомизация и по- этапная обработка— все эти подходы позволяют разрабатывать эффективные алгоритмы сортировки 4.7.1. Нижние пределы для сортировки Обсудим последний вопрос, касающийся сложности алгоритмов сортировки. Мы знаем несколько алгоритмов сортировки. Для всех время исполнения в наихудшем случае было равно O(«log/;), и ни один из них не выполнялся за линейное время. Для сорти- ровки я элементов неизбежно требуется рассмотреть каждый из них, поэтому времен- ная сложность любого алгоритма сортировки в наихудшем случае должна быть £1(п). Можем ли мы закрыть эту брешь в 0(1оуи)?
152 Часть I. Практическая разработка алгоритмов К сожалению, нет. Нижнюю границу fi(wlogw) можно установить на основе того об- стоятельства, что любой алгоритм сортировки должен вести себя по-разному при сор- тировке каждой из возможных разных м! перестановок п элементов. Время исполнения любого алгоритма сортировки на основе сравнений определяется разультатом каждого попарного сравнения. Набор всех возможных исполнений такого алгоритма можно представить в виде дерева с и! листьями. Минимальная высота дерева соответствует самому быстрому возможному алгоритму, и получается, что lg(«!) = 0(nlog/?). Эта нижняя граница важна по нескольким причинам. Прежде всего, этот подход можно расширить, чтобы получить нижние границы для многих приложений сортировки, включая определение уникальности элемента, поиск наиболее часто встречающегося элемента, а также создание выпуклых оболочек. Среди алгоритмических задач сорти- ровка является одной из немногих, обладающих нетривиальной нижней границей В г часе 9 мы рассмотрим альтернативный подход к доказательству маловероятности существования быстрых алгоритмов. 4.8. История из жизни. Адвокат Скиена Я веду тихий, достаточно честный образ жизни. Одной из наград за такой образ жизни является то, что сплю спокойно, не опасаясь никаких неприятных сюрпризов Поэтому я был крайне поражен, когда мне позвонила женщина-адвокат, которая хотела не просто поговорить со мной, но поговорить об алгоритмах сортировки. Ее фирма работала над делом, касающимся высокопроизводительных программ сорти- ровки, и им был нужен эксперт, который мог бы объяснить присяжным технические подробности. По первому изданию этой книги они поняли, что я кое-что знаю об алго- ритмах, и поэтому решили обратиться именно ко мне. Но прежде, чем нанять меня, они захотели узнать, как студенты оценивают мои преподавательские способности, чтобы удостовериться в том, что я могу доходчиво объяснять людям сложные понятия1. Уча- стие в этом деле оказалось увлекательной возможностью узнать, как работают дейст- вительно быстрые программы сортировки. Я полагал, что, наконец, смогу ответить на вопрос, какой из алгоритмов сортировки в памяти является самым быстрым. Будет это пирамидальная сортировка или быстрая сортировка? Какие особенности алгоритмов способствовали сведению к минимуму количества сравнений в практических прило- жениях? Ответ был довольно отрезвляющим. Сортировка в памяти никого не интересовала. Главное заключалось в сортировке громадных файлов, намного больших, чем которые могли поместиться в оперативную память. Основная работа состояла в записывании и считывании данных с диска, и хитроумные алгоритмы для сортировки в памяти нико- му не были интересны, т. к. в реальной жизни сортировать нужно многие гигабайты данных. Вспомним, что время обращения к жесткому диску относительно велико из-за необхо- димости позиционировать магнитную головку чтения/записи. После того как головка 1 Один мой цинично наогроснный коллега по профессорско-преподавательскому составу сказал, что это первый случай, когда кто-либо поинтересовался мнением студентов относительно преподана г елей.
Гпава 4. Сортировка и поиск 153 установлена в нужном месте, передача данных осуществляется очень быстро, и для чтения большого блока данных требуется приблизительно такое же время, что и для одного байта. Таким образом, целью является сведение к минимуму количества блоков данных для чтения/записи и координация этих операций, чтобы алгоритм сортировки никогда не простаивал, ожидая данные. Необходимость интенсивной работы с диском во время сортировки лучше всего де- монстрируется на ежегодных соревнованиях по сортировке Minutesort. Перед участни- ками стоит задача отсортировать наибольший объем данных за одну минуту. Дейст- вующим чемпионом в этом соревновании является Джим Вилли (Jim Wyllie) из отдела исследований IBM. На своем скромном 40-узловом кластере из 80 процессоров Itanium, оснащенном массивом из 2 520 сетевых дисков, он смог отсортировать П6 гигабайт данных за 58,7 секунд. Существует и другое, более приближенное к практике, соревно- вание Pennysort, целью которого является получение максимальной производительно- сти сортировки на один цент стоимости оборудования. Действующим чемпионом в этом соревновании является представленный китайскими разработчиками алгоритм BSIS. который на персональном компьютере стоимостью 760 долларов, оснащенном четырьмя приводами SATA, отсортировал 32 гигабайта за I 679 секунд. Информацию о текущих рекордах сортировки можно получить на веб-сайте Sort Benchmark по адре- су http://sortbenchniark.org/. Итак, какой алгоритм лучше всего подходит для сортировки данных вне оперативной памяти? Оказывается, это сортировка многоканальным слиянием с применением мно- жества инженерных и других специальных приемов. Создается пирамида из членов верхнего блока каждого из к отсортированных списков. Последовательно снимая с этой пирамиды верхний элемент и сливая вместе эти к списков, алгоритм создает об- щий отсортированный список. Так как пирамида находится в памяти, то эти операции выполняются с высокой ско- ростью. Когда имеется достаточно большой объем отсортированных данных, они запи- сываются на диск, тем самим освобождая память для новых данных. Когда начинают заканчиваться элементы верхнего блока одного из сливаемых отсортированных к спи- сков. загружается следующий верхний блок данных из этого списка. Оценить на этом уровне производительность программ/алгоритмов сортировки и ре- шить, какой из них действительно быстрее, очень трудно. Будет ли справедливо срав- нивать производительность коммерческой программы, предназначенной для обработки общих файлов, с производительностью кода, в котором убрано все лишнее и который оптимизирован для обработки целых чисел? В соревновании Minutesort в качестве входных данных для сортировки используются произвольно генерируемые записи раз- мером в 100 байтов. Сортировка таких записей существенно отличается от сортировки имен или целых чисел. Например, при сортировке этих записей широко применяется прием, при котором от каждого элемента берется короткий префикс, и сортировка сна- чала выполняется по этим префиксам, чтобы избежать перемещения лишних байтов. Какие же уроки можно извлечь из всего этого? Самый важный состоит в том, что следует всячески избегать втягивания себя в судеб- ное разбирательство в качестве как истца, так и ответ чика. Суды не являются инструментом для скорого разрешения разногласий. Юридические баталии во многом похожи на военные битвы: они очень быстро обостряются, стано-
154 Часть I. Практическая разработка алгоритмов вятся очень дорогостоящими в денежном и временном отношении, а также в отноше- нии душевного состояния, и обычно кончаются только тогда, когда обе стороны исто- щены и идут на компромисс. Мудрые люди могут решить свои проблемы, не прибегая к помощи судов. Усвоив этот урок должным образом сейчас, вы сможете сэкономить средства, в тысячи раз превышающие стоимость этой книги. Что касается технических аспектов, то важно уделять должное внимание производи- тельности внешних средств хранения данных при обработке очень больших наборов данных алгоритмами с низкой временной сложностью (например, линейной или рав- ной nlogH). В таких случаях даже такие постоянные множители, как 5 или 10, могут означать раз- ницу между возможностью и невозможностью сортировки. Конечно же, для наборов данных большого объема алгоритмы с квадратичным време- нем исполнения обречены на неудачу независимо от времени доступа к данным на дисках. 4.9. Двоичный поиск и связанные с ним алгоритмы Алгоритм двоичного поиска позволяет осуществлять быстрый поиск в массиве S от- сортированных ключей. Чтобы найти ключ q, мы сравниваем значение q со средним ключом массива 5[н/2]. Если значение ключа q меньше, чем значение ключа 5[л/2], значит, данный ключ должен находиться в верхней половине массива 5; в противном случае он должен находиться в его нижней половине. Рекурсивно повторяя этот про- цесс на половине, содержащей элемент с/, мы находим его за 1gсравнений, что яв- ляется большим улучшением по сравнению с ожидаемыми и/2 сравнениями при после- довательном поиске. Реализация алгоритма двоичного поиска на языке С показана в листинге 4 14. Листинг 4.14. Реализация алгоритма двоичного поиска int binary_search(item_type s[], item_type key, int low, int high) int middle; /* Индекс среднего элемента */ if (low > high) return (-1); /* Ключ не найден */ middle = (low+high)/2; if (s[middle] == key) return(middle); if (s[middle] > key) return (binary_search(s, key, low, middle-1)); else return (binary_search(s, key, middle+1, high) ); ) Вероятно, все это вы уже знаете. Но важно почувствовать, насколько быстрым являет- ся алгоритм двоичного поиска. Существует популярная детская игра "Двадцать вопро- сов". Суть этой игры состоит в том, что один из игроков загадывает слова, а второй
Глава 4. Сортировка и поиск 155 пытается угадать его. Если после 20 вопросов слово не отгадано, то выигрывает пер- вый игрок, в противном случае — второй. Но в действительности второй игрок всегда находится в выигрышном положении, т. к. он для угадывания слова может применить стратегию двоичного поиска. Для этого он берет словарь, открывает его посередине, выбирает слово (например, "ночь") и спрашивает первого игрока, находится ли зага- данное им слово перед словом "ночь". Процесс рекурсивно повторяется для соответст- вующей половины, пока слово не будет отгадано. Так как стандартные словари содержат от 50 000 до 200 000 слов, то можно быть уве- ренным, что 20 попыток будет более чем достаточно. 4.9.1. Частота вхождения элемента Простые варианты двоичного поиска порождают несколько интересных алгоритмов. Допустим, мы хотим сосчитать, сколько разданный ключ к (например, "Skiena") встре- чается в данном отсортированном массиве. Так как при сортировке все копии к соби- раются в один непрерывный блок, то задача сводится к поиску этого блока и после- дующего измерения его размера. Представленная ранее процедура двоичного поиска позволяет найти индекс одного из элементов в соответствующем блоке (х) за время O(lgn). Естественным способом опре- деления границ блока будет последовательная проверка элементов слева от х до тех пор, пока не будет найден элемент, отличающийся от ключа, и повторение процесса проверки для элементов справа от х. Разница между индексами левой и правой грани- цы блока, увеличенная на единицу, и будет количеством вхождений элемента к в дан- ный набор данных. Этот алгоритм исполняется за время O(lg« + .у), где s— количество вхождений ключа. Но если весь массив состоит из одинаковых ключей, то это время может ухудшиться до линейного. Алгоритм двоичного поиска можно ускорить, модифицировав его для поиска границ блока, содержащего элемент к, вместо самого к. Допустим, мы удалим проверку на равенство: if (s [middle] == key) return (middle); из реализации двоичного поиска в листинге 4.14 и для каждого неуспешного поиска вместо -1 будем возвращать индекс low. Теперь любой поиск будет заканчиваться не- удачей по причине отсутствия проверки на равенство. При сравнении ключа с одина- ковым элементом массива процесс поиска будет переходить в правую половину масси- ва, в конце концов, останавливаясь на правой границе блока одинаковых элементов. Левая граница блока определяется изменением направления двоичного сравнения на обратное и повторением поиска. Так как поиск выполняется за время O(\gn). зо под- счет количества вхождений элемента занимает логарифмическое время, независимо от размера блока. 4.9.2. Односторонний двоичный поиск Теперь допустим, что у нас есть массив, заполненный последовательностью нулей, за которыми следует неограниченная последовательность единиц, и нужно найти границу между этими двумя последовательностями. Если бы мы знали количество п элементов
156 Часть I. Практическая разработка алгоритмов массива, то на определение точки перехода посредством двоичного поиска ушло бы pgnl операций сравнения. При отсутствии границы мы можем последовательно вы- полнять сравнения по увеличивающимся интервалам (Я[1], Л[2], Л[4], Л[8], Л[16],...), пока не найдем первый ненулевой элемент. Теперь у нас имеется окно, содержащее целевой элемент, и мы можем применить двоичный поиск. Такой односторонний дво- ичный поиск возвращает границу р за, самое большее, 2pg р~\ операций сравнения, независимо от размера массива. Односторонний двоичный поиск особенно хорошо подходит для локализации элемента, расположенного недалеко от текущей позиции просмотра. 4.9.3. Корни числа Квадратным корнем числа п является такое число г, для которого г2 = п. Хотя операция вычисления квадратного корня имеется в любом карманном калькуляторе, нам будет полезно разработать эффективный алгоритм для его вычисления. Заметим, что квадратный корень числа п > 1 должен находиться в интервале от 1 до п. Пусть /= 1, /•= п. Теперь рассмотрим среднюю точку этого интервала т= (1+ г)/2 и отношение т~ к п. Если п > т~, то квадратный корень должен быть больше, чем т, по- этому мы устанавливаем I = т и повторяем процедуру. Если п < т~. то квадратный корень должен быть меньшим, чем т. поэтому устанавливаем г = т и повторяем про- цедуру. В любом случае мы сократили интервал наполовину посредством всего лишь одного сравнения. Продолжая действовать подобным образом, мы найдем квадратный корень без учета знака за Ign сравнений. Этот метод деления интервала пополам можно также применять для решения более общей задачи поиска корней уравнения. Число х называется корнем функции / если /(х) = 0. Возьмем два числа I и г, для которых//) > 0 и /г) < 0. Если f является непре- рывной функцией, то ее корень должен находиться в интервале между I и г. В зависи- мости от знака fljri), принимая т — (1+ г)/2, мы можем уменьшить это содержащее ко- рень окно наполовину за одно сравнение, прекращая поиск, как только наша оценка корня становится достаточно точной. Для обоих типов задач поиска корня известны алгоритмы их решения, которые выдают результат быстрее, чем двоичный поиск. В частности, вместо исследования средней точки интервала в этих алгоритмах применяется интерполяция для поиска подходящей точки, расположенной ближе к искомому корню. Тем не менее, метод двоичного поис- ка является простым и надежным, и работает так хорошо, насколько это возможно, не требуя дополнительной информации о самой функции. 4.10. Метод "разделяй и властвуй" Один из наиболее эффективных подходов к решению задач состоит в разбиении их на меньшие части, поддающиеся решению с большей легкостью. Задачи меньшего разме- ра являются менее сложными, что позволяет фокусировать внимание на деталях, кото- рые не попадают в поле зрения при исследовании всей задачи. Когда мы можем раз- бить задачу на более мелкие экземпляры задачи этого же типа, то становится очевид-
Глава 4. Сортировка и поиск 157 ным использование для ее решения рекурсивного алгоритма. Для эффективной парал- лельной обработки задачу необходимо разложить, по крайней мере, на столько мень- ших задач, сколько имеется процессоров. Это требование становится еще более важ- ным с развитием кластерных вычислений и многоядерных процессоров. Принцип разбиения задачи на меньшие части лежит в основе двух важных парадигм разработки алгоритмов. В частности, в главе 8 обсуждается динамическое программи- рование. Сузь этого метода состоит в удалении из задачи некоторого элемента, реше- нии получившейся меньшей задачи и использовании найденного решения, для кор- ректного возвращения элемента на место. А в методе "разделяй и властвуй" задача ре- курсивно разбивается на, скажем, половины, каждая половина решается по отдельности, после чего решения каждой половины объединяются в общее решение. Эффективный алгоритм получается в том случае, когда слияние решений половин за- нимает меньше времени, чем их решение. Классическим примером алгоритма типа "разделяй и властвуй" является алгоритм сортировки слиянием, рассмотренный в раз- деле 4.5. Слияние двух отсортированных списков, содержащих по п/2 элементов и по- лученных за время O(n\gn), занимает только линейное время. Принцип "разделяй и властвуй" применяется во многих важных алгоритмах, включая сортировку слиянием, быстрое преобразование Фурье и умножение матриц методом Страссена. Но я нахожу этот принцип трудным для практической разработки алгорит- мов иных, чем двоичный поиск и его разновидности. Возможность анализа алгоритмов типа "разделяй и властвуй" зависит от нашего умения выяснить асимптотику рекур- рентных соотношений, определяющих сложность таких рекурсивных алгоритмов. 4. 10.1. Рекуррентные соотношения Временная сложность многих алгоритмов типа "разделяй и властвуй" естественно формируется рекуррентными соотношениями. Оценка рекуррентных соотношений яв- ляется важным аспектом для понимания, в каких обстоятельствах от алгоритмов типа "разделяй и властвуй" можно ожидать хорошую производительность, а также является важным инструментом для общего анализа. Читатели, которых идея анализа не приво- дит в особый восторг, могут пропустить этот раздел, но должное представление о раз- работке можно получить, лишь понимая поведение рекуррентных соотношений. Что же собой представляет рекуррентное соотношение? Это уравнение, которое опре- делено посредством самого себя. В качестве примера рекуррентного соотношения можно привести последовазельность чисел Фибоначчи, определяемую равенством F„ = । + F,, 2 (см. раздел 8.1.1). С помощью рекуррентных соотношений можно вы- разить также многие другие аналитические функции. В частности, посредством рекур- рентного соотношения можно представить любой многочлен, например, линейную функцию: ап = а„ ! + 1,й| = 1 —>ап = п Любую показательную функцию также можно выразить посредством рекуррентного соотношения, например: а,, = 2а,,. |, а, = 1 —> а„ = 2"”1
158 Часть I. Практическая разработка алгоритмов Наконец, многие неооычные функции, которые непросто выразить посредством обыч- ной нотации, можно представить с помощью рекуррентного соотношения, например: a„ = па„. |, <2| = 1 —> а„ = п\ Все это означает, что рекуррентные соотношения являются очень гибким средством для представления функций. Кроме рекуррентных соотношений, свойством ссылаться на самих себя также обладают рекурсивные программы или алгоритмы, как можно ви- деть по общему корню обоих терминов. По существу, рекуррентные соотношения пре- доставляют способ анализировать рекурсивные структуры, такие как алгоритмы. 4. 10.2. Рекуррентные соотношения метода "разделяй и властвуй" Как уже упоминалось, алгоритмы типа "разделяй и властвуй" разбивают задачу на не- сколько (скажем, а) меньших подзадач, каждая из которых имеет размер п/b. Кроме этого, слияние решений подзадач в общее решение занимает время Ди). Пусть Г(и) — время решения алгоритмом наихудшего случая задачи размером и. Тогда Т{п) предос- тавляется следующим рекуррентным соотношением: Т(и) = aT(n/ti) +Дп) Рассмотрим примеры использования рекуррентных соотношений для решения задач. ♦ Сортировка. Время исполнения алгоритма сортировки слиянием определяется ре- куррентным соотношением Т(п) = 2Т(п/2) + О(и), т. к. алгоритм рекурсивно разделя- ет входные данные на равные половины, после чего выполняет слияние частных решений в общее за линейное время. Фактически это рекуррентное соотношение сводится к соотношению Т(п) = О(п\ер), которое было получено ранее. ♦ Двоичный поиск. Время исполнения алгоритма двоичного поиска представляется рекуррентным соотношением Т(п) = Т(п!2) + 0(1). т. к. каждый шаг уменьшения размера задачи вдвое выполняется за линейное время. Фактически это рекуррентное соотношение сводится к соотношению Т(п) = О(п\ор\ которое было получено ранее. ♦ Быстрое создание пирамиды. Процедура bubble down (см. листинг 4.5) создает пи- рамиду из п элементов, создавая две пирамиды, каждая из которых содержит п/2 элементов, а потом сливая их с корнем. Эта процедура занимает логарифмиче- ское время. Мы получим рекуррентное соотношение Т(п) = 2Т(п/2) + (9(lgn). Факти- чески оно сводится к соотношению Т(п) = О(п), которое было получено ранее. ♦ Умножение матриц. Как описано в разделе 2.5 4, время исполнения стандартного алгоритма для умножения двух матриц размером п х п равно О(п). т. к. для каждого из п~ элементов в матрице произведений мы вычисляем скалярное произведение п членов. Но в книге [Str69] рассматривается алгоритм типа "разделяй и властвуй", который для умножения двух матриц размером п х п манипулирует произведениями семи матриц размером п/2 х п!2. Временная сложность этого рекуррентного соотношения равна Т(п) = 7Т(п/2) + О(п ). Фактически, это рекуррентное соотношение сводится к 7'(п) = = О(п 8|). что невозможно предсказать, не решая соотношение.
Гпава 4. Сортировка и поиск 159 4. 10.3. Решение рекуррентных соотношений типа "разделяй и властвуй" (*) В действительности, рекуррентные соотношения типа "разделяй и властвуй" в форме Ди) = al\n/b) + fin) обычно очень легко решаются, т. к. решения обычно относятся к одному из трех отдельных классов: 1. Если для некоторой константы ь> 0 существует функция /(и) = (?(wlog',o~E), тогда 7'(и) = 0(и,°ел°). 2. Если /(») = 0(лг1ое',°). тогда 7'(z?) = 0(/JIog*"lgn). 3. Если для некоторой константы е> 0 существует функция Q(/7log/,‘'+E) и для не- которой константы с < 1 существует функция, такая что flnlb) < сДп), тогда Ли) = ©(/(«)). Хотя все эти формулы выглядят устрашающе, в действительности их совсем не трудно использовать. Вопрос заключается в определении, какой случай так называемой основ- ной теоремы является действительным для данного рекуррентного соотношения. Пер- вый случай применим для создания пирамиды и умножения матриц, а второй случай действителен для сортировки слиянием и двоичного поиска. Третий случай обычно нужен для не столь элегантных алгоритмов, в которых затраты на слияние подзадач превышают затраты на все остальные операции. Основную теорему можно представлять себе в виде черного ящика, которым мы умеем пользоваться, но устройство которого нам неизвестно. Однако после некоторого раз- мышления появляется понимание, как работает основная теорема. На рис. 4.9 показано рекурсивное дерево для типичного алгоритма типа "разделяй и властвуй", выражаемого рекуррентным соотношением Т(п) = aT(ji/b) + fin). Задача размером в п элементов разбиваезся на а подзадач размером п/b элементов. Каждая подзадача размером в к элементов выполняется за время О(/1кУ). Общее время исполнения алгоритма будет равно сумме временных затрат на выполнение подзадач, сложенной с накладными расходами на создание рекурсивного дерева. Высота этого дерева равна h = log/, и, а количество листьев равно ah = a'v6h". Посредством опреде- ленных алгебраических манипуляций последнее выражение можно упростить до >7,ое*" . Три случая основной теоремы соответствуют трем разным ситуациям, которые могут доминировать в зависимости от а. b и Ди): 1. Слишком много листьев. Если количество листьев превышает сумму затрат на внутреннюю обработку, то общее время исполнения будет равно (?(/?|ое',‘'). 2. Одинаковый объем работы на каждом уровне. По мере прохождения вниз по дере- ву размер каждой задачи уменьшается, но количество задач, подлежащих решению, увеличивается. Если сумма затрат на внутреннюю обработку одинакова для каждо- го уровня, то общее время исполнения вычисляется умножением затрат на каждом уровне (и1ое'’” ) на количество уровней (log/,/z) и будет равно <9(и1обА" 1цл).
160 Часть I. Практическая разработка алгоритмов размер подзадачи = п размер подзадачи = п/Ь размер подзадачи = п/Ь высота — log6 п размер подзадачи = Ь ширина = а ~ п Рис. 4.9. Рекурсивное дерево, полученное в результате разложения каждой задачи размером п на а задач размером п/Ь 3. Слишком большое время обработки корня Если с возрастанием п затраты на внут- реннюю обработку возрастают быстрыми темпами, то будут доминировать затраты на обработку корня. В таком случае общее время исполнения будет равно Замечания к главе Из алгоритмов сортировки, которые не рассмотрены в этой главе, наиболее интересен алгоритм сортировки методом Шелла (shellsort), представляющий собой более эф- фективную версию алгоритма сортировки вставками, и алгоритм поразрядной сорти- ровки (radix sort), являющийся эффективным алгоритмом для сортировки строк. Узнать больше об этих и всех других алгоритмах сортировки можно в книге [Кпи98], содер- жащей сотни страниц интересного материала по сортировке. В том виде, в каком он реализован в этой главе, алгоритм сортировки слиянием копи- рует сливаемые элементы во вспомогательный буфер, чтобы не потерять оригинальные значения сортируемых элементов. Посредством сложных манипуляций с буфером, этот алгоритм можно реализовать для сортировки элементов массива, не используя боль- ших объемов дополнительной памяти. В частности, алгоритм Кронрода (Kronrod) для слияния в памяти рассматривается в книге [Кпи98]. Рандомизированные алгоритмы рассматриваются более подробно в книгах [MR95] и [MU05], Задача подбора болтов и гаек была впервые представлена в книге [Raw92]. Сложный, но детерминированный алгоритм для ее решения с временной сложностью O(nlogn) рассматривается в книге [KMS96].
Гпава 4. Сортировка и поиск 161 Более подробное обсуждение алгоритмов типа "разделяй и властвуй" можно найти в книгах [CLRSOl], [КТ06] и |Мап89]. Отличный обзор основной теоремы дается в книге [CLRSOl]. 4.11. Упражнения Применение сортировки I. [3] Вам нужно разделить 2и игроков на две команды по п игроков в каждой. Каждому игроку присвоен числовой рейтинг, указывающий его игровые способности. Нужно раз- делить игроков наиболее несправедливым способом, т. е. создать самое большое нера- венство игровых способностей между командой Л и командой В Покажите, как можно решить данную задачу за время <9(«log«). 2. [3] Для каждой из следующих задач предоставьте алгоритм, который находит требуемые числа за данное время. Чтобы уменьшить объем решений, можете свободно использо- вать алгоритмы из этой книги в качестве процедур. Например, для множества S = {6, 13, 19 ,3, 8} разность 19-3 максимальна, а разность 8-6 — минимальна. а) Пусть 5— неотсортированный массив и целых чисел. Предоставьте алгоритм для поиска пары элементовх,у е Sc наибольшей разностью |х-у|. Время исполнения алго- ритма в наихудшем случае должно быть равным О(и). б) Пусть 5— отсортированный массив п целых чисел. Предоставьте алгоритм для по- иска пары элементов х,у g 5 с наибольшей разностью |х-у|. Время исполнения алго- ритма в наихудшем случае должно быть равным ()( I). в) Пусть S— неотсортированный массив и целых чисел. Предоставьте алгоритм для поиска пары элементов х, у е Sc наименьшей разностью |х-у| для х^у. Время исполне- ния алгоритма в наихудшем случае должно быть равным O(nlogn). г) Пусть .9— отсортированный массив п целых чисел. Предоставьте алгоритм для поис- ка пары элементов х, у е Sc наименьшей разностью |х-у| для х±у. Время исполнения алгоритма в наихудшем случае должно быть равным (9(и). 3. [3] Для входной последовательности из 2и действительных чисел разработайте алгоритм с временем исполнения 6>(nlog>7), который разбивает эти числа на п пар таким образом, чтобы минимизировать максимальную сумму значений в парах. Например, рассмотрим множество чисел {1, 3, 5, 9}. Эти числа можно разбить на следующие наборы: ({I, 3}, {5, 9}), ({1, 5},{3, 9}) и ({1, 9},{3, 5}). Суммы значений пар в этих наборах равны (4, 14), (6. 12) и (10, 8). Таким образом, в третьем наборе максимальная сумма равна 10. что яв- ляется минимумом для всех наборов. 4. [3] Допустим, что у нас есть п пар элементов, где первый член пары является числом, а второй — одним из трех цветов: красный, синий или желтый. Эти пары элементов отсор- тированы по числу. Разработайте алгоритм с временем исполнения О(л) для сортировки пар элементов по цвету (красный предшествует синему, а синий желтому) таким обра- зом, чтобы сохранить сортировку по числам для одинаковых цветов. Например: последовательность (I, синий), (3, красный), (4, синий), (6, желтый), (9. крас- ный) после сортировки будет такой: (3, красный), (9, красный), (1, синий), (4, синий), (6, желтый). 5. [3] Модой набора чисел называют число с наибольшим количеством вхождений в набор. Например, модой набора (4, 6, 2, 4, 3, 1) является число 4. Разработайте эффективный алгоритм поиска моды набора из п чисел. 6 Зак 3741
162 Часть I. Практическая разработка алгоритмов 6. [3] Дано: два набора элементов S| и S2 (оба размером п) и число г. Опишите алгоритм с временем исполнения O(nlogn) для определения, существует ли пара элементов, один из набора а другой из набора S2, сумма которых равна х. (Чтобы решение было зачтено частично, алгоритм может иметь время исполнения 0(/?2).) 7. [3] Предложите краткое описание метода для решения каждой из следующих задач. Укажите степень сложности в наихудшем случае для каждого из ваших методов. а) Вам дали огромное количество телефонных счетов и столь же огромное количество чеков по оплате этих счетов. Узнайте, кто не оплатил свой телефонный счет. б) Вам дали список всех книг школьной библиотеки, в котором указаны название каж- дой книги, ее автор, идентификационный номер и издательство. Также вам дали список 30 издательств. Узнайте, сколько книг в библиотеке были выпущены каждым издатель- ством. в) Вам дали все карточки регистрации выдачи книг из библиотеки института за послед- ний год, на каждой из которых указаны имена читателей, бравших данную книгу. Опре- делите, сколько людей брали из библиотеки хотя бы одну книгу. 8. [4] Имеется набор S, содержащий п действительных чисел, и действительное число х. Разработайте алгоритм для определения, содержит ли набор S два таких элемента, сум- ма которых равна х. а) Допустим, что набор 5 не отсортирован. Разработайте алгоритм для решения задачи за время O(/7logn). б) Допустим, что набор .S' отсортирован. Разработайте алгоритм для решения задачи за время О(п). 9. [4] Разработайте эффективный алгоритм для вычисления объединения множеств А и В, где п = max(|J|,|£|). Выход должен быть представлен в виде массива элементов, обра- зующих объединение множеств и входящих в это объединение больше, чем один раз. а) Допустим, что множества А и В не отсортированы. Разработайте алгоритм для реше- ния задачи за время O(wlogn). б) Допустим, что множества А и В отсортированы. Разработайте алгоритм для решения задачи за время О(п). Ю. [5] Имеется набор S, содержащий п целых чисел, и целое число Т. Разработайте алго- ритм с временем исполнения O(nk 'log/?) для определения, равна ли сумма двух целых чисел из 5 целому числу Т. II. [6] Разработайте алгоритм с временем исполнения О(/?), позволяющий найти все эле- менты, входящие больше чем и/2 раза в список из п элементов. Потом разработайте ал- горитм с временем исполнения О(/?)_ позволяющий найти все элементы, входящие в список из п элементов больше чем /?/4 раза. Пирамиды 12. [3] Разработайте алгоритм для поиска к. наименьших элементов в неотсортированном наборе из п целых чисел за время О(п + Alogn). 13. [5] Вы можете сохранить набор из п чисел в виде невозрастающей бинарной пирамиды или отсортированного массива. Для каждой из перечисленных задач укажите, какая из
Гпава 4. Сортировка и поиск 163 этих структур данных является лучшей, или не имеет значения, какую из них использо- вать. Обоснуйте свои ответы. а) Найти наибольший элемент. б) Удалить элемент. в) Сформировать структуру. г) Найти наименьший элемент. 14. [5] Разработайте алгоритм с временем исполнения 6?(??logA) для слияния к отсортиро- ванных списков с общим количеством и элементов в один отсортированный список. (Подсказка: используйте пирамиду, чтобы ускорить работу простейшего алгоритма со временем исполнения О(кп).) 15. [5] а) Разработайте эффективный алгоритм для поиска второго по величине элемента из и элементов. Задача решается меньше чем за In - 3 сравнений. б) Потом разработайте эффективный алгоритм для поиска третьего по величине эле- мента из п элементов. Сколько операций сравнения выполняет ваш алгоритм в наихуд- шем случае? Приходится ли вашему алгоритму в процессе работы находить максималь- ный и второй по величине элементы? Быстрая сортировка 16. [3] Используя применяемый в быстрой сортировке принцип разбиения основной задачи на меньшие подзадачи, разработайте, алгоритм для определения срединного элемента (median) массива п целых чисел с ожидаемым временем исполнения О(п). (Подсказка: нужно ли исследовать обе стороны раздела?) 17. [3] Срединным элементом п значений является [~и/ 2"| -е наименьшее значение. а) Допустим, что алгоритм быстрой сортировки всегда выбирает в качестве элемента- разделителя срединное значение текущего подмассива. Сколько операций сравнений выполнит алгоритм быстрой сортировки в наихудшем случае при таком условии? б) Допустим, что алгоритм быстрой сортировки всегда выбирает в качестве элемента- разделителя Г«/з"|-е наименьшее значение текущего подмассива. Сколько операций сравнений выполнит алгоритм быстрой сортировки в наихудшем случае при таком условии? 18. [5] Дан массив А из п элементов, каждый из которых окрашен в один из трех цветов: красный, белый или синий. Нужно отсортировать элементы по цвету в следующем по- рядке: красные, белые, синие. Разрешены только две операции: • examine(A.i) — возвращает цвет z-ro элемента массива А. • swap(A.ij) — меняет местами 7-й и /-й элементы массива А. Создайте эффективный алгоритм сортировки элементов в указанном порядке за линей- ное время. 19. [5] Инверсией перестановки является пара элементов, расположенных в неправильном порядке. а) Покажите, что в перестановке из п элементов может быть самое большее п(п -1)/2 инверсий. Какая перестановка (или перестановки) может содержать точно п(п- 1)/2 ин- версий?
164 Часть I Практическая разработка алгоритмов б) Допустим, что Р является перестановкой, а Р' — инволюция этой перестановки. По- кажите, что Р и Рг содержат в точности п(п -1)/2 инверсий. (в ) На основе предыдущего результата докажите, что ожидаемое количество инверсий в произвольной перестановке равно «(/? - 1)/4. 20. [3] Предоставьте эффективный алгоритм для упорядочивания п элементов таким обра- зом, чтобы все отрицательные элементы находились перед всеми положительными эле- ментами. Использование вспомогательного массива для временного хранения элемен- тов не разрешается. Определите время исполнения вашего алгоритма. Другие алгоритмы сортировки 21. [5] Устойчивыми называются такие алгоритмы сортировки, которые оставляют элемен- ты с одинаковыми ключами в таком же порядке, в каком они находились до сортиров- ки. Объясните, что нужно сделать, чтобы обеспечить устойчивость алгоритма сорти- ровки слиянием. 22. [3] Продемонстрируйте, что и положительных целых чисел в диапазоне от 1 до к можно отсортировать за время O(nlog^). Интересен случай к«н 23. [5] «Нам нужно отсортировать последовательность S из п целых чисел, содержащую много дубликатов; количество различных целых чисел в 5 равно (7(log/?). Разработайте алгоритм для сортировки таких последовательностей с временем исполнения в наихуд- шем случае 0(//log log/?). 24. [5] Пусть Л[1../?]— массив, в котором первые /?->//? элементов уже отсортированы. О порядке остальных элементов нам ничего не известно. Разработайте алгоритм для сортировки массива А за значительно лучшее время, чем «log/?. 25. [5] Допустим, что массив Л[1..«] может содержать числа из множества {1, ..., /?2}, но в действительности содержит самое большее loglog/? этих чисел. Разработайте алгоритм для сортировки массива А за время значительно меньшее, чем O(«log«). 26. [5] Необходимо отсортировать последовательность нулей (0) и единиц (1) посредством сравнений. При каждом сравнении значений х и у алгоритм определяет, какое из сле- дующих отношений имеет место: х < у, х = у или х > у. а) Разработайте алгоритм для сортировки данной последовательности за и - 1 сравне- ний в наихудшем случае. Докажите, что ваш алгоритм является оптимальным. б) Разработайте алгоритм для сортировки данной последовательности за 2/?/3 сравнений в среднем случае (допуская, что каждый из >? элементов может быть 0 или 1 с одинако- вой вероятностью). Докажите, что ваш алгоритм является оптимальным. 27. [6] Пусть Р— простой, но не обязательно выпуклый, многоугольник, aq— произволь- ная точка, не обязательно находящаяся в Р. Разработайте эффективный алгоритм поиска прямой линии, начинающейся в точке q и пересекающей наибольшее количество ребер многоугольника Р. Иными словами, в каком направлении нужно целиться из ружья, на- ходясь в точке q, чтобы пуля пробила наибольшее количество стен? Прохождение пули через одну из вершин многоугольника Р засчитывается, как пробивание только одной стены. Для решения этой задачи возможно создание алгоритма с временем исполнения O(nlog«).
Глава 4. Сортировка и поиск 165 Нижние пределы 28. [5] В одной из моих научных статей (см. книгу [Ski88]) я привел пример алгоритма сор- тировки методом сравнений с временной сложностью О(п log(Vtf)). Почему такой алгоритм оказался возможным, несмотря на то, что нижний предел сор- тировки равен Q(zzlogw)? 29. [5] Один из ваших студентов утверждает, что он разработал новую структуру данных для очередей с приоритетами, которая поддерживает операции insert, maximum и extract-max, с временем исполнения в наихудшем случае 0(1) для каждой. Докажите, что он ошибается. (Подсказка: для доказательства просто подумайте, каким образом та- кое время исполнения отразится на нижнем пределе времени исполнения сортировки, равном Q(nlogzz).) Поиск 30. [3] База данных содержит записи о 10 000 клиентов в отсортированном порядке. Из них сорок процентов считаются хорошими клиентами, т. е. на них приходится в сумме 60% обращений к базе данных. Такую базу данных и поиск в ней можно реализовать двумя способами' • Поместить все записи в один массив и выполнять поиск требуемого клиента посред- ством двоичного поиска. • Поместить хороших клиентов в один массив, а остальных в другой. Двоичный поиск сначала выполняется в первом массиве, и только в случае отрицательного результа- та — во втором. Выясните, какой из этих подходов дает лучшую ожидаемую производительность. Будут ли результаты иными, если в обоих случаях вместо двоичного поиска применить ли- нейный поиск в неотсортированном массиве? 31. [3] Допустим, имеется массив А отсортированных и чисел, циклически сдвинутых впра- во на к позиций. Например, последовательность {35, 42, 5, 15, 27, 29} представляет со- бой отсортированный массив, циклически сдвинутый вправо на к = 2 позиций, а после- довательность {27,29, 35,42, 5, 15} — массив, циклически сдвинутый на к = 4 позиций, а) Допустим, что значение к известно. Разработайте алгоритм с временем исполнения <7(1) для поиска наибольшего числа в массиве А. б) Допустим, что значение к неизвестно. Разработайте алгоритм с временем исполнения O(lgn) для поиска наибольшего числа в массиве А. Чтобы решение было зачтено час- тично, алгоритм может иметь время исполнения О(п). 32. [3] В игре "20 вопросов" первый игрок задумывает число от 1 до и. Второй игрок дол- жен угадать это число, задав как можно меньше вопросов, требующих ответа "да" или "нет". Предполагается, что оба играют честно. а) Какой должна быть оптимальная стратегия второго игрока, если число п известно? б) Какую стратегию можно применить, если число п неизвестно? 33. [5] Дана отсортированная последовательность {гц, а2.а,,} разных целых чисел. Раз- работайте алгоритм с временем исполнения O(lgw) для определения, содержит ли мас- сив такой индекс /, для которого а, = /. Например, в массиве {-10, -3, 3, 5, 7}, сп, = 3. Но массив {2,3,4, 5, 6, 7} не имеет такого индекса /.
166 Часть I. Практическая разработка алгоритмов 34. [5] Дана отсортированная последовательность {о,, а2, а,,} разных целых чисел в диа- пазоне от I до /и, где п < т. Чтобы решение было зачтено частично, разработайте алго- ритм с временем исполнения O(lgz?) для поиска целого числа х < т, отсутствующего в этой последовательности. Чтобы решение было засчитано полностью, алгоритм должен находить наименьшее такое целое число 35. [5] Пусть М— матрица размером п * т, в которой элементы каждой строки отсортиро- ваны в возрастающем порядке слева направо, а элементы каждого столбца отсортиро- ваны в возрастающем порядке сверху вниз. Разработайте эффективный алгоритм для определения местонахождения целого числа х в матрице М или для определения, что матрица не содержит данное число. Сколько сравнений числа х с элементами матрицы выполняет ваш алгоритм в наихудшем случае? Задачи по реализации 36. [5] Возьмем двумерный массив А размером п х п. содержащий целые числа (положи- тельные, отрицательные и ноль). Допустим, что элементы в каждой строке данного мас- сива отсортированы в строго возрастающем порядке, а элементы каждого столбца — в строго убывающем. (Соответственно, строка или столбец не может содержать двух нулей.) Опишите эффективный алгоритм для подсчета вхождений элемента 0 в массив А. Выполните анализ времени исполнения данного алгоритма. 37. [6] Реализуйте несколько различных алгоритмов сортировки, таких как сортировка ме- тодом выбора, сортировка вставками, пирамидальная сортировка, сортировка слиянием и быстрая сортировка. Экспериментальным путем оцените сравнительную производи- тельность этих алгоритмов в простом приложении, считывающим текстовый файл большого объема и отмечающим только один раз каждое встречающееся в нем слово. Это приложение можно реализовать, отсортировав все слова в тексте, после чего про- сканировав отсортированную последовательность для определения одного вхождения каждого отдельного слова. Напишите краткий доклад с вашими выводами. 38. [5] Реализуйте алгоритм внешней сортировки, использующий промежуточные файлы для временного хранения файлов, которые не помещаются в оперативную память. В ка- честве основы для такой программы хорошо подходит алгоритм сортировки слиянием. Протестируйте свою программу как на файлах с записями малого размера, так и на файлах с записями большого размера. 39. [8] Разработайте и реализуйте алгоритм параллельной сортировки, распределяющий данные по нескольким процессорам. Подходящим алгоритмом будет разновидность ал- горитма сортировки слиянием. Оцените ускорение работы алгоритма с увеличением ко- личества процессоров. Потом сравните время исполнения этого алгоритма с временем исполнения реализации чисто последовательного алгоритма сортировки слиянием. Ка- ковы ваши впечатления? Задачи, предлагаемые на собеседовании 40. [3] Какой алгоритм вы бы использовали для сортировки миллиона целых чисел? Сколь- ко времени и памяти потребует такая сортировка? 41. [3] Опишите преимущества и недостатки наиболее популярных алгоритмов сортировки. 42. [3] Реализуйте алгоритм, который возвращает только однозначные элементы массива.
Гпава 4 Сортировка и поиск 167 43. [5] Как отсортировать файл размером в 500 Мбайт на компьютере, оснащенным опера- тивной памятью размером всего лишь в 2 Мбайт? 44. [5] Разработайте стек, поддерживающий выполнение операций занесения в стек, снятия со стека и извлечения наименьшего элемента. Каждая операция должна иметь постоян- ное время исполнения. Возможна ли реализация стека, удовлетворяющего этим требо- ваниям9 45. [5] Дана строка из трех слов. Найдите наименьший (т. е. содержащий наименьшее коли- чество слов) отрывок документа, в котором присутствуют все три слова. Предоставля- ются индексы расположения этих слов в строках поиска, например wordl: (1,4.5), word2: (4,9.10) и word3: (5,6,15). Все списки отсортированы. 46. [6] Есть 12 монет, одна из которых тяжелее или легче, чем остальные. Найдите эту монету, выполнив лишь три взвешивания. Задачи по программированию Эта задачи доступны на сайтах http://www.programming-challenges.com и http:// uva.onlinejudge.org. 1. Vito’s Family. 110401/10041. 2. Stacks of Flapjacks. 110402/120. 3. Bridge. 110403/10037. 4. ShoeMaker's Problem. 110405/10026. 5. ShellSort. 110407/10152.
ГЛАВА 5 Обход графов Графы являю1ся одной из важнейших областей теории вычислительных систем. Это абстрактное понятие, посредством которого можно описывать разнообразные реальные явления— организацию транспортных систем, человеческих взаимоотношений, сети передачи данных и т. п. Возможность формального моделирования такого множества разных реальных структур позволяет программисту решать широкий круг прикладных задач. Более конкретно, граф G' = (И. Е) состоит из набора вершин Ии набора ребер Е, соеди- няющих пары вершин. С помощью графов можно представить практически чюбые взаимоотношения. Например, посредством графов можно создать модель сети дорог, представляя населенные пункты вершинами, а дороги между ними — соединяющими соответствующие вершины ребрами. С помощью графов можно также моделировать электронные схемы, представляя компоненты вершинами, а соединения компо- нентов — ребрами. Такие графы изображены на рис. 5.1 Кис. 5.1. Моделирование электронных схем (а) и сети дорог (б) посредством графов Представление задачи в виде графа является ключевым подходом к решению многих алгоритмических задач. Теория графов предоставляет язык для описания свойств взаи- моотношений, и просто поразительно, как часто запутанные прикладные задачи под- даются простому описанию и решению посредством применения свойств классических графов. Разработка по-настоящему оригинальных алгоритмов графов является очень трудной задачей. Ключевым аспектом эффективного использования алгоритмов на графах яв- ляется правильное моделирование задачи, чтобы можно было воспользоваться уже су- ществующими алгоритмами. Знакомство с разными типами алгоритмических задач
Гпава 5 Обход графов 169 важнее знания деталей отдельных алгоритмов, особенно если учесть, что во второй части этой книги можно найти решение задачи по ее названию. В этой главе рассматриваются основные структуры данных и операции обхода графов, которые можно будет использовать при поиске решений базовых задач на графах. Вглже 6 рассматриваются более сложные алгоритмы для работы с графами: построе- ние минимальных остовных деревьев (minimum spanning tree), кратчайшего пути и по- токов в сети. При этом правильное моделирование задачи остается наиболее важным аспектом ее решения. Затратив время на ознакомление с задачами и их решениями во второй части этой книги, вы сэкономите много времени при решении реальных задач в будущем. 5.1. Разновидности графов Графом называется упорядоченная пара множеств G = (ЕЕ). где V— подмножество вершин (или узлов), а Е = Г* V—упорядоченное или неупорядоченное подмножество ребер, соединяющих пары вершин из V. В моделировании дорожной сети вершины представляют населенные пункты (или пересечения), определенные пары которых со- единены дорогами (ребрами). В анализе исходного кода компьютерной программы вершины могут представлять операторы языка, а ребра будут соединять операторы х иу. если у выполняется после х. В анализе человеческих взаимоотношений вершины представляют отдельных людей, а ребра соединяют тех, кто близок друг другу. На выбор конкретного типа структуры данных для представления графов и алгоритмов для работы с ними оказывают влияние несколько фундаментальных свойств графов. Поэтому первым шагом в решении любой задачи на графах будет определение подхо- дящего типа графа. На рис. 5.2 показано несколько основных типов графов, а далее дано их краткое описание ♦ Неориеитированные/ориентированные. Граф G = (К Е) является неориентирован- ным (undirected), если из (х. у) е Е следует, что (у, х) также является членом Е. В противном случае говорят, что граф ориентированный (directed). Дорожные сети между городами обычно неориентированные, т. к. по любой обычной дороге можно двигаться в обоих направлениях. А вот дорожные сети внутри городов почти всегда ориентированы, т. к. в большинстве городов найдется, по крайней мере, несколько улиц с односторонним движением. Графы последовательности исполнения про- грамм обычно ориентированы, т. к. программа исполняется от одной строчки кода к другой в одном направлении и меняет направление исполнения только при ветв- лениях. Большинство графов, рассматриваемых в теории графов, являются неори- ентированными. ♦ Взвешенные/невзвешенные. Каждому ребру (или вершине) взвешенного (weighted) графа G присваивается числовое значение, или вес. Например, в зависимости от приложения, весом ребер графа дорожной сети может быть их протяженность, мак- симальная скорость движения, пропускная способность и т. п. Разные вершины и ребра невзвешенных (unweighted) графов не различаются по весу. Разница между взвешенными и невзвешенн'ымн графами становится особенно оче- видной при поиске кратчайшего маршрута между двумя вершинами. В случае не-
170 Часть I. Практическая разработка алгоритмов взвешенных графов самый короткий маршрут должен состоять из наименьшего ко- личества ребер, и его можно найти посредством поиска в ширину, рассматриваемо- го далее в этой главе. Поиск кратчайшего пути во взвешенных графах требует ис- пользования более сложных алгоритмов и рассматривается в главе 6. нсвзвсшенныи разреженный плотный циклическим ациклическим вложенный топологическим помеченный неявный явный непомеченный Рис. 5.2. Основные разновидности графов неориентированный ориентированный простои сложным 12 взвешенный ♦ Простые/сложные Наличие ребер некоторых типов затрудняют работу с графами. Петлей (self-ioop, loop) называется ребро (х, х), т. е. ребро, имеющее только одну вершину. Кратными (multiedge) называются ребра, соединяющие одну и ту же пару вершин (х,у). Наличие в графе обеих этих структур требует особого внимания при реализации ал- горитма работы с графом, поэтому графы, которые не содержат их, называются простыми (simple) или обыкновенными. ♦ Разреженные/плотные. Граф является разреженным (sparse), когда в действитель- ности ребра определены только для малой части возможных пар вершин (('2') для простого неориентированного графа из п вершин). Граф, у которого ребра опреде- лены для большей части возможных пар вершин, называется плотным (dense). Не существует четкой границы между разреженными и плотными графами; но, как правило, у плотных графов отношение количества ребер к количеству вершин обычно описывается квадратичной функцией, а у разреженных —линейной. Как правило, разреженность графов диктуется конкретным приложением. Графы дорожных сетей должны быть разреженными по причине физического и практиче-
Гпава 5. Обход графов 171 ского ограничения на количество дорог, которые могут пересекаться в одной точке. На самом ужасном перекрестке, который мне когда-либо приходилось проезжать, сходилось всего лишь семь дорог. Подобным образом количество проводников электрической или электронной схемы, которые можно соединить в одной точке, также ограничено, за исключением, возможно, проводов питания или заземления. ♦ Циклические/ацикчические. Ациклический (acyclic) граф не содержит циклов. Де- ревья являются связными ациклическими неориентированными графами. Деревья представляют собой самые простые графы, рекурсивные по своей природе, т. к., ра- зорвав любое ребро, мы получим два меньших дерева. Для обозначения ориентированных ациклических графов часто используется аббре- виатура DAG (directed acyclic graph). Графы DAG часто возникают в задачах кален- дарного планирования, где ориентированное ребро (х.у) обозначает, что мероприя- тие* должно выполниться раньше, чем мероприятие_р. Для соблюдения таких огра- ничений на предшествование вершины графа DAG упорядочиваются операцией, называющейся топологической сортировкой (topological sort). Топологическая сор- тировка обычно является первой выполняемой операцией в любом алгоритме на графе DAG (подробности см. в разделе 5.10.1). ♦ Вложенные/топологические. Граф является вложенным (embedded), если его вер- шинам и ребрам присвоены геометрические позиции. Таким образом, любое визу- альное изображение графа является укладкой (embedding), но это обстоятельство необязательно является важным для алгоритма. Иногда структура графа полностью определяется геометрией его укладки. Напри- мер, если у нас имеется набор точек в плоскости и мы ищем самый короткий мар- шрут их обхода (т. е. решаем задачу коммивояжера), то в основе решения будет ле- жать полный граф (complete), т. е. граф, в котором каждая вершина соединена со всеми остальными. Веса обычно определяются расстоянием между каждой парой точек. Другим примером топологии графа, имеющей геометрическое происхождение, яв- ляются решетки точек. В большинстве задач выполняется обход соседних точек решетки, вследствие чего ребра определяются неявно, исходя из геометрии. ♦ Неявные/явные. Некоторые графы не создаются заранее с целью последующего об- хода, а возникают по мере решения задачи. Хорошим примером такого графа будет граф поиска с возвратом. Вершины этого неявного графа поиска представляют со- стояния вектора поиска, а ребра соединяют пары состояний, которые можно гене- рировать непосредственно друг из друга. Часто бывает, что работать с неявным графом проще (в силу отсутствия необходимости сохранять его полностью), чем яв- ным образом создавать граф для последующего анализа. ♦ Помеченные/непомеченные. В помеченном (labeled) графе каждая вершина имеет свою метку (идентификатор), что позволяет отличать ее от других вершин. В непо- меченных (unlabeled) графах такие обозначения не применяются. Вершины графов, возникающие в прикладных задачах, часто помечаются значащи- ми метками, например, названиями городов в графе транспортной сети. Распро- страненной задачей является проверка на изоморфизм, т. е. выяснение, является ли
172 Часть I Практическая разработка алгоритмов топологическая структура двух графов идентичной, если игнорировать метки гра- фов. Такие задачи обычно следует решать методом поиска с возвратом, пытаясь присвоить каждой вершине каждого графа такую метку, чтобы структуры были идентичными. 5.1.1. Граф дружеских отношений Чтобы продемонстрировать важность моделирования задачи должным образом, рас- смотрим граф, в котором вершины представляют людей, а вершины соединяются реб- рами в тех случаях, когда между обозначаемыми соответствующими вершинами людьми существуют дружеские отношения. Такие графы называются сги/пачьныл/н сетями и их можно четко определить для любого набора людей, будь то ваши соседи, однокурсники или коллеги, или жители всего земного шара. В последние годы возник- ла целая наука анализа социальных сетей, т. к. многие интересные аспекты поведения людей лучше всего поддаются пониманию в виде свойств графа дружеских отноше- ний. Пример графа дружеских отношений между людьми представлен на рис. 5.3. Рис. 5.3. Пример графа дружеских отношений Для решения реальных задач в большинстве случаев применяются графы разреженного типа. Граф дружеских отношений является хорошим примером разреженных графов. Даже самый общительный человек в мире знает лишь незначительную часть населения планеты. Мы воспользуемся графами дружеских отношений, чтобы продемонстрировать опи- санную ранее терминологию графов. Понимание терминологии является важной частью умения работать с графами. ♦ Если я твой друг, то означает ли это. что ты мой друг? Иными словами, мы хо- тим выяснить, является ли граф ориентированным. Граф называется неориентиро- ванным. если существование ребра (х. у) всегда влечет за собой существование реб- ра (у, х). В противном случае говорят, что граф является ориентированным. Граф типа "знаю о нем/ней" является ориентированным, т. к. каждый из нас знает о мно- гих людях, которые ничего не знают о нас. Граф типа "провел ночь с ним/ней" предположительно является неорйентированным, т. к. для такого занятия требуется партнер. Лично мне хотелось бы, чтобы граф дружеских отношений также был не- ориентированным. ♦ Насколько ты хороший друг? Во взвешенных графах каждому ребру присваивается числовой атрибут. Мы можем смоделировать уровень дружеских отношений, при-
Глава 5. Обход графов 173 своив каждому ребру соответствующее числовое значение, например, от-10 (враги) до 10 (родные братья). Также, в зависимости от приложения, весом ребер графа до- рожной сети может быть их протяженность, максимальная скорость движения, про- пускная способность и т. п. Граф называется невзвешенным, если все его ребра имеют одинаковый вес. ♦ Друг ли я сам себе? Иными словами, мы хотим выяснить, является ли граф простым, т. е. не содержит петель и кратных ребер. Петлей называется ребро типа (х, х). Ино- гда люди поддерживают несколько типов отношений друг с другом. Например, возможно, х и у были однокурсниками в институте, а сейчас работают вместе на од- ном предприятии. Такие взаимоотношения можно моделировать посредством крат- ных ребер, снабженных разными метками. Так как с простыми графами легче работать, то нам может быть лучше объявить, что никто не является другом самому себе. ♦ У кого больше всех друзей? Степенью вершины называется количество ее ребер. В графе дружеских отношений наиболее общительный человек будет представлен вершиной наивысшей степени, а одинокие отшельники — вершинами нулевой сте- пени. В плотных графах большинство вершин имеет высокую степень, в отличие от раз- реженных графов, в которых количество ребер сравнительно небольшое. В регу- лярных. или однородных, графах все вершины имеют одинаковую степень. Регу- лярный граф дружеских отношений представляет по-настоящему предельный уро- вень социальности. ♦ Живут /и мои друзья рядом со мной? Социальные сети в большой степени находят- ся под влиянием географического фактора. Многие из наших друзей являются тако- выми лишь потому, что они просто живут рядом с нами (например, соседи) или ко- гда-то жили вместе с нами (например, соседи по комнате в студенческом общежи- тии). Таким образом, для полного понимания социальной сети требуется вложенный граф, где каждая вершина связана с географической точкой, в которой живет кон- кретный член сети. Эта географическая информация может быть и не закодирована в графе явным образом, но тот факт, что граф дружеских отношений по своей при- роде является вложенным, влияет на нашу интерпретацию любого анализа. ♦ О, ты ее тоже знаешь? Службы социальных сетей, такие как MySpace или LinkedIn, основаны на явном определении связей между членами этих сетей и их друзьями, которые также являются членами. Графы таких социальных сетей состоят из направленных ребер от члена (вершины) х, заявляющего о своей дружбе с другим членом, к этому члену (вершине) у. С другой стороны, полный граф дружеских отношений всего населения Земли явля- ется неявным по той причине, что каждый знает, кто его друг, но не может знать о дружеских отношениях других людей. Гипотеза шести шагов утверждает, что лю- бые два человека в мире (например, профессор Скиена и президент США) связаны короткой цепочкой из промежуточных людей, но не предоставляет никакой инфор- мации. как именно определить эту связывающую цепочку. Самый короткий извест- ный мне путь содержит три звена— Стивен Скиена— Боб МакГрэт — Джон Мар-
174 Часть I. Практическая разработка алгоритмов бергер — Джордж У. Буш (Steven Skiena — Bob McGrath — John Marberger — George W. Bush). Но может существовать и более короткий, неизвестный мне, путь, например, если Джордж Буш учился в колледже вместе с моим дантистом. Но т. к. граф дружеских отношений является неявным, то проверить эту или другие воз- можные цепочки не так-то просто. ♦ Вы действительно личность или всего лишь лицо в толпе? Этот вопрос сводится к тому, является ли граф дружеских отношений помеченным или нет. То есть, имеет ли каждая вершина метку, отражающую ее личность, и важна ли такая метка для нашего анализа? Во многих исследованиях социальных сетей метки графов не играют большой роли. В качестве метки вершинам графа часто присваивается порядковый номер, что обеспечивает удобство идентификации и в то же время сохраняет анонимность представляемого ею лица. Вы можете протестовать, что номер вас обезличивает, и у вас есть имя, но попробуйте доказать это программисту, который реализует алгоритм. В графе, отображающем распространение инфекционного заболевания, достаточно пометить вершины информацией о том, болен ли данный человек, а его имя роли не играет. Подведение итогов С помощью графов можно моделировать большое разнообразие структур и взаимоотно- шений. Для работы с графами й обмена информацией о них используется терминология теории графов. 5.2. Структуры данных для графов Выбор правильной структуры данных для графа может иметь огромное влияние на производительность алгоритма. Двумя основными структурами данных для графов яв- ляются матрицы смежности (adjacency matrix) и списки смежности (adjacency list), по- казанные на рис. 5.4. I 2 .3 4 5 1 0 10 0 1 2 10 111 .3 0 10 10 4 0 110 1 5 110 10 Рис. 5.4. Матрица смежности и список смежности Предположим, что граф G = (У, Е) содержит п вершин и т ребер. ♦ Матрица смежности. Граф G можно представить с помощью матрицы М размером п * п, где элемент M\i.j} = 1, если (j,j) является ребром графа G, и 0 в противном случае. Таким образом мы можем дать быстрый ответ на вопрос "Содержит ли граф G ребро (z,j)?", а также быстро отобразить вставки и удаления ребер. Но такая мат-
Глава 5. Обход графов 175 рица может потребовать большой объем памяти для графов с большим количеством вершин и относительно небольшим количеством ребер. Рассмотрим, например, граф, представляющий карту улиц Манхэттена. Каждое пе- ресечение улиц отображается на графе в виде вершины, а соседние пересечения со- единяются ребрами. Каким будет размер такого графа? Уличная сеть Манхэттена, по сути, представляет собой решетку из 15 авеню, пересекаемых 200 улицами. Это дает нам около 3 000 вершин и 6 000 ребер, т. к. каждая вершина соседствует с че- тырьмя другими вершинами, а каждое ребро является общим для двух вершин. Эф- фективное сохранение такого скромного объема данных не должно вызывать ника- ких проблем, но матрица смежности заняла бы 3 000 х 3 000 = 9 000 000 ячеек, почти все из которых были бы пустыми. Можно было бы сэкономить некоторый объем памяти, упаковывая в одном слове несколько бит состояния или эмулируя треугольную пирамиду на неориентирован- ных графах. Но с использованием этих методов теряется простота, которая делает матрицу смежности такой привлекательной, и. что более критично, время исполне- ния на разреженны', графах остается по своей сути квадратичным. ♦ Списки смежности. Более эффективным способом представления разреженных графов является использование связных списков для хранения соседствующих вер- шин. Хотя списки смежности требуют использования указателей, вы легко с ними справитесь, когда наберетесь немного опыта работы со связными структурами данных. В списках смежности проверку на присутствие определенного ребра (/,/) в графе G выполнить труднее, т. к. для того, чтобы найти данное ребро, необходимо выпол- нить поиск по соответствующему списку. Однако разработать алгоритм, не нуж- дающийся в таких запросах, на удивление легко. Обычно перебираются все ребра графа при одном его обходе в ширину или глубину, и при посещении текущего реб- ра обновляется его состояние. Преимущества той или иной структуры смежности для представления графов показаны в табл. 5.1. Таблица 5.1. Сравнение матриц и списков смежности Задача Оптимальный вариант Проверка на вхождение ребра (л, у) в граф Матрица смежности Определение степени вершины Списки смежности Объем памяти для небольших графов Списки смежности (т - п), т. к. матрица смежности (п~) Объем памяти для больших графов Матрица смежности (с небольшим преимуществом) Вставка или удаление ребра Матрица смежности 0(1), т. к. списки смежности O(d) Обход графа Списки смежности @(т + и), т. к. матрица смежности (-)(«') Пригодность для решения большинства проблем Списки смежности
176 Часть I Практическая разработка алгоритмов Подведение итогов Для большинства приложений на графах списки смежности являются более подходящей структурой данных, чем матрицы смежности. В этой главе для представления графов мы будем использовать списки смежности. В частности, представление графа осуществляется таким образом: для каждого графа ведется подсчет количества вершин, каждой из которых присваивается идентификаци- онный номер в диапазоне от I до п. Ребра представляются посредством массива связ- ных списков. Соответствующий код показан в листинге 5.1. Листинг 5.1. Реализация графов посредством списков смежности «define MAXV 1000 typedef struct I int у; int weight; struct edgenode *next; } edgenode; typedef struct t edgenode ‘edges[MAXV+1]; int degree[MAXV+1]; int nvertices; int nedges; bool directed; } graph; /* Максимальное количество вершин */ Информация о смежности */ /* Вес реОра, если есть ‘/ /* Следующее реОро в списке */ /‘ Информация о смежности */ /* Степень каждой вершины */ /‘ Количество вершин в графе */ /‘ Количество реОер в графе */ • Граф ориентированный? */ Ориентированное ребро (х.у) представляется структурой типа edgenode в списке смеж- ности. Поле degree содержит степень данной вершины. Неориентированное ребро (.г, у) входит дважды в любою структуру графа на основе смежности — один раз в виде у в списке для х и второй раз в виде х в списке для у. Булев флаг directed указывает, является ли данный граф ориентированным. Для демонстрации использования этой структуры данных мы покажем, как выполнить считывание графа из файла. Типичный формат графа (листинг 5.2) таков: первая строчка содержит количество вершин и ребер в графе, а за ней следуют строчки, в каждой из которых указаны вершины очередного ребра. Листинг 5.2. Формат графа initializejgraph(graph *g, bool directed) ( int 1; /* Счетчик */ g -> nvertices = 0; g -> nedges = 0; g -> directed = directed; for (i=l; i<=MAXV; i++) g->degree[i] = 0; for (i=l; i<=MAXV; i++) g->edges[i] = NULL; Собственно чтение графа заключается во вставке каждого ребра в эту структуру. Соот- ветствующий код представлен в листинге 5.3.
Глава 5. Обход графов 177 Листинг 5.3. Считывание графа read_graph (graph ' rg, bool directed) int i ; int m; int x, y; /* Счетчик */ /★ Количество вершин */ /* Вершины в ребре (х,у) */ initializegraph(g, directed); scant("id ,d", S (g->nvertices), am); for (i=l; i<=m; i++) { scant("%d %d", &x, &y); insert edge(g, x, y, directed); В листинге 5.3 критичной является процедура insert edge. Новый узел edgenode встав- ляется в начало соответствующего списка смежности, т. к. порядок не имеет значения. Процедуре вставки передается параметр в виде булева флага directed для обозначения, сколько копий каждого ребра нужно вставить — одну или две. Код процедуры вставки показан в листинге 5.4. Обратите внимание на использование рекурсии для решения этой задачи. Листинг 5.4. Вставка ребра 1 insert_edge (graph *g, int x, int y, bool directed) edgenode *p; p = malloc(sizeof(edgenode)); p->weight = NULL; p->y = y; p->next = g->edges [x] ; g->edges[x] = p; g->degree[x] ++; if (directed == FALSE) insert_edge(g,y,x,TRUE); else g->nedges ++; /* Временный указатель */ /* Выделяем память для edgenode*/ /* Вставка в начало списка */ Для вывода графа на экран достаточно двух вложенных циклов— один для вершин, другой для их смежных ребер (листинг 5.5). 5.5. Вывод графа на экран printgraph(graph *g) int i; /* Счетчик */ edgenode *p; /* Временный указатель */ for (i=l; i<=g->nvertices; i++) { print!("%d: ",i);
178 Часть I. Практическая разработка алгоритмов р = g->edges[i]; while (р != NULL) 1 printff" %d",p->yi. p p-^-next; i print!("\n"); ) ) Было бы разумно использовать хорошо спроектированный тип данных графа в качест- ве модели для создания своего собственного, а еще лучше, в качестве основы для раз- рабатываемого приложения. Я рекомендую использовать библиотеку типов LEDA (см. раздел 19.1.1) или Boost (см. раздел 19.1.3), которые я считаю лучше всего спроек- тированными структурами данных графов общего назначения, имеющимися в настоящее время. Эти библиотеки, скорее всего, будут более мощными (и, следовательно, будут иметь больший размер и замедлять работу в большей степени), чем вам нужно, но они предоставляют такое количество правильно работающих функций, которые вы. веро- ятнее всего, не сможете реализовать самостоятельно с такой степенью эффективности. 5.3. История из жизни. Жертва закона Мура Я являюсь автором библиотеки алгоритмов для работы с графами, называющейся Combinatorica (www.combinatorica.com), которая предназначена для работы с компью- терной системой алгебраических вычислений Mathematica. Производительность явля- ется большой проблемой в Mathematica, вследствие применяемой в ней аппликативной модели вычислений (в ней не поддерживаются операции записи в массивы с постоян- ным временем исполнения) и накладных расходов на интерпретацию кода (в отличие от компилирования). Код на языке программирования пакета Mathematica обычно ис- полняется от 1 000 до 5 000 раз медленнее, чем код на языке С. Эти особенности данного приложения могут значительным образом замедлять его ра- боту. Что еще хуже. Mathematica занимала большой объем памяти, требуя для эффек- тивной работы целых 4 Мбайт оперативной памяти, что в 1990 году, когда я завершил работу над Combinatorica. было потрясающе большим объемом. При этом любая попытка вычислений на достаточно больших структурах неизбежно вызывала пере- полнение виртуальной памяти. В такой среде мой графический пакет имел надежду эффективно работать только на графах очень небольшого размера (рис. 5.5). Одним из проектных решений, принятых мной вследствие вышеописанных недостат- ков Mathematica, было использование матриц смежности вместо списков смежности в качестве основной структуры данных графов для Combinatorica. Такое решение может сначала показаться странным. Разве в ситуации нехватки памяти не выгоднее исполь- зовать списки смежности, экономя каждый байт? Вообще говоря, да. но ответ не будет таким простым в случае очень малых графов. Представление взвешенного графа, со- стоящего из п вершин и т ребер посредством списка смежности требует приблизи- тельно п + 2т слов, где составляющая 2т отражает требования к памяти для хранения информации о конечных точках и весе каждого ребра. Таким образом, использование списков смежности предоставляет преимущество в отношении объема требуемой па-
Гпава 5 Обход графов 179 мяти только в том случае, если выражение п + 2т значительно меньше, чем /?2. Размер матрицы смежности остается управляемым для п< 100 и. конечно же. для плотных графов этот размер вдвое меньше, чем размер списков смежности. Рис. 5.5. Репрезентативные графы в Combinatonca: пути, не имеющие общих ребер (а). Гамильтонов цикл в гиперкубе (б), обход в глубину дерева поиска (в) Для меня более насущной заботой была необходимость минимизации накладных рас- ходов. обусловленных использованием интерпретируемого языка. Результаты эталон- ных тестов представлены в табл. 5.2. Таблица 5.2. Результаты эталонных тестов старой Combinatorica на рабочих станция S|in начального уровня с 1990 по 2004 г. (время исполнения в секундах) Год 1990 1991 1998 2000 2004 Команда/Машина Sun-3 Sun-4 Sun-5 Ultra 5 Sun Blade PlanarQ[GridGraph[4,4]] 234.10 69,65 27,50 3,60 0,40 Length [Partitions[30]] 289,85 73.20 24,40 3.44 1,58 VertexConnectivity [GridGraph[3,3]] 239,67 47,76 14,70 2.00 0.91 RandomPartitionf 1000] 831,68 267,5 22,05 3.12 0,87 В 1990 году решение двух довольно сложных, но имеющих полиномиальное время вы- полнения. задач на графах из 9 и 16 вершин занимало на моем настольном компьютере несколько минут! Квадратичный размер структуры данных определенно не мог оказать большое влияние на время исполнения этих задач, т. к. 9 * 9 равно всего лишь 81. По своему опыту я знал, что язык программирования пакета Mathematica работает лучше со структурами данных постоянного размера, наподобие матриц смежности, чем со структурами данных, имеющими непостоянный размер, какими являются списки смежности. Тем не менее, несмотря на все эти проблемы с производительностью, библиотека Combinatorica оказалась очень хорошим приложением, и тысячи людей использовали
180 Часть I Практическая разработка алгоритмов этот пакет для всевозможных интересных экспериментов с графами. Combinatorica ни- когда не претендовала на звание высокопроизводительной библиотеки алгоритмов. Большинство пользователей быстро осознало, что вычисления на больших графах не- реальны. но, тем не менее, с энтузиазмом работало с этой библиотекой как с инстру- ментом для математических исследований и среды моделирования. Все были довольны. Но по прошествии нескольких лет пользователи Combinatorica начали спрашивать, по- чему вычисления на графах небольшого размера занимают так много времени. Это ме- ня не удивляло, так как я знал, что моя программа всегда была медленной. Но почему людям потребовались годы, чтобы заметить это? Объяснение заключается в том. что скорость работы компьютеров удваивается при- близительно каждые два года. Это явление носит названия закона Мура. Ожидания пользователей относительно производительности прикладных программ возрастают в соответствии с этими улучшениями в аппаратном обеспечении. Частично из-за того, что Combinatorica была рассчитана на работу со структурами данных графов квадра- тичного размера, она недостаточно хорошо масштабировалась на разреженные графы. С годами требования пользователей становились все жестче, и, наконец, я решил, что Combinatorica нуждается в обновлении Мой коллега. Срирам Пемараджу (Sriram Pemmaraju), предложил мне свою помощь. Через десять лет после первоначального выпуска библиотеки мы (преимущественно он) полностью переписали Combinatorica, воспользовавшись более быстрыми структурами данных. В новой версии Combinatorica для хранения графов используется очень эффективная структура данных в виде списка ребер. Размер списков ребер, как и размер списков смежности, линейно зависит от размера графа (ребра плюс вершины). Это заметно улучшило производительность большинства функций для работы с графами. Повыше- ние производительности особенно драматично для "быстрых" алгоритмов обработки графов. Это алгоритмы с линейным или почти линейным временем исполнения — ал- горитмы обхода графов, топологической сортировки и поиска компонент связности и/или двусвязности. Последствия этой модификации проявляются во всем пакете в ви- де уменьшения времени работы и более экономного расхода памяти. Теперь Combinatorica может обрабатывать графы, которые в 50-100 раз больше, чем те. с ко- торыми могла работать старая версия. На рис. 5.6, а приводится график времени исполнения функции MinimumSpanningTree для обеих версий Combinatorica. Для сравнения производительности новой и старой версий использовались разрежен- ные (решетчатые) графы, разработанные специально, чтобы выделить разницу между этими двумя структурами данных. Да, новая версия библиотеки намного быстрее, но обратите внимание, что разница в производительности становится заметной только для графов больших, чем те, для работы с которыми предназначалась старая версия Combinatorica. Но относительная разница во времени исполнения увеличивается с воз- растанием п. На рис. 5.6, б показано соотношение времени исполнения старой и новой версии в зависимости от размера графа. Разница между линейным и квадратичным ростом размера является асимптотической, поэтому с возрастанием п последствия пе- рехода на новую версию становятся еще более заметными.
Глава 5. Обход графов 181 б) Рис. 5.6. Сравнение производительности старой и новой версий Combinatorica: абсолютное время исполнения для каждой версии (а) и соотношение времен исполнения to) Обратите внимание на странный всплеск на графике при п~ 250. Скорее всего, это следствие перехода между разными уровнями иерархии памяти. Такие явления не ред- кость в современных сложных компьютерных системах. При разработке структур дан- ных производительность кэша должна быть одним из главных, но не важнейшим, при- нимаемых во внимание обстоятельств. Повышение производительности, достигнутое благодаря использованию списков смежности, намного превышает любое улучшение, обусловленное применением кэша. Из нашего опыта разработки и модифицирования библиотеки Combinatorica можно извлечь такие два основных урока. ♦ Чтобы ускорить время исполнения программы, нужно лишь подождать некоторое время. Передовое аппаратное обеспечение со временем доходит до пользователей всех уровней. В результате улучшений в аппаратном обеспечении, имевших место в течение 15 лет. скорость работы первоначальной версии библиотеки Combinatorica возросла больше чем в 200 раз. В этом контексте дальнейшее повышение произ- водительности вследствие модернизации библиотеки является особенно значи- тельным. ♦ Асимптотика в конечном счете является важной. Я не сумел предвидеть развитие технологии, и это было ошибкой с моей стороны. Хотя никто не может предсказы- вать будущее, можно с довольно большой степенью уверенности утверждать, что в будущем компьютеры будут обладать большим объемом памяти и работать быст- рее, чем компьютеры сейчас. Это дает преимущество более эффективным алгорит- мам и структурам данных, даже если сегодня их производительность не намного выше, чем у альтернативных решений. Если сложность реализации вас не останав- ливает, подумайте о будущем и выберите самый лучший алгоритм. 5.4. История из жизни. Создание графа — Только на чтение данных у этого алгоритма уходит пять минут. На получение сколько-нибудь интересных результатов не хватит никакого времени.
182 Часть I Практическая разработка алгоритмов Молодая аспирантка была полна энтузиазма, но не имела ни малейшего понятия о мо- щи правильно выбранных структур данных. Как было описано в разоеле 3.6, мы экспериментировали с алгоритмами для разбиения сетки треугольников на полосы для быстрого рендеринга триангулированных поверх- ностей. Задачу поиска наименьшего количества полос, которые покрывают каждый треугольник в сетке, можно смоделировать как задачу на графах. В таком графе каж- дый треугольник сетки представляется вершиной, а смежные треугольники представ- ляются ребром между соответствующими вершинами Такое представление в виде двойственного графа содержит всю информацию, необходимую для разбиения тре- угольной сетки на полосы треугольников (рис. 5.7). Рис. 5.7. Двойственный граф (пунктирные линии) триангулированной поверхности Первым шагом в разработке программы, которая генерирует хороший набор полос треугольников, является создание двойственного графа триангулированной поверхно- сти. Решение этой задачи я и поручил аспирантке. Через несколько дней она заявила, что одно лишь создание такого графа для поверхности из нескольких тысяч треуголь- ников занимает около пяти минут процессорного времени. — Не может быть! — сказал я. — Ты, должно быть, строишь граф крайне нерацио- нально. Какой формат данных ты используешь? — Вначале идет список ЗО-координат используемых в модели вершин, а за ним — список треугольников. Каждый треугольник описывается списком из трех индексов координат вершин. Вот небольшой пример (листинг 5.6). : Листинг 5.6. Пример описания треугольников VERTICES 4 0.000000 240.000000 0.000000 204.000000 240.000000 0.000000 204.000000 0.000000 0.000000 0.000000 0.000000 0.000000 TRIANGLES 2 0 13 12 3
Гтаеа 5. Обход графов 183 — Понятно. Значит, первый треугольник использует все точки, кроме третьей, т. к. все индексы начинаются с нуля. Два треугольника должны иметь общую сторону, опреде- ляемую точками I и 3. — Так оно и есть. — подтвердила она. — Хорошо. Теперь расскажи мне, как ты строишь двойственный граф. — Когда я знаю количество имеющихся вершин, я могу игнорировать информацию о вершинах. Геометрическое расположение точек не влияет на структуру графа. В мо- ем двойственном графе будет такое же количество вершин, как и треугольников. Я создаю структуру данных в виде списка смежности, содержащего такое количество вершин. При считывании каждого треугольника я сравниваю его со всеми другими треугольниками, чтобы узнать, не имеют ли они две общие конечные точки. Если име- ют, то я добавляю ребро между новым треугольником и этим. Я не мог сдержать раздражение. — Но ведь именно в этом и заключается твоя пробле- ма! Ты сравниваешь каждый треугольник со всеми остальными треугольниками, вследствие чего создание двойственного графа будет квадратичным по числу тре- угольников. Считывание входного графа должно занимать линейное время! — Я не сравниваю каждый треугольник со всеми другими треугольниками. В действи- тельности, в среднем он сравнивается с половиной или с третью других треугольников. — Превосходно. Тем не менее, у тебя алгоритм имеет квадратичную сложность (т. е. Он слишком медленный. Признавать свое поражение она не собиралась. — Вы только критикуете мой алгоритм, а можете ли Вы помочь мне исправить его? Вполне разумно. Я начал думать. Нам был нужен какой-то быстрый способ отбросить большинство треугольников, которые не могли быть смежными с новым треугольни- ком (i,j, к). Что нам в действительности было нужно, так это отдельный список всех треугольников, проходящих через точки /, j и к. По формуле Эйлера для планарных графов средняя точка соединяет меньше чем шесть треугольников. Это позволило бы сравнивать каждый новый треугольник менее чем с двадцатью другими треуголь- никами. — Нам нужна структура данных, состоящая из массива элементов для каждой верши- ны в первоначальном наборе данных. Этим элементом будет список всех треугольни- ков, которые проходят через данную вершину. При считывании нового треугольника мы находим три соответствующих списка в массиве и сравниваем каждый из них с но- вым треугольником. В действительности, нужно проверить только два из этих трех списков, т. к. любые смежные треугольники будут иметь две общие точки. Нам нужно будет добавить в наш граф признак смежности для каждой пары треугольников, имеющей две общие вершины. Наконец, новый треугольник добавляется в каждый из трех обработанных списков, чтобы они были текущими для считывания следующего треугольника. Она немного поразмыслила над этим и улыбнулась.— Понятно. Я сообщу Вам о ре- зультатах. Наследующий день она доложила, что создание графа занимает несколько секунд да- же для больших моделей. Затем она написала хорошую программу для разбиения три- ангулированной поверхности на полосы треугольников, как описано в разделе 3.6.
184 Часть I. Практическая разработка алгоритмов Урок, который следует извлечь из этой истории, заключается в том, что даже такие элементарные задачи, как инициализация структур данных, могут оказаться узким ме- стом при разработке алгоритмов. Большинство программ, работающих с большими объемами данных, должно иметь линейное или почти линейное время исполнения. Та- кие высокие требования к производительности не прощают небрежности. Сфокусиро- вавшись на необходимости получения линейной производительности, вы обычно мо- жете найти соответствующий детерминистический или эвристический алгоритм для решения поставленной задачи. 5.5. Обход графа Самой фундаментальной задачей на графах, возможно, является систематизированное посещение каждой вершины и каждого ребра графа. В действительности, все основные служебные операции по работе с графами (такие как распечатка или копирование гра- фов или преобразования графа из одного представления в другое) являются приложе- ниями обхода графа (graph traversal). Лабиринты обычно представляются в виде графов, где вершины обозначают пересече- ния путей, а ребра— пути лабиринта. Таким образом, любой алгоритм обхода графа должен быть достаточно мощным, чтобы вывести нас из произвольного лабиринта. Чтобы обеспечить эффективность такого алгоритма, мы должны гарантировать, что не будем постоянно возвращаться в одну и ту же точку, оставаясь в лабиринте навечно. А для правильности нашего алгоритма нам нужно выполнять обход лабиринта систем- ным образом, который гарантирует, что мы выберемся из лабиринта. В поисках выхода нам нужно посетить каждую вершину и каждое ребро графа. Ключевая идея обхода графа — пометить каждую вершину при первом ее посещении и помнить о том. что не было исследовано полностью. Хотя в сказках для обозначения пройденного пути использовались такие способы, как хлебные крошки и нитки, в на- ших обходах графов мы будем пользоваться булевыми флагами или перечислимыми типами. Каждая вершина будет находиться в одном из следующих трех состояний: ♦ неоткрытая (undiscovered) — первоначальное, нетронутое состояние вершины; ♦ открытая (discovered)— вершина обнаружена, но мы еще не проверили все инци- дентные ей ребра; ♦ обработанная (processed)— все инцидентные данной вершине ребра были посе- щены. Очевидно, что вершину нельзя обработать до того, как она открыта, поэтому в процес- се обхода графа состояние каждой вершины начинается с неоткрытого, переходит в открытое и заканчивается обработанным. Нам также нужно иметь структуру, содержащую все открытые, но еще не обработан- ные вершины. Первоначально открытой считается только одна вершина— начало об- хода графа. Для полного исследования вершины v нужно изучить каждое исходящее из нее ребро. Если какие-либо ребра идут к неоткрытой вершине л, то эта вершина поме- чается как открытая и добавляется в список для дальнейшей обработки. Ребра, иду- щие к обработанным вершинам, игнорируются, т. к. их дальнейшее исследование не
Гпава 5. Обход графов 185 сообщит нам ничего нового о графе. Также можно игнорировать любое ребро, идущее к открытой, но не обработанной вершине, т. к. эта вершина уже внесена в список вершин, подлежащих обработке. Каждое неориентированное ребро рассматривается дважды, по одному разу при иссле- довании каждой из его вершин. Ориентированные ребра рассматриваются только один раз, при исследовании его источника. В конечном счете, все ребра и вершины в связ- ном графе должны быть посещены. Почему? Допустим, что имеется непосещенная вершина и. чья соседняя вершина v была посещена. Эта соседняя вершина v будет со временем исследована, после чего мы непременно посетим вершину и. Таким образом, мы в конечном счете найдем все. что можно найти. Далее мы обсудим механизм работы алгоритмов обхода графов и важность того, в ка- ком порядке выполняется обход. 5.6. Обход в ширину Обход в ширину (breadth-first traversal) является основой для многих важных алгорит- мов для работы с графами. Далее приводится базовый алгоритм обхода графа в шири- ну. На определенном этапе каждая вершина графа переходит из состояния неоткры- тая в состояние открытая. При обходе в ширину неориентированного графа каждому ребру присваивается направление, от "открывающей" к открываемой вершине. В этом контексте вершина и называется родителем (parent) или предшественником (predecessor) вершины v, а вершина v— потомком вершины и. Так как каждый узел, за исключением корня, имеет только одного родителя, получится дерево вершин графа. Это дерево (рис. 5.8) определяет кратчайший путь от корня ко всем другим узлам графа. Рис. 5.8. Неориентированный граф и ею дерево обхода в ширину б) Это свойство делает обход в ширину очень полезным в решении задач поиска крат- чайшего пути. В листинге 5.7 приводится псевдокод алгоритма обхода графа в ширину. Листинг 5.7. Обход графа в ширину BFS Ю, s for each vertex u e V[G) — {s) do state [ul = "undiscovered" p[u] - nil, т. e. в начале вершины-родители отсутствуют
186 Часть I. Практическая разработка алгоритмов state[s] = "discovered" p[s] - nil = {s} while Q / 0 do u = dequeue[QJ обрабатываем вершину u требуемым образом for each v Adj [u] do □Срабатываем ребро (u, v) требуемым образом if state[v] = "undiscovered" then state[v] = "discovered" p[v] = u enqueue[Q,v] state[u] = "processed" Ребра графа, которые не включены в дерево обхода в ширину, также имеют особые свойства. Для неориентированных графов не попавшие в дерево ребра могут указывать только на вершины на том же уровне, что и родительская вершина, или на вершины, расположенные на уровень ниже. Эти свойства естественно следуют из того факта, что каждое ребро в дереве должно быть кратчайшим путем в графе. А для ориентирован- ных графов ребро (и, v), указывающее в обратном направлении, может существовать в любом случае, когда вершина v расположена ближе к корню, чем вершина и. Реализация Процедура обхода в ширину bfs использует два массива булевых значений для хране- ния информации о каждой вершине графа. Вершина открывается при первом ее посе- щении. Когда все исходящие из вершины ребра были исследованы, то вершина счита- ется обработанной. Таким образом, в процессе обхода состояние каждой вершины на- чинается с неоткрытого, переходит в открытое и заканчивается обработанным. Эту информацию можно было бы хранить с помощью одной переменной перечислимого типа, но мы используем две булевы переменные. bool processed[MAXV+1]; bool discovered[MAXV+1]; int parent[MAXV+1]; /* Обработанные вершины*/ /* Открытые вершины */ /* Отношения открытия */ Сначала каждая вершина помечается как неоткрытая (листинг 5.8). Листинг 5.8. Инициализация вершин initialize_search(graph *g) int i; /* Счетчик */ for (i=l; i<=g->nvertices; i++) [ processed[i] = discovered!!] = FALSE; parent[i] = -1; После открытия вершина помещается в очередь. Так как эти вершины обрабатываются в порядке FIFO, то первыми обрабатываются вершины, поставленные в очередь пер-
Гпава 5. Обход графов 187 выми, т. е. ближайшие к корню. Процедура обхода графа в ширину приводится в лис- тинге 5.9. Листинг 5.9, Обход графа D ширину bfs(graph *g, int start) queue q; /+ Очередь вершин для оОраОотки */ int v; /* Текущая вершина */ int у; /* Следующая вершина */ edgenode *р; /★ Временный указатель */ init_queue(&q); enqueue(&q,start); discovered[start] = TRUE; while (empty_queue(&q) == FALSE) { v = dequeue(&q); process_vertex_early(v); processed[v] = TRUE; p = g->edges[v]; while (p != NULL) { У = p->y; if ((processed[y] — FALSE) g->directed) process_edge(v,y); if (discovered[y] == FALSE) { enqueue (&q, y) ; discovered[y] TRUE; parent[y] = v; J p = p->next; } process_vertex_late(v); 5.6.1. Применение обхода Процедура bfs использует функции process_yertex_early(), process vertex_late О И process edged. С их помощью мы можем настроить действие, предпринимаемое про- цедурой обхода при посещении каждого ребра и вершины. Первоначально вся обра- ботка вершин будет осуществляться при входе, поэтому функция process vertex late() не выполняет никаких действий (листинг 5.10). process_vertex_late (int v) /* Нет действий */
188 Часть I. Практическая разработка алгоритмов Следующие функции предназначены для вывода на экран (или принтер) каждой вер- шины и каждого ребра (листинг 5.11). Листинг 5.11. Функции process_vertex_early () и process_edge () process_vertex_early(int v) ( printf("processed vertex %d\n",v); ) process_edge(int x, int y) ( printf("processed edge (%d, %d)\n", x, y) ; } С помощью функции process edgeO можно также подсчитывать количество ребер (листинг 5.12). Листинг 5.12. Функция process_edge () для подсчета количества ребер process_edge(int х, int у) { nedges - nedges + 1; 1 Разные алгоритмы предпринимают разные действия при посещении вершин и ребер. Использование этих функций предоставляет нам гибкость в настройке требуемого дей- ствия 5.6.2. Поиск путей Массив parent, который строится в процедуре bfs, очень полезен для поиска разнооб- разных путей в графе. Вершина, открывшая вершину /. определяется как parent[i]. Так как в процессе обхода открываются все вершины, то каждая вершина, за исключе- нием корневой, имеет родителя. Родительское отношение определяет "дерево откры- тий", корнем которого является первоначальный узел обхода. Так как вершины открываются в порядке возрастающего расстояния от корня, то это дерево обладает очень важным свойством. Однозначно определенный путь от корня до каждой вершины х е V использует наименьшее количество ребер (или, что эквива- лентно, промежуточных вершин), возможное в любом маршруте графа от корня до вершины х. Этот путь можно воссоздать, следуя по цепи предшественников от вершины х по на- правлению к корню. Обратите внимание, что в этом случае нам нужно двигаться в об- ратном направлении. Мы не можем найти путь от корня к вершине х, потому что указа- тели родителей имеют противоположное направление. Вместо этого путь нужно искать по направлению от вершины х к корню. Так как это направление противоположно то- му, в котором требуется выполнять проход, мы можем либо сохранить путь, а потом явно обратить его с помощью стека, либо переложить эту работу на рекурсивную про- цедуру, показанную в листинге 5.13.
Гпава 5. Обход графов 189 Листинг 5.13. Изменение направления пути посредством рекурсии find_path (int start, int end, int parentsf]) ( if ((start == end) 1 (end — -11) printf("\n d",start); else | lmd_path (start, parents [end], parents) ; printtl" d",end); При обходе в ширину графа, изображенного на рис. 5.8, наш алгоритм выдал следую- щие отношения "вершина/родитель": Вершина I 2 3 4 5 6 Роди гель -1 I 2 5 1 1 Согласно этим родительским отношениям, самый короткий путь от вершины 1 к вер- шине 4 проходит через набор вершин {1,5,4}. При использовании обхода в ширину для поиска кратчайшего пути от вершины х к вершине у нужно иметь в виду следующее: дерево кратчайшего пути полезно только в том случае, если корнем поиска в ширину является вершина х; кроме того, поиск в ширину дает самый короткий путь только для невзвешенных графов. Алгоритмы по- иска кратчайшего пути во взвешенных графах рассматриваются в разделе 6.3.1. 5.7. Применение обхода в ширину Большинство элементарных алгоритмов для работы с графами выполняют один или два обхода графа для получения всей информации о графе. Любой из таких алгорит- мов, если он корректно реализован с использованием списков смежности, обязательно имеет линейное время исполнения, т. к. обход в ширину исполняется за время ()(п + т) как на ориентированных, так и на неориентированных графах. Это оптимальное время, т. к. именно за такое время можно прочитать граф из п вершин и т ребер. Секрет мастерства заключается в умении видеть ситуации, в которых применение об- хода гарантированно даст положительные результаты. Далее приводится несколько примеров использования обхода. 5.7.1. Компоненты связности Граф называется связным (connected), если имеется путь между любыми двумя его вершинами. Гипотеза шести шагов утверждает, что любые два человека в мире связа- ны короткой цепочкой из людей, попарно знакомых друг с другом. Если теория шести шагов правильна, то граф дружеских отношений должен быть связным. Компонентой связности (connected component) неориентированного графа называется максимальный набор его вершин, для которого существует путь между каждой парой
190 Часть I. Практическая разработка алгоритмов вершин. Эти компоненты являются отдельными "кусками" графа, которые не соедине- ны между собой. В качестве примера отдельных компонент связности в графе друже- ских отношений можно привести первобытные племена где-то в джунглях, которые еще не были открыты для остального мира. А отшельник в пустыне или крайне непри- ятный человек будет примером компоненты связности, состоящей из одной вершины. Удивительно, какое большое количество кажущихся сложными проблем сводится к поиску или подсчету компонент связности. Например, вопрос, можно ли решить какую-то головоломку (скажем, кубик Рубика), начав с определенной позиции, по сути, представляет собой вопрос, является ли связным граф разрешенных конфигу- раций. Компоненты связности можно найти с помощью обхода в ширину, т. к порядок перечисления вершин не имеет значения. Начнем обход с первой вершины. Все эле- менты, обнаруженные в процессе этого обхода, должны быть членами одной и той же компоненты связности. Потом повторим обход, начиная с любой неоткрытой вершины (если таковая имеется), чтобы определить вторую компоненту связности, и т. д., до тех пор, пока не будут обнаружены все вершины. Соответствующая процедура приводится в листинге 5.14. Листинг 5.14. Процедура поиска компонент связности сjnnected_components(graph *g) int с; int i; /* Номер компоненты */ /* Счетчик */ initialize search(g); c = 0; for (i=l; i<=g->nvertices; i++) it (discovered[i] == FALSE) { c = c+1; printf("Component %d:",c); bfs(g,i); printf f\n") ; ) ) process_vertex_early(int v) 1 printf(" »d",v); ) process_edge(int x, int y) { ) Обратите внимание на увеличение значения счетчика с, содержащего номер текущей компоненты, при каждом вызове функции bfs. Изменив соответствующим образом действие функции process vertex (). каждую вершину можно было бы явно связать
Глава 5. Обход графов 191 с номером ее компоненты (вместо того, чтобы выводить на экран вершины каждой компоненты). Для ориентированных графов существуют два понятия связности, и это определяет существование алгоритмов поиска компонент сильной и слабой связности. Любую из них можно найти за время О(п + т). что показано в разделе 15.1. 5.7.2. Раскраска графов двумя цветами В задаче раскраски вершин (vertex colouring) требуется присвоить метку (или цвет) каждой вершине графа таким образом, чтобы любые две соединенные ребром верши- ны были разного цвета. Присвоение каждой вершине своего цвета позволяет избежать любых конфликтов. Но при этом нужно использовать как можно меньшее количество цветов. Задачи раскраски вершин часто возникают в приложениях календарного пла- нирования, например, при выделении регистров в компиляторах. Подробное обсужде- ние алгоритмов раскраски вершин можно найти в разделе 16.7. Граф называется двудольным (bipartite), если его вершины можно правильно раскра- сить двумя цветами. Важность двудольных графов заключается в том, что они возни- кают естественным образом во многих приложениях. Рассмотрим граф сексуальных связей среди гетеросексуалов. Мужчины здесь могут вступать в сексуальные связи только с женщинами и наоборот. Таким образом, в этой простой модели правильная двухцветная раскраска определяется полом. Но как мы можем правильно раскрасить граф в два цвета, таким образом разделив мужчин и женщин? Предположим, что начальная вершина обхода представляет муж- чину. Тогда все смежные вершины должны представлять женщин, при условии, что граф в действительности является двудольным. Мы можем расширить алгоритм обхода в ширину таким образом, чтобы раскрашивать каждую новую открытую вершину цветом, противоположным цвету ее предшествен- ника. Потом мы проверяем, не связывает какое-либо ребро, которое не задействовалось в процессе открытия вершин, две вершины одного цвета. Такая связь будет означать, что граф нельзя раскрасить в два цвета. Процедура двухцветной раскраски графа пока- зана в листинге 5.15. Листинг 5.15. Процедура двухцветной раскраски графа twocolor(graph *g) { int i; /* счетчик ★/ for (i=l; i<=(g->nvertices); i++) color[i] = UNCOLORED; bipartite = TRUE; initialize_search(&g); for (i=l; i<=(g->nvertices); i++) if (discovered[i] == FALSE) I color[i] = WHITE; bfs(g,i); ) )
192 Часть I. Практическая разработка алгоритмов process_edge(int х, int у) { if (color[х] == color[у]) ' bipartite = FALSE; printf ("Warning: not bipartite due to (%d, ”>d) \n",x,y) ; } ct>lor[y] = complement (color [x] ) ; ) complement(int color) ( if (color == WHITE) return(BLACK) ; if (color == BLACK) return(WHITE); return(UNCOLORED); 1 Первой вершине в любой компоненте связности можно присвоить любой цвет (пол). Алгоритм обхода в ширину может разделить мужчин и женщин, но, исходя только из структуры графа, не может определить, какой цвет представляет какой пол. Подведение итогов Обходы в ширину и в глубину предоставляют механизмы для посещения каждой вершины и каждого ребра графа. Они лежат в основе большинства простых, эффективных алгорит- мов для работы с графами. 5.8. Обход в глубину Существует два основных алгоритма обхода графов: обход в ширину (breadth-first search, BFS) и обход в глубину (depth-first search, DFS). Для некоторых задач нет абсо- лютно никакой разницы, какой тип обхода (поиска) использовать, но для других эта разница является критической. Разница между поиском в ширину и поиском в глубину заключается в порядке иссле- дования вершин. Этот порядок зависит полностью от структуры-контейнера, исполь- зуемой для хранения открытых, но не обработанных вершин. ♦ Очередь. Помещая вершины в очередь типа FIFO, мы исследуем самые старые не- исследованные вершины первыми. Таким образом, наше исследование медленно распространяется вширь, начиная от стартовой вершины. В этом суть обхода в ши- рину. ♦ Стек. Помещая вершины в стек с порядком извлечения LIFO, мы исследуем их, отклоняясь от пути для посещения очередного соседа, если таковой имеется, и воз- вращаясь назад, только если оказываемся в окружении ранее открытых вершин. Та- ким образом, мы в своем исследовании быстро удаляемся от стартовой вершины, и в этом заключается суть обхода в глубину. Наша реализация процедуры обхода в глубину отслеживает время обхода для каждой вершины. Каждый вход в любую вершину и выход из нее считаются затратой времени. Для каждой вершины ведется учет затрат времени на вход и выход.
Глава 5 Обход графов 193 Процедуру обхода в глубину можно реализовать рекурсивным методом, что позволяет избежать явного использования стека. Псевдокод алгоритма обхода в глубину показан в листинге 5.16. Листинг 5.16. Обход в глубину DFS'G,U state LuJ = "discovered" обрабатываем вершину и, если необходимо entry [u] = time time = time + 1 for each v e Adj [и] do обрабатываем ребро (и, v), если необходимо if state[v] = "undiscovered" then P[v] = u DFS(G,v) state[u] = "processed" exit[u] = time time = time + 1 При обходе в глубину интервалы времени, потраченного на посещение вершин, обла- дают интересными свойствами. В частности: ♦ Посещение предшественника. Допустим, что вершина х является предшественни- ком вершины у в дереве обхода в глубину; Это подразумевает, что вершина х долж- на быть посещена раньше, чем вершина % т. к. невозможно быть рожденным рань- ше своего отца или деда. Кроме этого, выйти из вершины х можно только после вы- хода из вершины у, т. к. механизм поиска в глубину предотвращает выход из вершины х до тех пор, пока мы не вышли из всех ее потомков. Таким образом, вре- менной интервал посещения у должен быть корректно учтен его предшественни- ком х. ♦ Количество потомков. Разница во времени выхода и входа для вершины v свиде- тельствует о количестве потомков этой вершины в дереве обхода в глубину. Пока- зания часов увеличиваются на единицу при каждом входе и каждом выходе из вер- шины, поэтому количество потомков данной вершины будет равно половине разно- сти между моментом выхода и моментом входа. Мы будем использовать время входа и выхода в разных приложениях обхода в глуби- ну, в частности, в топологической сортировке и в поиске компонент двусвязности (или компонент сильной связности). Нам нужно будет выполнять разные действия при каж- дом входе и выходе из вершины, для чего из процедуры dfs будут вызываться функ- ции process_vertex^early О И process vertex late () соответственно. Другим важным свойством обхода в глубину является то, что он разбивает ребра не- ориентированного графа на два класса: древесные (tree edges) и обратные (back edges). Древесные ребра используются при открытии новых вершин и закодированы в роди- тельском отношении. Обратные ребра — это те ребра, у которых второй конец являет- ся предшественником расширяемой вершины, и поэтому они направлены обратно к дереву. 1 Зак 3741
194 Часть I. Практическая разработка алгоритмов Удивительным свойством обхода в глубину является то, что все ребра попадают в одну из этих двух категорий. Почему ребро не может соединять одноуровневые узлы, а только родителя с потомком? Потому, что все вершины, достижимые из данной вер- шины v, уже исследованы ко времени окончания обхода, начатого из вершины v. и та- кая топология невозможна для неориентированных графов. Данная классификация ре- бер является принципиальной для алгоритмов, основанных на обходе в глубину. На рис. 5.9 изображен неориентированный граф и дерево его обхода в глубину. и) Рис. 5.9. Неориентированный граф (а) и дерево его обхода в глубину (б) Реализация Обход в глубину можно рассматривать как обход в ширину с использованием стека вместо очереди. Достоинством реализации обхода в глубину посредством рекурсии (листинг 5.17) является то, что она позволяет обойтись без явного использования стека. Листинг 5,17. Обход в глубину dfs(graph *g, int v) < edgenode *p; /* Временный указатель */ int у; /* Следующая вершина */ if (finished) return; /* Завершение поиска */ discovered[v] = TRUE; time = time + 1; entry_time[v] = time; process_vertex_early(v); p = g->edges[v]; while (p != NULL) ( у = p->y; if (discovered[y] == FALSE) { parent[y] = v; process_edge(v,y); dfs(g,y) ; )
Гпава 5. Обход графов 195 else if ((!processed[y]) II (g->directed)) process_edge(v,y); if (finished) return; p = p->next; process_vertex late(v) ; time = time + 1; exit_time[v] = time; processed [v] = TRUE; Обход в глубину, по сути, использует ту же самую идею, что и поиск с возвратом, ко- торый рассматривается в разделе 7.1. В обоих алгоритмах мы перебираем все возмож- ности, продвигаясь вперед, когда это удается, и возвращаясь обратно, когда больше не осталось неисследованных элементов для дальнейшего продвижения. Оба алгоритма легче всего воспринимать как рекурсивные алгоритмы. Подведение итогов При обходе в глубину вершины упорядочиваются по времени входа/выхода, а ребра раз- биваются на два типа — древесные и обратные. Именно такая организация и делает обход в глубину столь мощным алгоритмом. 5.9. Применение обхода в глубину По сравнению с другими парадигмами разработки алгоритмов, обход в глубину не ка- жется чем-то страшным. Но он довольно сложен, вследствие чего для его корректной работы необходимо правильно реализовать все до мельчайших подробностей. Правильность алгоритма обхода в глубину зависит от того, когда именно обрабатыва- ются ребра и вершины. Вершину v можно обработать до обхода исходящих из нее ре- бер (процедура processvertex early о) ИЛИ после ЭТОГО (процедура process_vertex_ lateo). Иногда действия выполняются в обоих случаях, например, инициализация до обхода ребер посредством функции process vertex eariyO специфической структуры данных вершин, которая будет модифицирована операциями обработки ребер, а после обхода подвергнута анализу посредством функции process_vertex_iate (). В неориентированных графах каждое ребро (л,у) находится в списке смежности как для вершины х, так и для вершины у. Таким образом, для обработки каждого ребра (х,у) существуют два момента— при исследовании вершины х и вершины у. Ребра разбиваются на древесные и обратные при первом исследовании ребра. Момент, в ко- торый мы видим ребро в первый раз, естественно подходит для выполнения специфи- ческой обработки ребра. В тех случаях, когда мы встречаемся с ребром во второй раз, может возникнуть необходимость в каком-то другом действии. Но если мы встречаем ребро (х, у), двигаясь из вершины х, как нам определить, не бы- ло ли это ребро уже пройдено из вершины у? На этот вопрос легко ответить, если вер- шина у неоткрытая: ребро (х,у) становится древесным ребром, следовательно, мы ви- дим его в первый раз. Ответ также очевиден, если вершина у не была полностью обра- ботана: поскольку мы рассматривали ребро при исследовании вершины у, это второе
196 Часть I. Практическая разработка алгоритмов посещение этого ребра. А если вершина у является предшественником вершины х и, вследствие этого, находится в открытом состоянии? Подумав, мы поймем, что это должен быть первый проход, если только вершина у не является непосредственным предшественником вершины х. т. е. ребро (у, х) является древесным. Это можно уста- новить проверкой равенства у == parent [х]. Каждый раз, когда я пробую реализовать алгоритм на основе обхода в глубину, я убе- ждаюсь. что этот тип алгоритмов достаточно сложен. Поэтому я рекомендую вам тща- тельно изучить рассмотренные реализации этого алгоритма, чтобы понять, где и поче- му могут возникать проблемы. 5.9.1. Поиск циклов Наличие обратных ребер является ключевым фактором при поиске циклов в неориен- тированных графах. Если в графе нет обратных ребер, то все ребра являются древес- ными и дерево не содержит циклов. Но любое обратное ребро, идущее от вершины х к предшественнику у, создает цикл, или замкнутый маршрут между вершинами у и х Цикл легко найти посредством обхода в глубину, как показано в листинге 5.18. Листинг 6.18. Поиск цикла process_edge(int х, int у) { if (parent [х] != у) ( /* найдено обратное ребро! ’’/ printf("Cycle from %d to %d:",y,x), find__path (y,x, parent) ; printf("\n\n"); finished = TRUE; } ) Правильность данного алгоритма поиска цикла зависит от обработки каждого неори- ентированного ребра только один раз. В противном случае двойной проход по любому неориентированному ребру может создать фиктивный двухвершинный цикл (х;у,х). Поэтому после обнаружения первого цикла устанавливается флаг finished, что ведет к завершению работы процедуры. 5.9.2. Шарниры графа Допустим, вы — диверсант в тылу врага, которому нужно вывести из строя телефон- ную сеть противника. Какую из показанных на рис. 5.Ю телефонных станций вы бы решили взорвать, чтобы причинить как можно больший ущерб? Обратите внимание, что в графе на рис. 5.Ю имеется одна точка, удаление которой от- соединяет связную компоненту графа. Такая вершина называется шарниром (или раз- деляющей вершиной, или точкой сочленения). Любой граф, содержащий шарнир, яв- ляется уязвимым по своей природе, т. к. удаление этой одной вершины ведет к потере связности между другими вершинами.
Гпава 5. Обход графов 197 В разделе 5.7.1 мы рассмотрели алгоритм на основе поиска в ширину для определения связных компонент графа. В общем, связностью графа (graph connectivity) называется наименьший набор вершин, в результате удаления которых граф станет несвязным. Если граф имеет шарнир, то его связность равна единице. Более устойчивые графы, л которых шарниры отсутствуют, называются двусвязными. Связность также рассмат- ривается в разделе 15.8. Шарниры графа можно с легкостью найти простым перебором. Для этого мы временно удаляем вершину г. после чего выполняем обход оставшегося графа в ширину или в глубину, чтобы выяснить, является ли он после этого связным. Эта процедура повторя- ется для каждой вершины графа. Общее время для п таких обходов равно ()(п(т + и)). Но существует изящный алгоритм с линейным временем исполнения, который прове- ряет все вершины связного графа за один обход в глубину. Какую информацию о шарнирах может предоставить нам дерево обхода в глубину? Прежде всего, это дерево содержит все вершины графа. Если дерево обхода в глубину представляет весь граф, то все внутренние (не листовые) вершины будут шарнирами, т. к. удаление любой из них отделит лист от корня. А удаление листа разъединить де- рево не может, т. к листовая вершина не соединяет с деревом никакие другие верши- ны. На рис. 5.11 изображено дерево обхода графа в глубину, содержащее два шарни- ра— вершины 1 и 2. Обратное ребро (5.2) не дает вершинам 3 и 4 быть шарнирами. Вершины 5 и 6 не являются шарнирами, потому что они листья. Рис. 5.10. Шарнир является самой слабой точкой графа Рис. 5.11. Дерево обхода графа в глубину Корень дерева обхода представляет специальный случай. Если он имеет только одного потомка, то корень также является листом. Но если корень имеет больше одного по- томка, то удаление корня разделяет потомков, что означает, что корень является шар- ниром. Общие графы более сложные, чем деревья. Но обход общего графа в глубину разделяет ребра на древесные и обратные. Обратные ребра можно рассматривать как страховоч- ные фалы, связывающие вершину с одним из ее предшественников. Например, нали- чие обратного ребра от вершины х к вершине у гарантирует, что ни одна из вершин на пути между X и у не может быть шарниром. Удаление любой из этих вершин не нару-
198 Часть I. Практическая разработка алгоритмов шит связности дерева, т. к. обратное ребро, как страховочный фад. будет удерживать их связанными с остальными компонентами дерева. При поиске шарниров графа требуется информация о том, в какой степени обратные ребра связывают части дерева обхода в глубину после возможного шарнира с предше- ствующими узлами. Пусть переменная reachable_ancestor[v] обозначает самый стар- ший предшественник вершины v, достижимый через древесные и обратные ребра. Сначала значение этой переменной равно reachable ancestor [v]=v (листинг 5.19). /* Самый старший достижимый предшественник вершины v */ /* Степень исхода вершины v дерева обхода в глубину*/ int reachable_ancestor[MAXV+1]; int tree_out_degree[MAXV+1]; process_vertex_early(int v) ( reachable_ancestor[v] = v; ) Переменная reachable_ancestor [v] обновляется при каждом обнаружении обратного ребра, которое ведет к более старшему предшественнику, чем текущий. Относитель- ный возраст (ранг) предшественников можно определить по значению переменной entry_time. содержащей время входа в вершину (листинг 5.20). Листинг 5.20. Определение возраста предшественников process_edge(int х, int у) { int class; /* класс ребра */ class = edge_classification(х,у); if (class == TREE) tree outdegree[x] = tree_out_degree[x] + 1; if ((class == BACK) && (parent[x] != y)) { if (entry_time[y] < entry_time[reachable_ancestor[x]]) reachable_ancestor[x] = y; 1 Здесь исключительно важно определить, как достижимость влияет на то, является ли вершина v шарниром. Существует три типа шарниров, как показано на рис. 5.12. ♦ Корневые шарниры. Если корень дерева обхода в глубину имеет свыше одного по- томка, то он должен быть шарниром, так как при его удалении никакое ребро не может соединить поддерево второго потомка с поддеревом первого потомка. ♦ Мостовые шарниры. Если самой первой достижимой из вершины v является вер- шина родителъ\у\, то удаление ребра (родитель[у], v) разрывает граф. Очевидно, что вершина (родитель^]) должна быть шарниром, т. к. ее удаление отсоединяет вершину v от остального дерева обхода графа в глубину. По этой же причине шар-
Гпава 5. Обход графов 199 ниром является и вершина v. если только она не является листом дерева обхода в глубину. Удаление любого листа удаляет только сам лист, но ничего после него. ♦ Родительские шарниры. Если самой первой вершиной, достижимой из вершины v, является родитель вершины v, то удаление этого родителя должно отрезать верши- ну v от остальной части дерева, если только данный родитель не является корнем дерева. В листинге-5.21 приводится процедура для определения соответствия вершины, из ко- торой выполняется возврат после обхода всех ее исходящих ребер, одному из этих трех типов шарниров. "Возраст" вершины v представляется переменной entry_time [v]. Вычисляемое время time v обозначает самую старшую вершину, достигаемую посредством обратных ре- бер. Возвращение к предшественнику выше вершины v исключает возможность, что вершина v может быть шарниром. Листинг 5.21. Определение типа шарнира ........................_____ process_vertex_late (int v) ( bool root; /* Эта вершина — корень дерева обхода в глубину? */ int time_v; /* Самое раннее время достижимости вершины v */ int time_parent; /* Самое раннее время достижимости вершины parent[v] */ if (parent[v] < 1) { /* Вершина v — корень? */ if (tree_out_degree[v] > 1) printf("root articulation vertex: %d \n",v); return;
200 Часть I. Практическая разработка алгоритмов root = (parent[parent[v]] < 1); /★ Вершина parent[v] корень? */ if ((reachableancestor[v] == parent[v]) && (iroot) printf("parent articulation vertex: %d \n",parent[v]); if (reachable_ancestor[v] == v) { printf("bridge articulation vertex: %d \n",parent[v]); if (tree_out_degree[v] > 0) /* Вершина v — лист? */ printf("bridge articulation vertex: %d \n",v); ) time_v = entry time[reachable_ancestor[v]]; time_parent = entry_time[ reachable_ancestor[parent[v]] ]; if (time_v < time_parent) reachable_ancestor[parent[v]] = reachable_ancestor[v]; 1 В последних строчках кода данной процедуры определяется момент, в который мы превращаем самый старый достижимый предшественник вершины в ее родителя, а именно тот момент, когда он выше, чем текущий предшественник родителя. В качестве альтернативы, надежность можно рассматривать не с точки зрения слабости вершин, а с точки зрения слабости ребер. Возможно, нашему разведчику вместо выво- да из строя телефонной станции было бы легче повредить кабель. Ребро, чье удаление разъединяет граф, называет мостом (bridge). Граф, не имеющий мостов, называется реберно-двусвязным (edge-biconnected). Является ли мостом данное ребро (х. у), можно с легкостью определить за линейное время, удалив это ребро и проверив, является ли получившийся граф связным. Более того, можно найти все мосты за такое же линейное время О(п + ту Ребро (х.у) являет- ся мостом, если, во-первых, оно древесное, а во-вторых, нет обратных ребер, которые соединяли бы вершину у или нижележащие вершины с вершиной х или вышележащи- ми вершинами. Соответствие ребра этим условиям можно проверить с помощью слег- ка модифицированной функции для определения достижимого предшественника (см. листинг 5.21). 5.10. Обход в глубину ориентированных графов Обход в глубину неориентированного графа полезен тем, что он аккуратно упорядочи- вает ребра графа, как показано на рис. 5.9. При обходе неориентированных графов каждое ребро или находится в дереве обхода в глубину, или является обратным ребром к предшествующему ребру в дереве. Давайте повторно рассмотрим, почему это так. Допустим, что при обходе мы выходим на пря- мое ребро (х, у), направленное к вершине-потомку. В таком случае мы бы открыли ребро (х, у) при исследовании вершины у, что делает его обратным ребром. Или допус- тим. что мы выходим на поперечное ребро (х, у), связывающее две вершины, не имею- щие отношения друг к другу. Опять же, мы открыли бы это ребро при исследовании вершины у, что делает его древесным ребром. Для ориентированных графов диапазон допустимых маркировок обхода в глубину мо- жет быть более широким. В самом деле, при обходе ориентированных графов могут использоваться все четыре типа ребер, показанные на рис. 5.13.
Глава 5. Обход графов 201 Древесные ребра Прямое ребро Обратное ребро Поперечные ребра Рис. 5.13. Возможные типы ребер Но эта классификация оказывается полезной при разработке алгоритмов для работы с ориентированными графами. Для каждого типа ребра обычно предпринимается соот- ветствующее ему действие. Тип ребра можно без труда определить, исходя из состояния вершины, времени ее от- крытия и ее родителя. Соответствующая процедура показана в листинге 5.22. Листинге 22 Определение типа ребра mt edge_classification(int х, int у) if (parentfy] == х) return(TREE); if (discovered[у] && !processed[у]) return(BACK); if (processed[y] && (entry_time[y]>entry_time[x])) return(FORWARD); if (processed[y] && (entry_time[y]<entry_time[x])) return(CROSS); printf("Warning: unclassified edge (%d,%d)\n",x,y); Как и в случае с процедурой обхода в ширину, реализация алгоритма обхода в глубину содержит места, в которые можно вставить функцию выборочной обработки каждой вершины или ребра, например, копировать, выводить на экран или подсчитывать их. Оба алгоритма начинают с обхода всех ребер в одной компоненте связности. Так как для обхода несвязного графа нам нужна начальная вершина в каждой компоненте, то обход должен начинаться с любой вершины, оставшейся неоткрытой после обхода компоненты. При условии правильной инициализации получаем законченный алго- ритм обхода ориентированного графа в глубину (листинг 5.23). Листинг 5.23. Алгсритк. обхода в глубину ориентированного графа )FS-graph (G) for each vertex u e V[G]do state[u] = "undiscovered" for each vertex u e V[G] do if state[u] = "undiscovered" then инициализируем новую компоненту, если необходимо DFS(G,u)
202 Часть I. Практическая разработка алгоритмов Я рекомендую читателям проверить и убедиться в правильности этих четырех состоя- ний вершины. Все, сказанное мною о сложности алгоритма обхода в глубину, вдвойне справедливо для орграфов. 5.10.1. Топологическая сортировка Топологическая сортировка является наиболее важной операцией на бесконтурных орграфах. Данный тип сортировки упорядочивает вершины вдоль линии таким обра- зом, что все ориентированные ребра направлены слева направо. Такое упорядочивание ребер невозможно в графе, содержащем ориентированный цикл, т. к. невозможно, что- бы ребра шли по одной линии в одну сторону и при этом возвращались назад в исход- ную точку. Любой бесконтурный орграф имеет, по крайней мере, одно топологическое упорядочи- вание (рис. 5.14). Рис. 5.14. Бесконтурный орграф с единственным топологическим упорядочиванием (G, А, В, С, F. Е, D) Важность топологической сортировки состоит в том, что она позволяет упорядочить вершины графа таким образом, что каждую вершину можно обработать перед обра- боткой ее потомков. Допустим, что ребра представляют управление очередностью та- ким образом, что ребро (х,у) означает, что работу х нужно выполнить раньше, чем ра- боту у. Тогда любое топологическое упорядочивание определяет правильное календар- ное расписание. Более того, бесконтурный орграф может содержать несколько таких упорядочиваний. Но топологическая сортировка имеет и другие применения. Например, предположим, что нам нужно найти самый короткий (или самый длинный) путь в бесконтурном орграфе от точки х до точки у. Никакая вершина, расположенная в топологическом упорядочивании после вершины у, не может находиться в этом пути, т. к. из такой вершины невозможно попасть обратно в вершину у. Все вершины можно обработать слева направо в топологическом порядке, принимая во внимание влияние их исходя- щих ребер, и будучи в полной уверенности, что мы просмотрим все, что нужно, перед тем как оно потребуется. Топологическая сортировка оказывается очень полезной для решения практически любых алгоритмических задач по орграфам, что подробно обсу- ждается в разделе 15.2. Обход в глубину является эффективным средством для осуществления топологической сортировки. Орграф является бесконтурным, если не содержит обратных ребер. Топо- логическое упорядочивание бесконтурного орграфа осуществляется посредством мар-
Гпава 5. Обход графов 203 кировки вершин в порядке, обратном тому, в котором они отмечаются как обработан- ные. Почему это возможно? Рассмотрим, что происходит с каждым ориентированным ребром (.v,j’)- обнаруженным при исследовании вершины х. ♦ Если вершина^ не открыта, то мы начинаем обход в глубину из вершины у, преж- де чем мы можем продолжать исследование вершины х. Таким образом, вершина у помечается как обработанная до того, как такой статус присваивается вершине х, и в топологическом упорядочивании вершина х будет находиться перед вершиной у, что и требуется. ♦ Если вершина у открыта, но не обработана, то ребро (х, у) является обратным ребром, что запрещено в бесконтурном орграфе. ♦ Если вершина^ обработана, то она помечается соответствующим образом раньше вершины х. Следовательно, в топологическом упорядочивании вершина х будет на- ходиться перед вершиной у, что и требуется. В листинге 5.24 приводится реализация топологической сортировки. Листинг 5.24. Топологическая сортировка —-- ...................—... ..........................................— ......... ...........—....— —......— process_vertex_late (int v) push(&sorted,v); process_edge (int x, int y) int class; /* Класс ребра */ class = edge_classification(x,у); if (class == BACK) printf("Warning: directed cycle found, not a DAG\n”); topsort (graph *g) int i; /* Счетчик */ init_stack(&sorted); for (i=l; i<=g->nvertices; i++) if (discovered[i] == FALSE) dfs(g,i); print_stack(&sorted); !* Выводим топологическое упорядочивание */ Каждая вершина кладется в стек после обработки всех исходящих ребер. Самая верх- няя вершина в стеке не имеет входящих ребер, идущих от какой-либо вершины, имеющейся в стеке. Последовательно снимая вершины со стека, получаем их тополо- гическое упорядочивание. 5.10.2. Сильно связные компоненты Классическим применением поиска в глубину является разложение ориентированного графа на енпьно связные компоненты (strongly connected components). Орграф является
204 Часть I. Практическая разработка алгоритмов сильно связным (strongly connected), если для любой пары его вершин (v, и) вершина v достижима из вершины и и наоборот. В качестве практического примера сильно связ- ного графа можно привести граф дорожных сетей с двусторонним движением. Является ли граф G = (К Е) сильно связным, можно с легкостью проверить посредст- вом обхода графа за линейное время. Начнем обход с произвольной вершины v. Каж- дая вершина графа должна быть достижимой из этой вершины и. следовательно, от- крыта обходом в ширину или глубину с начальной точкой в вершине v. В противном случае граф G не может быть сильно связным. Потом создадим граф G' = (И. Е1) с точ- но таким же набором вершин и ребер, как и граф G, но с обратной ориентацией ребер, т. е., ориентированное ребро (х, у) е Е тогда и только тогда, когда (у, х) е Е'. Таким образом, любой путь от вершины v к вершине г в графе G' соответствует пути от вер- шины г к вершине г в графе G. Выполнив обход в глубину графа С'от вершины v, мы найдем все вершины с путями к этой вершине в графе G. Граф является сильно связ- ным тогда и только тогда, когда вершина v достижима из любой вершины в графе G и все вершины в графе G достижимы из вершины v. Графы, которые сами не являются сильно связными, можно разбить на сильно связные компоненты, как показано на рис. 5.15. а. Рис. 5.15. Сильно связные компоненты графа (и) и дерево обхода в глубину (б) Сильно связные компоненты графа и соединяющие их слабо связные ребра можно найти с помощью обхода в глубину. Этот алгоритм основан на том обстоятельстве, что с помощью обхода в глубину легко найти ориентированный цикл, поскольку такой цикл дает любое обратное ребро и путь вниз в дереве обхода в глубину. Все вершины в этом цикле должны быть в одной и той же сильно связной компоненте. Таким образом мы можем сжать вершины в этом цикле в одну вершину, представляющую всю компо- ненту', после чего повторить обход. Этот процесс прекращается, когда исчерпаны все ориентированные циклы, а каждая вершина представляет отдельную сильно связную компоненту.
Глава 5. Обход графов 205 Наш подход к реализации этой идеи похож на поиск компонент двусвязности в разделе 5.9.2. Однако мы пересмотрим понятие самой старшей достижимой вершины в связи с наличием "недревесных" ребер и необходимостью возвращения из вершин. Так как мы имеем дело с ориентированным графом, то нам нужно принимать во вни- мание прямые ребра (ведущие от вершины к потомку) и поперечные ребра (ведущие от вершины обратно к ранее открытой вершине, не являющейся предшественником). Наш алгоритм (листинг 5.25) отделяет от дерева обхода по одной сильной компоненте за шаг и присваивает всем ее вершинам номер данной компоненты. Листинг 5.25. Алгоритм разложения графа на сильно связные компоненты strong_components (graph *g) I int i; Счетчик */ for (i=l; i<=(g->nvertices); i++) low[i] = i; sccfi] - -1; ) component s_found = 0; init_stack(bactive); initialize_search(&g); for (i=l; i<=(g->nvertices); i++) if (discovered!!] == FALSE) { dfs(g,i) ; Переменная iow[v] представляет самую старшую известную вершину, находящуюся в той же самой сильно связной компоненте, что и вершина v. Этд вершина не обязатель- но является родителем вершины v, но благодаря поперечным ребрам может находиться на одном уровне с ней. Поперечные ребра, которые идут к вершинам из предыдущих сильно связных компонент графа, нам не нужны, т. к. от них нет пути назад к вершине г. но в других случаях поперечные ребра подлежат обработке. Прямые ребра не влияют на достижимость по ребрам дерева обхода в глубину и поэтому их можно не прини- мать во внимание (листинг 5.26). ...— ..............-<г --......... - ................ .!...—-............. Листинг 5.26. Процедура обработки ребер 1st low[MAXV+l]; /* Самая старшая вершина наверняка находится в компоненте вершины v */ int scc[MAXV+l]; /* Номер сильной компоненты для каждой вершины */ process_edge(int х, int у) । int class; /* Класс ребра */ class = edge_classification(х,у); if (class == BACK) { if (entry_time[y] < entry_time[ lowfx] ] ) low[x] = y; i
206 Часть I. Практическая разработка алгоритмов if (class == cross) { if (scc[y] == -1) /* Компонента еще не присвоена */ if (entry_time[у] < entry_time[ low[x) ] ) low[x] = у; I Новая сильно связная компонента считается обнаруженной, когда самой нижней вер- шиной, достижимой из вершины v, является эта же вершина у. В таком случае эта ком- понента снимается со стека. В противном случае, мы объявляем родителя самым стар- шим достижимым предшественником и выполняется возврат (листинг 5.27). Листинг 6.27. Процедуры обработки вершин process_vertex_early(int v) ( push(^active,v); ) process vertexlate(int v) { if Qow[v] == v) { /* Ребро (parent[v],v) отрезает сильно связную компоненту */ popcomponent(v); ) if (entry_time[low[v]] < entry_time[low[parent[v]]]) lowfparent[v]] = low[v); } pop component(int v) { int t; /* Заполнитель вершины */ components_found = components_found + 1; scc[v] = components_found; while ((t = pop(sactive)) != v) { scc[t] = components_found; I Замечания к главе Рассмотренный в главе материал по обходу графов является расширенной версией ма- териала из главы 9 книги [SR03], Самое лучшее описание библиотеки для работы с графами Combinatorica можно найти в старом [Ski9OJ и новом [PS03] издании книги, посвященной работе с этой библиотекой. Доступное введение в теорию социальных сетей можно найти в книгах [ВагОЗ] и [Wat04].
Гпава 5 Обход графов 207 5.11. Упражнения Алгоритмы для эмуляции графов I. [3] Даны графы G, и G2 (рис. 5.16). Рис. 5.16. Примеры графов а) Укажите порядок вершин при обходе в ширину, начинающемся с вершины А. В случае конфликтов рассматривайте вершины в алфавитном порядке (т. е. А предшествует Z). б) Укажите порядок вершин в обходе в глубину, начинающемся с вершины А. В случае конфликтов рассматривайте вершины в алфавитном порядке (т. е. А предшествует Z). 2 [3] Выполните топологическую сортировку на графе G, представленном на рис. 5.17. Рис. 5.17. Пример графа Обход графов 3. [3] Докажите методом индукции, что между любой парой вершин дерева существует однозначный путь. 4. [3] Докажите, что при обходе в ширину неориентированного графа G каждое ребро мо- жет быть либо древесным, либо поперечным, где х не является ни предшественником, ни потомком вершины у в поперечном ребре (х, у).
208 Часть I. Практическая разработка алгоритмов 5. [3] Разработайте алгоритм с линейным временем исполнения для вычисления хромати- ческого числа графов, где степень каждой вершины не выше второй. Должны такие графы быть двудольными? 6. [5] В обходах в ширину и глубину неоткрытая вершина помечается как открытая при первом ее посещении и как обработанная после того, как она была полностью исследо- вана. В любой момент времени в открытом состоянии могут находиться несколько вер- шин одновременно. а) Опишите граф из п вершин с заданной вершиной v, у которого при обходе в ширину, начинающемся в вершине v, одновременно ®(и) вершин находятся в открытом состоя- нии. б) Опишите граф из п вершин с заданной вершиной v, у которого при обходе в глубину, начинающемся в вершине v, одновременно ®(и) вершин находятся в открытом состоя- нии. в) Опишите граф из п вершин с заданной вершиной v, у которого в некоторой точке обхода в глубину, начинающегося в вершине v, одновременно ®(д) вершин находятся в неоткрытом состоянии, в то время как ®(и) вершин находятся в обработанном со- стоянии. (Следует иметь в виду, что при этом могут также существовать открытые вершины.) 7. [4] Возможно ли восстановить двоичное дерево, зная его обход в ширину и симметрич- ный обход? Если возможно, то опишите алгоритм для этого. Если невозможно, то пре- доставьте контрпример. Решите задачу для случаев обхода в ширину и глубину. 8. [3] Предоставьте правильный эффективный алгоритм для преобразования одной струк- туры данных неориентированного графа G в другую. Для каждого алгоритма предос- тавьте значение временной сложности, полагая, что граф состоит из п вершин и т ребер. а) Преобразовать матрицу смежности в списки смежности. б) Преобразовать список смежности в матрицу инцидентности. Матрица инцидентности М содержит строку для каждой вершины и столбец для каждого ребра. Если вершина i является частью ребра j, то Л/[7, /] = 1, в противном случае M[iJ] = 0. в) Преобразовать матрицу инцидентности в списки смежности. 9. [3] Дано арифметическое выражение в виде дерева. Каждый лист дерева является целом числом, а каждый внутренний узел представляет одну из стандартных арифметических операций (+,-,*,/). Например, представление выражения 2 + 3*4 + (3*4)/5 в виде дерева показано на рис. 5.18, а. Разработайте алгоритм с временем исполнения О(н) для вы- числения такого выражения, где п — количество узлов в дереве. 10. [5] Арифметическое выражение представлено в виде бесконтурного орграфа, из кото- рого удалены общие вложенные выражения. Каждый лист является целым числом, а каждый внутренний узел представляет одну из стандартных арифметических операций (+,-,*,/). Например, представление выражения 2 + 3*4 + (3*4)/5 в виде бесконтурного орграфа показано на рис. 5.18, б. Разработайте алгоритм для вычисления такого выра- жения за время О(п + /и), где п— количество вершин бесконтурного орграфа, а т— количество ребер. (Подсказка: для достижения требуемой эффективности модифици- руйте алгоритм для случая с деревом.)
Гпава 5. Обход графов 209 Рис. 5.18. Представление выражения 2 + 3*4 + (3*4)/5 в виде дерева (а) и бесконтурного орграфа (б) 11. [8] В разделе 5.4 описывается алгоритм для эффективного создания двойственного графа триангуляции, который не гарантирует линейного времени исполнения. Для дан- ной задачи разработайте алгоритм с линейным временем исполнения в наихудшем слу- чае. Разработка алгоритмов 12. [5] Квадратом (square) орграфа G = (Е, £ ) называется граф G2 = (Е. £2). при условии, что (w, w) е Е2 тогда и только тогда, когда существует такая вершина v е Е, для которой (г/, v) е £ и (v, w) g £, т. е. от вершины и к вершине w существует путь, состоящий ров- но из двух ребер. Предоставьте эффективные алгоритмы для создания списков и матриц смежности для таких графов. 13. [5] Вершинным покрытием (vertex cover) графа G = (Е. £) называется такое подмноже- ство вершин Е’ g Е, для которого каждое ребро в £ инцидентно, по крайней мере, одной вершине в V'. а) Разработайте эффективный алгоритм поиска вершинного покрытия минимального размера, если G является деревом. б) Пусть G = (Е, £) будет такой граф, в котором вес каждой вершины равен степени ее ребра. Разработайте эффективный алгоритм поиска вершинного покрытия графа мини- мального веса. в) Пусть G = (Е, Е) будет деревом с вершинами с произвольным весом. Разработайте эффективный алгоритм поиска вершинного покрытия графа минимального веса. 14. [3] Вершинным покрытием графа G= (Е,Е) называется такое подмножество вершин Е' g V, в котором каждое ребро в £ содержит, по крайней мере, одну вершину из Е'. Если мы удалим все листья из любого дерева обхода в глубину графа G, то будут ли оставшиеся вершины составлять вершинное покрытие графа G? Если да, предоставьте доказательство; если нет, приведите контрпример. 15. [5] Вершинным покрытием графа G = (Е, Е) называется такое подмножество вершин Е' g Е, в котором каждое ребро в £ содержит, по крайней мере, одну вершину из Е'. Не- зависимым множеством (independent set) графа G = (Е, £) называется подмножество вершин Е' е Е, такое, что £ не содержит ни одного ребра, обе вершины которого явля- лись бы членами Е'.
210 Часть I. Практическая разработка алгоритмов Независимым вершинным покрытием (independent vertex cover) называется такое под- множество вершин, которое является как независимым множеством, так и вершинным покрытием графа G. Разработайте эффективный алгоритм для проверки, содержит ли граф G независимое вершинное покрытие. К какой классической задаче на графах сво- дится данная задача? 16. [5] Независимым множеством (independent set) неориентированного графа G = (К, £) называется такое подмножество вершин U, для которого Е не содержит ни одного реб- ра, обе вершины которого являлись бы членами U. а) Разработайте эффективный алгоритм поиска независимого множества максимального размера, если G является деревом. б) Пусть G - (Г, £) будет деревом, каждая вершина которого имеет вес, равный степени данной вершины. Разработайте эффективный алгоритм поиска независимого множества максимального размера дерева G. в) Пусть G = (И, £) будет деревом с вершинами с произвольно присвоенным весом. Раз- работайте эффективный алгоритм поиска независимого множества максимальною раз- мера дерева G. 17. [5] Нужно определить, содержит ли данный неориентированный граф G = (И, £) тре- угольник, т. е. цикл длиной 3. а) Разработайте алгоритм с временем исполнения GQ И|3) поиска треугольника в графе, если таковой имеется. б) Усовершенствуйте ваш алгоритм для исполнения за время С)(|1/|>|£|). Можно пола- гать, что | И| < |£|. Обратите внимание на то. что эти пределы позволяют оценить время для преобразова- ния представления графа G из матрицы смежности в список смежности (или в противо- положном направлении). 18. [5] Имеется набор фильмов /V/,, М2,.... Л7* и набор клиентов, каждый из которых указы- вает два фильма, которые он бы хотел посмотреть в ближайшие выходные. Фильмы по- казываются по вечерам в субботу и воскресенье. Одновременно могут демонстриро- ваться несколько фильмов. Вам нужно решить, какие фильмы следует показывать в субботу, а какие в воскресенье, таким образом, чтобы каждый клиент смог посмотреть указанные им фильмы. Возмож- но ли составить расписание, по какому каждый фильм показывается только один раз? Разработайте эффективный алгоритм поиска такого расписания. 19. [5] Диаметр дерева Т = (К, £) задается формулой: тахб(н.г), где д(и, v) — количество w.rel' ребер в пути от вершины и к вершине v. Опишите эффективный алгоритм для вычисле- ния диаметра дерева, докажите его правильность и проанализируйте время его испол- нения. 20. [5] Имеется неориентированный граф G из п вершин и т ребер и целое число к. Разра- ботайте алгоритм с временем исполнения О(т + п) для поиска максимального порож- денного подграфа (induced subgraph) Н графа G, каждая вершина которого имеет сте- пень > к. или докажите, что такого графа не существует. Порожденным подграфом £= (G. R) графа G = (Г. £) называется подмножество G вершин И графа G и все реб- ра R графа G, для которых обе вершины каждого ребра являются членами U.
I Гпава 5. Обход графов 211 21. [6] Даны две вершины v и w в орграфе G = (И. £). Разработайте алгоритм с линейным временем исполнения для поиска разных кратчайших путей (не обязательно не имею- щих общих вершин) между v и w. Примечание: все ребра графа G невзвешенные. 22. [6] Разработайте алгоритм с линейным временем исполнения для удаления всех вершин второй степени из графа путем замены ребер (г/, v) и (у, vv) ребром (г/, vv). Также нужно удалить множественные копии ребер, оставив только одно ребро. Обратите внимание, что удаление копий ребра может создать новую вершину второй степени, которую нуж- но будет удалить, а удаление вершины второй степени может создать кратные ребра, которые также нужно будет удалить. Ориентированные графы 23. [5] Перед вами стоит задача выстроить п непослушных детей друг за другом, причем у вас имеется список из т утверждений типа "i ненавидит Если i ненавидит /. то будет разумно не ставить i позади j, т. к. i может что-нибудь швырнуть в j. а) Разработайте алгоритм с временем исполнения О(т + и) для выстраивания детей с соблюдением перечисленных в списке условий. Если такое упорядочивание невоз- можно, алгоритм должен сообщить об этом. б) Допустим, что вам нужно выстроить детей в несколько рядов таким образом, что если / ненавидит J, то i должен стоять в каком-нибудь ряду впереди J. Разработайте алгоритм для определения минимального количества рядов, если это возможно. 24. [3] Определите максимальное количество компонент, на которое можно уменьшить количество слабо связных компонент ориентированного графа, добавив в него одно ориентированное ребро. Что можно сказать по поводу количества сильно связных ком- понент? 25. [5] Ориентированным деревом (arborescence) орграфа G называется такое корневое де- рево, что существует ориентированный путь от корня ко всем другим вершинам графа. Разработайте правильный эффективный алгоритм для тестирования, содержит ли граф G ориентированное дерево. Укажите временную сложность алгоритма. 26. [5] Истоком орграфа С= (Г, £) называется такая вершина v, из которой все другие вершины графа G достижимы ориентированным путем. а) Разработайте алгоритм с временем исполнения О(и + т) (где п = |£l и т = |£|) для проверки того, является ли данная вершина v истоком для графа G. б) Разработайте алгоритм с временем исполнения О(п+ т) (где и = |Р] и т = |£|) для проверки того, содержит ли граф G исток. 27. [9] Турниром (tournament) называется полный орграф, т. е. такой граф G = (Г, £), в ко- тором для всех и, vG Vтолько или (и, v) или (у, и) принадлежит множеству £. Покажите, что каждый турнир содержит Гамильтонов путь, т. е. путь, который проходит через каждую вершину только один раз. Разработайте алгоритм поиска этого пути. Шарниры графа 28. [5] Шарниром графа G называется вершина, удаление которой разъединяет граф. Дан граф G из п вершин и т ребер. Разработайте алгоритм с временем исполнения О(п + т) для поиска вершины в графе G, не являющейся шарниром, т. е. вершины, чье удаление не разъединяет граф.
212 Часть I. Практическая разработка алгоритмов 29. [5] По условиям предыдущей задачи разработайте алгоритм с временем исполнения О(п + т) для поиска такой последовательности удалений п вершин, в которой ни одно удаление не разъединяет граф. (Подсказка: рассмотрите возможность обхода в глубину и в ширину.) 30- [3] Дан связный неориентированный граф G. Ребро е, удаление которого разъединяет граф, называется мостам. Должен ли каждый мост е быть ребром в дереве обхода в глубину графа G? Если да, то предоставьте доказательство этого; если нет, то предос- тавьте соответствующий контрпример. Задачи, предлагаемые на собеседовании 31. [5] Какие структуры данных используются при обходе в глубину и при обходе в ши- рину? 32. [4] Напишите функцию, которая обходит дерево двоичного поиска и возвращает /-й по порядку узел. Задачи по программированию Эти задачи доступны на сайтах http://vvww.programining-cliallenges.com и http:// uva.onlinejudge.org. 1. Bicoloring. 110901/10004. 2. Playing with Wheels. 110902/10067. 3. The Tourist Guide. 110903/10099. 4. Edit Step Ladders. 110905/10029. 5. Tower of Cubes. 110906/10051.
ГЛАВА 6 Алгоритмы для работы со взвешенными графами Рассмотренные в главе 5 структуры данных и алгоритмы обхода предоставляют базо- вые конструктивные блоки для выполнения любых вычислений на графах. Но все эти алгоритмы применялись для невзвешенных графов (unweighted graphs), т. е. графов, в которых все ребра имеют одинаковый вес. Но для взвешенных графов (weighted graph) существует отдельная область задач. В ча- стности, ребрам графов дорожных сетей присваиваются такие числовые значения, как длина, скорость движения или пропускная способность данного отрезка дороги. Опре- деление кратчайшего пути в таких графах оказывается более сложной задачей, чем об- ход в ширину в невзвешенных графах, но открывает путь к обширному диапазону при- ложений. Рассматриваемые в главе 5 структуры данных поддерживают графы со взвешенными ребрами неявно, но сейчас мы сделаем эту возможность явной. Для этого мы сначала определим структуру списка смежности, состоящую из массива связных списков, в ко- торых ребра, исходящие из вершины х, указываются в списке edges [к] (листинг 6.1). Листинг 6.1. Определение структуры списка смежности typedef struct { edgenode *edges[MAXV+1]; int degree[MAXV+1]; int nvertices; int nedges; int directed; graph; /* Информация о смежности */ /* Степень каждой вершины */ /* Количество вершин в графе /* Количество ребер в графе /* Граф ориентированный? */ Каждая переменная edgenode представляет собой запись из трех полей (листинг 6.2). Первое поле описывает вторую конечную точку ребра (у), второе (weight) предоставля- ет возможность присваивать ребру вес, а третье (next) указывает на следующее ребро в списке. Листинг 6.2. Структура переменной edgenode typedef struct { int у; int weight; struct edgenode *next; edgenode; /* Информация о смежности */ /* Вес ребра, если есть */ /* Следующее ребро в списке */ Далее мы рассмотрим несколько сложных алгоритмов, использующих эту структуру данных, включая алгоритмы поиска минимальных остовных деревьев, кратчайших
214 Часть I. Практическая разработка алгоритмов маршрутов и максимальных потоков. То обстоятельство, что для этих задач оптимиза- ции существуют эффективные решения, привлекает к ним наше особое внимание. Вспомните, что для первой рассмотренной нами задачи для взвешенных графов (зада- чи коммивояжера) детерминистического алгоритма решения не существует. 6.1. Минимальные остовные деревья Остовным деревом (spanning tree) графа G = (V. Е) называется подмножество ребер множества Е, которые создают дерево, содержащее все вершины V. В случае взвешен- ных графов особый интерес представляют минимальные остовные деревья, т. е. остов- ные деревья с минимальной суммой весов ребер. Минимальные остовные деревья позволяют решить задачу, в которой требуется соеди- нить множество точек (представляющих города, дома, перекрестки и другие объекты) наименьшим объемом дорожного полотна, проводов, труб и т. п. Любое дерево — это, в сущности, наименьший (по количеству ребер) возможный связный граф, в то время как минимальное остовное дерево является наименьшим связным графом по весу ре- бер. В геометрических задачах набор точек /?ь .... определяет полный граф, в кото- ром ребру (v„ vz) присваивается вес. равный расстоянию между точками р, и pz. На рис. 6.1 показан пример геометрического минимального остовного дерева. Рис. 6.1.1laoop точек (а), его минимальное остовное дерево (б) и кратчайший маршрут от центра дерева (в) Дополнительную информацию о минимальных остовных деревьях см. в разделе 15.3. Минимальное остовное дерево имеет наименьшую протяженность изо всех возможных остовных деревьев. Но граф может содержать несколько минимальных остовных де- ревьев. Более того, все остовные деревья невзвешенного графа (или графа со всеми ребрами одинакового веса) являются минимальными остовными деревьями, т. к. каж- дое из них содержит ровно п - 1 ребер. Такое остовное дерево можно найти с помощью алгоритма обхода в глубину или ширину. Найти минимальное остовное дерево во взвешенном графе гораздо труднее. Тем не менее, в последующих разделах рассматри- ваются два разных алгоритма для решения этой задачи, каждый из которых демонст- рирует пригодность определенных эвристических "жадных" алгоритмов.
Гпава 6 Алгоритмы для работы со взвешенными графами 215 6.1.1. Алгоритм Прима Алгоритм Прима для построения минимального остовного дерева начинает обход с одной вершины и создает дерево, добавляя по одному ребру, пока не будут включены все вершины. "Жадные" алгоритмы принимают решение по следующему шагу, выбирая наилучшее локальное решение изо всех возможных вариантов, не принимая во внимание глобаль- ную структуру. Так как мы ищем дерево с минимальным весом, то естественно пред- ложить "жадный" алгоритм, последовательно выбирающий ребра с наименьшим весом, которые увеличат количество вершин дерева. Псевдокод алгоритма для построения минимального остовного дерева приведен в листинге 6.3. Листинг 6.3. Алгоритм Прима Prim-MST(G) Выбираем произвольную вершину s, с которой надо начинать построение дерева While (остаются вершины, не включенные в дерево) Выбираем ребро минимального веса между деревом и вершиной вне дерева Добавляем выбранное ребро и вершину в дерево Tprlm Очевидно, что алгоритм Прима создает остовное дерево, т. к. добавление ребра между вершиной в дереве и вершиной вне дерева не может создать цикл. I Io почему именно это остовное дерево должно иметь наименьший вес изо всех остовных деревьев? Мы видели достаточно примеров других "жадных" эвристических алгоритмов, которые не дают оптимального общего решения. Поэтому нужно быть особенно осторожным, что- бы доказать любое такое утверждение. В данном случае мы применим метод доказательства от противного. Допустим, чго существует граф G, для которого алгоритм Прима не возвращает минимальное остов- ное дерево. Так как мы создаем дерево инкрементальным способом, то это означает, что существует момент, в который было принято неправильное решение. Перед добав- лением ребра (.г, j') дерево Т’рн,,, состояло из набора ребер, являющихся поддеревом оп- ределенного минимального остовного дерева Д,,,,,, но остовное дерево, полученное до- бавлением к этому поддереву ребра (х.у), больше не является минимальным (рис. 6.2, а). Рис. 6.2. Выдает ли алгоритм Прима неправильное решение? Нет, т. к. v2) > d(.x. у)
216 Часть I. Практическая разработка алгоритмов Но как это могло случиться? Как можно видеть на рис. 6.2, б, в остовном дереве Тт||1 должен быть путь р от вершины х к вершине у. Этот путь должен содержать ребро (vi. v2), в котором вершина v, находится в дереве Грп|11, а вершина v2— нет. Вес этого ребра должен быть, по крайней мере, равен весу ребра (х,у), иначе алгоритм Прима выбрал бы это ребро до ребра (х, у). когда у него была такая возможность. Удалив из дерева Tmjll ребро (vb v2) и вставив вместо него ребро (х.у), мы получим остовное дере- во с весом не большим, чем прежнее дерево. Это означает, что выбор алгоритмом Прима ребра (х.у) не был ошибочным. Таким образом, создаваемое алгоритмом Прима остовное дерево должно быть минимальным. Реализация Алгоритм Прима создает минимальное остовное дерево поэтапно, начиная с указанной вершины. На каждом этапе мы вставляем в остовное дерево одну новую вершину. "Жадного" алгоритма достаточно для гарантии правильности выбора: чтобы соединить вершины в дереве с вершиной вне дерева он всегда выбирает ребро с наименьшим ве- сом. Самой простой реализацией этого алгоритма было бы представлять каждую вер- шину булевой переменной, указывающей, находится ли уже эта вершина в дереве (массив intree в листинге 6.4). а потом просмотреть все ребра на каждом этапе, чтобы найти ребро с минимальным весом и единственной вершиной intree. В листинге 6.4 приведена реализация алгоритма Прима. Листинг 6.4. Реализация алгоритма Прима ... —- • ------— -.................. ....... .. _ .............. primfgraph ♦g, int start) ( int i; /* Счетчик */ edgenode *p; /* Временный указатель */ bool intree[MAXV+1]; /★ Вершина уже в дереве? */ int distance[MAXV+1]; /* Стоимость добавления к дереву */ int v; /* Текущая вершина для обработки */ int w; /* Кандидат на следующую вершину */ int weight; /★ Вес ребра ★/ int dist; /* Наилучшее текущее расстояние от начала */ for (i=l; i<=g->nvertices; i++) < intree[i| = FALSE; distance[i] = MAXINT; parent[i] = -1; } distance[start] 0; v = start; while (intree[v] == FALSE) { intree[v] = TRUE; p = g->edges[v]; while (p != NULL) w = p->y; weight = p->weight; if ((distance[w] > weight) Si (intree[w] == FALSE)) | distance[w] = weight;
Гпава 6. Алгоритмы для работы со взвешенными графами 217 parent[w] = v; р = p->next; ' = 1; dist = MAXINT; for (i=l; i<=g->nvertices; i++) if ((intree[i] -= FALSE) && (dist > distance[i])) { dist distance[i]; v = i; В процедуре поддерживается информация о весе каждого ребра, связывающего с дере- вом каждую находящуюся вне дерева вершину. Из множества этих ребер к дереву до- бавляется ребро с наименьшим весом. После каждой вставки ребра необходимо обнов- лять стоимость (в виде весов ребер) просмотра вершин, не входящих в дерево. Но так как единственным изменением в дереве является последняя добавленная вершина, то все возможные обновления весов будут определяться исходящими ребрами этой вер- шины. Анализ Итак, алгоритм Прима является правильным, но насколько он эффективен? Это будет зависеть от используемых для его реализации структур данных. В псевдокоде алгоритм Прима исполняет п циклов, просматривая все т ребер в каждом цикле, что дает нам алгоритм с временной сложностью, равной O(inn). Но в нашей реализации проверка всех ребер в каждом цикле ие выполняется. Мы рас- сматриваем не более п известных ребер с наименьшим весом, представленных в масси- ве parent, и не более п ребер, исходящих из последней добавленной вершины v для обновления этого массива. Благодаря наличию булева флага для каждой вершины, ука- зывающего, находится ли эта вершина в дереве, мы можем проверить, связывает ли текущее ребро дерево с вершиной вне дерева, за постоянное время. Общее время исполнения алгоритма Прима равно О(н2), что является хорошей иллюст- рацией того, как удачная структура данных способствует ускорению алгоритмов. Более того, применение более сложной структуры данных в виде очереди с приоритетами позволяет осуществить реализацию алгоритма Прима с временной сложностью, равной О(т + Mlg/i), поскольку удается быстрее найти ребро с минимальным весом для расши- рения дерева в каждом цикле. Само минимальное остовное дерево или его вес можно реконструировать двумя раз- ными способами. Самым простым способом было бы дополнить код в листинге 6.4 операторами для вывода обнаруженных ребер или общего веса всех выбранных ребер. В качестве альтернативы можно закодировать топологию дерева в массиве parent, что- бы она совместно с первоначальным графом предоставляла всю информацию о мини- мальном остовном дереве.
218 Часть I. Практическая разработка алгоритмов 6.1.2. Алгоритм Крускала Алгоритм Крускала является альтернативным подходом к построению минимальных остовных деревьев, который оказывается более эффективным на разреженных графах. Подобно алгоритму Прима, алгоритм Крускала является "жадным", но, в отличие от него, он не создает дерево, начиная с определенной вершины. Алгоритм Крускала наращивает связные компоненты вершин, создавая в конечном счете минимальное остовное дерево. Первоначально каждая вершина является отдель- ной компонентой будущего дерева. Алгоритм последовательно ищет ребро для добав- ления в расширяющийся лес путем поиска самого легкого ребра среди всех ребер, со- единяющих два дерева в лесу. При этом выполняется проверка, не находятся ли обе конечные точки ребра-кандидата в одной и той же связной компоненте. В случае по- ложительного результата проверки такое ребро отбрасывается, т. к. добавление его создало бы цикл в будущем дереве. Если же конечные точки находятся в разных ком- понентах, то ребро принимается и соединяет две компоненты в одну. Так как каждая связная компонента всегда является деревом, то выполнять явную проверку на циклы нет необходимости. В листинге 6.5 приводится псевдокод алгоритма Крускала. ; Листинг 6.5. Алгоритм Крускала • .......................... ................... .......................... Kruskal-MST(G) Помещаем ребра в очередь с приоритетами, упорядоченную по весу count = О while (count < n - 1) do рассматриваем следующее ребро (v, w) if (component (v) / component(w)) добавляем в дерево Tkruskai объединяем component(v) и component(w) Так как этот алгоритм вставляет п- 1 ребер, не порождая циклы, он, очевидно, создает остовное дерево для любого связного графа. Но откуда мы знаем, что это минимальное остовное дерево? Будем доказывать от противного. Допустим, что оно не является та- ковым. Так же, как и в доказательстве правильности алгоритма Прима, допустим, что существует какой-то граф, для которого он возвращает неправильный ответ (рис. 6.3). Рис. 6.3. Выдает ли алгоритм Крускала неправильное решение? Нет. т. к. <7(vb v2) > d(x, у) б)
Гпава 6. Алгоритмы для работы со взвешенными графами 219 В частности, должно существовать ребро (л\у), первоначальная вставка которого в де- рево Tkfwkui не дает ему быть минимальным остовным деревом Г,,,,,,. Вставка такого ребра (х.у) в дерево ТПШ1 создаст цикл с путем от вершины г к вершине у. Так как вер- шины х и у были в разных компонентах во время вставки ребра (х. у), то. по крайней мере, одно ребро (скажем, ребро (vt, v,)) в этом пути было бы проверено алгоритмом Крускала позже, чем ребро (х,у). Но это означает, что вес ir(vi.v2)> w(x. у), поэтому обмен местами этих двух ребер дает нам дерево с весом, самое большее, Г1111П. Поэтому выбор ребра (х,у) не мог быть ошибочным, из чего следует правильность алгоритма. Какова временная сложность алгоритма Крускала? Упорядочивание т-го количества ребер занимает время Ctynlgw). Цикл for выполняет т итераций, в каждой из которых проверяется связь двух деревьев и ребро. При наиболее очевидном подходе это можно реализовать посредством поиска в ширину или глубину в разреженном графе, имею- щем, самое большее, т ребер и п вершин, что дает нам алгоритм с временем исполне- ния О(тп). Но если удастся выполнять проверку компонент быстрее, чем за время ()(/?). то можно получить более эффективную реализацию алгоритма. В действительности, рассматри- ваемая в следующем разделе сложная структура данных позволяет выполнять такую проверку за время O(lgn). С использованием этой структуры алгоритм Крускала испол- няется за время (?(/nlgw), что быстрее, чем время исполнения алгоритма Прима для разреженных графов. В очередной раз обращаю ваше внимание на эффект, оказывае- мый правильной структурой данных на реализацию простых алгоритмов. Реализация Реализация алгоритма Крускала показана в листинге 6.6. Листинг 6.6. Реализация алгоритма Крускал kruskal(graph *g) int i; /* Счетчик ♦/ set_union s; /* Структура данных set_union */ edge_pair e[MAXV+1]; /* Структура данных массива ребер */ bool weight_compare(); set_union_init(&s, g->nvertices); to_edge_array(g, e); /* Сортируем ребра по возрастающему весу ’/ qsort(&e, g->nedges, sizeof(edge_pair),weight_compare); for (i=0; i<(g->nedges); i++) { if !same_component(s,e[i].x,e[i],y)) { printf("edge (%d,%d) inMSTXn",e[i],x,e[i].y); union_sets(Ss,e[i].x,e(i],y); } ) На рис. 6.4 показаны граф G (a) и минимальные остовные деревья для него, создавае- мые алгоритмами Прима (б) и Крускала (в). Числами возле ребер указан порядок их вставки; приоритет вставки равнозначных ребер определяется произвольным путем.
220 Часть I. Практическая разработка алгоритмов Рис. 6.4. Граф С и минимальные остоиные деревья, создаваемые алгоритмами Прима и Крускала 6.1.3. Поиск-объединение Разбиением множества (set partition) называется разбиение элементов некоего универ- сального множества (например, целых чисел от I до п) на несколько непересекающих- ся (disjointed) подмножеств, таким образом, каждый элемент исходного множества должен быть ровно в одном из получившихся подмножеств. Разбиения множества воз- никают естественным образом в таких задачах на графах, как связные компоненты (каждая вершина находится ровно в одной компоненте связности) и раскраска графа (человек может быть женского или мужского пола, но мы исключаем варианты отсут- ствия пола или принадлежности к обоим полам). Алгоритмы для генерирования раз- биений множеств и связных объектов представлены в разделе 14 6. Связные компоненты графа можно представить в виде разбиения множества. Для эф- фективного исполнения алгоритма Крускала нам нужна структура данных, обеспечи- вающая эффективную поддержку таких операций: ♦ одна компонента (v\, vj— выполняет проверку, находятся ли вершины V| и V2 в одной и той же компоненте связности текущего графа; ♦ объединить компоненты (С|, СД— объединяет данную пару связных компонент в одну по предоставленному ребру между ними. Каждая из двух очевидных структур-кандидатов для поддержки этих операций в дей- ствительности обеспечивает эффективную поддержку только одной из них. В частно- сти. явно пометив каждый элемент номером его компоненты, мы сможем выполнять проверку "одна компонента" за постоянное время, но обновление номеров компонент после слияния будет занимать линейное время. С другой стороны, операцию слияния компонент можно рассматривать как вставку ребра в граф, но тогда нам нужно выпол- нить полный обход графа, чтобы найти связные компоненты в случае необходимости. Решением является применение специальной структуры данных (рис. 6.5), где каждое подмножество представляется в виде "обратного" дерева с указателями от узлов к их родителям. Каждый узел дерева содержит элемент множества, а имя множеству присваивается по корневому ключу. По причинам, которые станут понятными далее, мы для каждой вершины v отслеживаем также информацию о количестве элементов поддерева, имеющего корень в этой вершине. В листинге 6.7 приводится определение структуры данных set union.
Гпава 6. Алгоритмы для работы со взвешенными графами 221 Рис. 6.5. Пример структуры данных: в виде деревьев (а) и в виде массива (б) Листинг 6.7. Определение структуры данных setjunion ' *......----------------------------...........—................^.л«-....л..^.1.. ..... ...-------------------- --------1 typedef struct { int p[SET_SIZE+l]; /* Родительский элемент */ int size [SET_SIZE+1]; /* Количество элементов в поддереве t */ int n; /* Количество элементов в множестве */ set_union; Реализуем необходимые нам операции над компонентами посредством двух более про- стых операций — find и union: ♦ find(i) — находит корень дерева, содержащего элемент /, переходя по указателям на родителей до тех пор, пока это возможно. Возвращает метку корня; ♦ union(i, j) — связывает корень одного из деревьев (скажем, дерева, содержащего элемент /) с корнем дерева, содержащего другой элемент (скажем, элемент/); таким образом, операция find(i) теперь эквивалентна операции find(j). Мы стремимся минимизировать время исполнения любой последовательности опера- ций union и find. Так как древесные структуры могут быть сильно разбалансирован- ными, то нам необходимо ограничить высоту наших деревьев. Самым очевидным спо- собом контроля высоты будет принятие правильного решения, какой из двух корней компонент должен стать корнем объединенной компоненты после каждого выполне- ния операции union. Минимизировать высоту дерева можно, сделав меньшее дерево поддеревом большего. Почему это так? Потому что высота всех узлов в корневом поддереве остается одина- ковой, в то время как высота всех узлов, вставленных в это дерево, увеличивается на единицу. Таким образом, объединение с меньшим деревом позволяет оставить без из- менений высоту большего дерева. Реализация Реализация операций union и find приводится в листинге 6.8. Листинг 6.8. Реализация операций union и find set_union_init (set union *s, int n) int i; for (i=l; i<=n; i++) { s->p[i] =i; /* Счетчик */
222 Часть I. Практическая разработка алгоритмов s->size[i] = 1; } s->n =• n; int find(set_union *s, int x) { if (s->p[x] == x) return(x); else return(find(s, s->p[x]) ); ) int unionsets(set_union *s, int si, int s2) int rl, r2; /* Корни подмножеств */ rl = find(s, si); r2 = find(s, s2); if (rl == r2) return; /* Уже находится в этом множестве */ if (s->size[rl] >= s->size[r2]) { s->size[rl] = s->size[rl] + s->size[r2]; s->p[ r2 ] = rl; ) else t s->size[r21 = s->size[rl] + s->size[r2]; s->p[ rl ] = r2; } bool samecomponent(set_union *s, int si, int s2) return I find(s, si) = find(s, s2) ); Анализ При каждом исполнении операции union дерево с меньшим количеством узлов стано- вится потомком. Но каким высоким может быть такое дерево в зависимости от количе- ства узлов в нем? Рассмотрим, каким может быть наименьшее возможное дерево с вы- сотой Л. Высота деревьев с одним узлом равна 1. Наименьшее дерево высотой 2 имеет два узла, полученных в результате объединения двух одноузловых деревьев. Когда из- меняется высота? При объединении двух одноузловых деревьев этого не происходит, т. к. эти деревья просто становятся потомками корневого дерева высотой 2. И только при объединении двух деревьев высотой 2 мы получаем дерево высотой 3, имеющее чегыре узла. Уловили закономерность? Высота дерева увеличивается на единицу при удваивании количества узлов. Сколько раз мы можем удваивать количество узлов, пока не исчер- паем все п узлов? Нам удастся сделать максимум lg2« удваиваний. Таким образом, мы можем выполнять как операцию объединения union, так и операцию поиска find за время O(logH)- что достаточно хорошо для алгоритма Крускала. В действительности, поиск-объединение может выполняться даже быстрее (см. раздел 12 5).
Гпава 6. Алгоритмы для работы со взвешенными графами 223 6.1.4. Разновидности остовных деревьев Алгоритм поиска минимального остовного дерева обладает несколькими интересными свойствами, полезными для решения родственных задач, в частности, поиска таких разновидностей остовных деревьев, как: ♦ максимальные остовные деревья. Допустим, телефонная компания наняла подряд- чика для прокладки телефонного кабеля между домами. Компания платит подряд- чику с учетом протяженности установленного кабеля. К сожалению, подрядчик ока- зался недобросовестным, и пытается получить наибольший доход ог этой работы. В терминах теории графов для этого ему нужно создать наиболее тяжелое остовное дерево. Максимальное остовное дерево любого графа можно найти, просто поменяв у всех ребер знак веса с плюса на минус и применив алгоритм Прима. Максималь- ное по модулю дерево в графе с отрицательным весом ребер будет максимальным остовным деревом в первоначальном графе. Большинство алгоритмов для графов не поддается адаптации под отрицательные числа с такой легкостью. Более того, алгоритмы поиска кратчайшего пути некор- ректно работают с отрицательными числами и не возвращают самый длинный путь при использовании этого подхода; ♦ остовные деревья с минимальным произведением весов ребер. Допустим, нам нужно найти остовное дерево с минимальным произведением весов ребер, полагая, что вес всех ребер положительный. Так как Ig(a-A) = lg(«) + lg(A), то минимальное остовное дерево графа, в котором веса ребер были заменены их логарифмами, даст нам ос- товное дерево с минимальным произведением весов ребер первоначального графа; ♦ минимальное остовное дерево. Иногда требуется найти остовное дерево, в котором наибольший вес ребра минимален среди всех таких деревьев. Любое минимальное остовное дерево обладает этим свойством. Доказательство этого следует непосред- ственно из факта правильности алгоритма Крускала. Минимальные остовные деревья имеют интересные области применения, когда вес ребра представляет стоимость, пропускную способность и т. д. Менее эффектив- ным, но концептуально более простым способом решения таких задач может быть удаление всех "тяжелых" ребер графа с проверкой полученного графа на связность. Эту проверку можно выполнить простым обходом в ширину или глубину. Если все т ребер графа имеют разный вес, то такой граф имеет единственное мини- мальное остовное дерево. В противном случае возвращаемое алгоритмом Прима или Крускала дерево определяется способом, который применяется в алгоритме для выбора одного из нескольких ребер с одинаковым весом. Существует две важных разновидности минимальных остовных деревьев, которые не поддаются решению этими способами. ♦ Дерево Штейнера. Допустим, нам нужно проложить проводку (ребра графа) между домами (вершины графа), но при выполнении этой задачи мы можем создавать до- полнительные промежуточные пункты (вершины) в качестве общих соединений. Дерево, полученное в результате решения этой задачи, называется минимальным деревом Штейнера и рассматривается в разделе 16.10.
224 Часть I Практическая разработка алгоритмов ♦ Остовное дерево с минимальной степенью веришн. Предположим, нам нужно найти минимальное остовное дерево, у которого наибольшая степень вершины минималь- на. Таким деревом будет простой путь си-2 вершинами степени 2 и двумя конеч- ными вершинами степени 1. Путь, который проходит через каждую вершину только один раз, называется гамильтоновым путем, и рассматривается в разделе 16.5. 6.2. История из жизни. И все на свете только сети Однажды мне сообщили, что небольшая компания по тестированию печатных плат нуждается в консультации по алгоритмам. Вскоре я оказался в непримечательном зда- нии за одним столом с президентом компании Integri-Test и ведущим техническим спе- циалистом. — Мы являемся ведущей компанией, поставляющей роботизированные устройства для тестирования печатных плат. Наши клиенты предъявляют очень высокие требования к надежности выпускаемых ими печатных плат. В частности, перед тем как устанавли- вать компоненты на печатную плату, они проверяют, что на ней нет разрывов в дорож- ках. Это означает, что им нужно проверить, что каждая пара точек на плате, которые должны быть соединенными, в действительности соединены. — И как вы выполняете это тестирование? — спросил я. — Мы используем два роботизированных манипулятора, оснащенные электрическими щупами. Для проверки соединения между двумя точками манипуляторы подключают щупы к этим обеим точкам. Если дорожка между точками исправна, то щупы замыка- ют цепь. А для сетей мы фиксируем один из манипуляторов в одной точке, а вторым последовательно проверяем все остальные точки сети. — Погодите! — вскричал я. — Что такое сеть на печатной плате? — Печатная плата содержит несколько наборов точек, соединенных между собой по- средством металлизированных дорожек (рис. 6.6, а). Вот эти наборы точек и соеди- няющие их дорожки мы и называем сетями. Иногда сеть состоит из двух точек, т. е. из одного изолированного провода, а иногда содержит 100-200 точек, как в случае с про- водом питания или заземления. — Понятно. Значит, у вас есть список всех соединений между парами точек на печат- ной плате и вы хотите проверить целостность этих соединяющих дорожек. Он отрицательно покачал головой. — Не совсем так. Вход для нашей программы тес- тирования состоит только из контактных точек сети (рис. 6.6, б). Мы не знаем распо- ложения отрезков соединяющей дорожки, но это и не требуется. Все, что нам нуж- но, — это проверить, что точки сети соединены вместе. Для этого один щуп приставля- ется к самой левой точке сети, а второй перемещается по всем остальным точкам, проверяя, что все они соединены с первой точкой. В таком случае они должны быть соединены друг с другом. — Хорошо. Значит, правый манипулятор должен посетить все остальные точки сети. А каким образом вы определяете последовательность этих посещений?
Гпава 6 Алгоритмы для работы со взвешенными графами 225 Рис. 6.6. Пример сети печатной платы: металлизированная соединяющая дорожка (а), контактные точки 66). минимальное остовное дерево контактных точек бес- контактные точки, разбитые на кластеры (г) На этот вопрос ответил технический специалист. — Мы сортируем точки слева напра- во и обходим их в этом порядке. Это хороший способ? — Вы что-нибудь знаете о задаче коммивояжера? — спросил я. Он был инженер-электрик, а не программист. — Нет. А что это? — Это название задачи, которую вы пытаетесь решить. Нужно посетить набор точек в порядке, который минимизирует время обхода. Алгоритмы для решения этой задачи всесторонне изучены. Для небольших сетей можно найти оптимальный маршрут мето- дом полного перебора. А для больших сетей очень хорошее приближение оптимально- го маршрута можно найти посредством эвристических алгоритмов. — Для более под- робной информации о решении задачи коммивояжера я бы показал им раздел 16.4 этой книги, если бы она была у меня под рукой. Президент компании что-то записал в своем блокноте, а потом нахмурился.— Хоро- шо. Допустим, вы сможете лучше упорядочить точки. Но проблема заключается не в этом. Когда мы наблюдаем за работой нашего робота, то иногда видим, что правый манипулятор доходит до самого правого края платы для данной сети, в то время как левый манипулятор стоит неподвижно. Мне кажется, что было бы полезно разбивать сети на меньшие части, чтобы сбалансировать нагрузку на манипуляторы. Я уселся поудобнее и стал обдумывать задачу. Оба манипулятора (левый и правый) решают взаимосвязанные задачи коммивояжера. Левый манипулятор перемещается от одной к другой самой левой точке каждой сети, в то время как правый посещает все другие точки каждой сети. Разбив каждую сеть на несколько меньших сетей, мы мо- жем избежать ситуации, когда правый манипулятор проходит путь через всю плату. Кроме того, разбивка всей сети на части увеличит количество точек в задаче комми- вояжера для левого манипулятора, вследствие чего каждое его перемещение также бу- дет небольшим. — Вы правы. Разбивка больших сетей на несколько меньших должна повысить эффек- тивность работы манипуляторов. Сети должны быть небольшого размера, как по коли- честву точек, так и по физической площади. Но нам необходимо иметь гарантию в том. 8 Зак. 3741
226 Часть I. Практическая разработка алгоритмов что если мы проверим связность каждой малой сети, то также проверим связность большой сети, членами которой они являются. Одной общей точки для двух малых сетей будет достаточно, чтобы доказать, что состоящая из этих сетей большая сеть также связная, т. к. ток может протекать между любой парой точек. Теперь нам нужно было решить задачу разбиения каждой сети на несколько меньших сетей, что является задачей кластеризации, или группирования. Для группирования часто применяются минимальные остовные деревья (см. раздел 15.3). Это и было ре- шением нашей конкретной задачи кластеризации. В частности, мы могли найти мини- мальное остовное дерево точек сети и разбить его на несколько кластеров, разрывая любое слишком длинное ребро остовного дерева (рис. 6.6, в). Как можно видеть на рис. 6.6, г, каждый кластер точек будет иметь ровно одну общую точку с другим кла- стером, что обеспечивает связность, т. к. мы охватываем все ребра остовного дерева. Форма кластеров следует из расположения точек в сети. Если точки расположены на плате вдоль линии, то минимальным остовным деревом будет маршрут, а кластерами будут пары точек. А если точки расположены плотной группой, то этот кластер будет хорошей площадкой для игры в классики правым манипулятором. Я объяснил моим собеседникам идею создания минимального остовного дерева графа. Президент выслушал, что-то опять записал в своем блокноте и снова нахмурился. — Мне нравится ваша идея кластеризации. Но минимальные остовные деревья опре- деляются на графах, а у нас есть только точки. Где мы возьмем веса для ребер? — О, мы можем рассматривать их как полный граф, в котором соединена каждая пара точек. А вес ребра определяется как расстояние между двумя точками. Или не опреде- ляется...? Я опять начал размышлять. Стоимость ребра должна отражать время перемещения манипулятора между двумя точками. В то время как расстояние связано с временем перемещения, это не обязательно взаимозаменяемые понятия. — У меня вопрос относительно вашего робота. Скорость перемещения манипулятора по горизонтали такая же. как и по вертикали? Они подумали немного.— Да, такая же. Мы используем одинаковые двигатели для перемещения манипулятора по горизонтали и вертикали. Так как оба двигателя рабо- тают независимо друг от друга, то манипулятор можно перемещать одновременно по горизонтали и по вертикали. — То есть, перемещение на один фут влево и один фут вверх занимает точно такое же время, как перемещение на один фут только влево? Это означает, что вес каждого реб- ра должен соответствовать не евклидову расстоянию между двумя точками, а наи- большему расстоянию между ними либо по горизонтали, либо по вертикали. Это назы- вается метрикой £,Л. но мы можем зафиксировать ее, изменяя вес ребер в графе. Проис- ходит ли еще что-нибудь необычное с вашими роботами? — спросил я. — Роботу нужно некоторое время, чтобы развить рабочую скорость. Также, наверное, нужно принять во внимание ускорение и замедление движения манипулятора. — Абсолютно верно. Чем точнее мы смоделируем перемещение манипулятора между двумя точками, тем лучшим будет наше решение. Но теперь у нас есть очень четкая формулировка решения. Давайте закодируем его и посмотрим, насколько оно хорошо.
Гпава 6. Алгоритмы для работы со взвешенными графами 227 Они явно сомневались, что этот подход принесет какую-нибудь пользу, но согласились подумать. Несколько недель спустя они позвонили мне и сообщили, что новый алго- ритм сокращает дистанцию маршрута примерно на 30% по сравнению с их первона- чальным подходом за счет небольшого увеличения предпроцессорных вычислений. Но так как их тестовое оборудование стоило 200 000 долларов, а персональный компью- тер— всего лишь 2 000 долларов, то это был замечательный компромисс. Кроме того, это было особенно выгодным, т. к. в случае тестирования партии одинаковых плат предварительную обработку нужно было выполнять только один раз. Ключевой идеей, приведшей к успешному решению, было моделирование задачи по- средством классических алгоритмов для решения задач на графах. Я понял, что передо мной задача коммивояжера, как только разговор зашел о минимизации протяженности перемещений манипулятора. Когда я выяснил, что для проверки связности точек они неявно использовали звездообразное остовное дерево, было естественным задать во- прос, не улучшило бы производительность минимальное остовное дерево. Эта идея проложила путь к идее кластеризации, т. е. разбиению каждой сети на несколько меньших сетей. Наконец, внимательный подход к разработке метрик расстояний, что- бы точно смоделировать издержки самого манипулятора, позволил нам внедрить в ре- шение такие сложные свойства, как ускорение и замедление манипулятора, не изменяя фундаментальной модели графа или структуры алгоритма. Подведение итогов Большинство приложений с использованием графов можно свести к стандартным задачам на графах, к которым можно применить хорошо известные алгоритмы. Это такие задачи, как построение минимальных остовных деревьев, кратчайших путей и прочие задачи, представленные в каталоге задач этой книги. 6.3. Поиск кратчайшего пути Путь (path) — это последовательность ребер, соединяющих две вершины. Так как ки- норежиссер Мел Брукс (Mel Brooks) приходится двоюродным братом мужу сестры моего отца, то в графе дружеских отношений между мной и ним есть путь (рис. 6.7), хотя мы никогда лично не встречались. я (Стив) отец тетя Ева дядя Ленни кузен Мел Рис. 6.7. Мел Брукс приходится двоюродным братом мужу сестры моего отца Но если бы я хотел произвести на читателей впечатление, заявляя о близком знакомст- ве с кузеном Мелом, то лучше бы было говорить, что мой дядя Ленни и он росли вме- сте. Через дядю Ленни длина пути дружеских отношений между мной и кузеном Ме- лом составляет 2 звена, адлина пути кровных и родственных связей составляет 4 звена. Наличие нескольких путей свидетельствует о важности определения кратчайшего пу- ти между двумя узлами, даже в приложениях, не имеющих отношения к транспорти- ровке.
228 Часть I. Практическая разработка алгоритмов В невзвешенном графе кратчайший путь от вершины х к вершине t можно найти по- средством обхода в ширину с исходной точкой в вершине s. Дерево обхода в ширину предоставляет путь, состоящий из минимального количества ребер. Это кратчайший путь, если все ребра имеют одинаковый вес. Но во взвешенных графах обход в ширину не предоставляет кратчайшего пути. В дан- ном типе графа кратчайший путь может содержать большее количество ребер, точно так же, как и кратчайший (во временном измерении) путь от дома к офису может про- ходить по нескольким отрезкам второстепенных дорог, вместо одного или двух пря- мых отрезков главной дороги, как показано на рис. 6.8. Рис. 6.8. Кратчайший путь от точки х к точке I можег проходить через несколько промежуточных точек В этом разделе мы рассмотрим два разных алгоритма поиска кратчайшего пути во взвешенных графах. 6.3.1. Алгоритм Дейкстры Алгоритм Дейкстры является предпочтительным методом поиска кратчайших путей в графах со взвешенными ребрами и/или вершинами. Алгоритм находит кратчайший путь от заданной начальной вершины х ко всем другим вершинам графа, включая тре- буемую конечную вершину /. Допустим, что кратчайший путь от вершины х к вершине / графа G проходит через оп- ределенную промежуточную вершину х. Очевидно, что этот путь должен содержать кратчайший путь от вершины s к вершине х, в качестве префикса, ибо в противном случае можно было бы сократить путь s—t, используя более короткий префиксный путь х—х. Таким образом, прежде чем найти кратчайший путь от начальной вершины х к ко- нечной вершине /. нам нужно найти кратчайший путь от начальной вершины х к про- межуточной вершине х. Алгоритм Дейкстры работает поэтапно, находя на каждом этапе кратчайший путь от вершины х к некой новой вершине. Говоря конкретно, вершинах такова, что сумма distfs. v,) + и’(г„х) минимальна для всех необработанных I < i < п. где w(/,j)— длина ребра между вершинами i и у, a dist(j,j) —- длина кратчайшего пути между ними. Здесь напрашивается стратегия, аналогичная динамическому программированию. Кратчайший путь от вершины s к самой себе является тривиальным, при условии от- сутствия ребер с отрицательным весом, поэтому dist(s.s) = 0. Если (х. у) является са-
Гпава 6 Алгоритмы для работы со взвешенными графами 229 мым легким ребром, входящим в вершину х, то это означает, что dist(s,y) = w(x, у). Оп- ределив кратчайший путь к вершине х. мы проверяем все исходящие из нее ребра, чтобы узнать, не существует ли лучшего пути из начальной вершины х к какой-либо неизвест- ной вершине через вершину х. Псевдокод этого алгоритма представлен в листинге 6.9. Листинг 6.9. Алгоритм Дейкстры ShortestPath-Dijkstra (G, s, t) known == {s) for i 1 to n, dist[il = ~ for each edge (s, v), dist[v] w(s, v) last s while (last + t) select v,»...., неизвестная вершина, минимизирующая dist[v] for each edge 'v,lel,, x) , dist[x] = min[dist[x], dj.St[Vnex‘] + W(Vnext, X) ] last = vri known = known (vr. Основная идея алгоритма Дейкстры очень похожа на основную идею алгоритма При- ма. В каждом цикле мы добавляем одну вершину к дереву вершин, для которых мы знаем кратчайший путь из вершины х. Так же, как и в алгоритме Прима, мы сохраняем информацию о наилучшем пути на данное время для всех вершин вне дерева и встав- ляем их в дерево в порядке возрастания веса. Разница между алгоритмом Дейкстры и алгоритмом Прима состоит в том, как они оце- нивают привлекательность каждой вершины вне дерева. В задаче поиска минимально- го остовного дерева нас интересовал только вес следующего возможного ребра дерева. А при поиске кратчайшего пути нам нужна также информация о ближайшей к вершине 5 вершине вне дерева, т. е.. кроме веса нового ребра, мы хотим знать еще и расстояние от вершины х к смежной с ней вершине дерева. Реализация Псевдокод алгоритма Дейкстры скрывает его сходство с алгоритмом Прима. В дейст- вительности. разница между этими двумя алгоритмами очень незначительная. В лис- тинге 6.10 приводится реализация алгоритма Дейкстры, которая, по сути, является реа- лизацией алгоритма Прима, в которой изменены всего лишь три строчки кода (отмече- ны в листинге комментарием) и имя функции. Листинг 6.10. Реализация алгоритма Дейкстры dijkstra (graph *g, int start) ( int i; edgenode *p; bool intree[MAXV+1]; int distance[MAXV+1]; int v; /* Для Прима — prim(g, start) */ /* Счетчик */ /* Временный указатель */ /* Вершина уже в дереве? +/ /* Расстояние вершины от начала */ /* Текущая вершина для обработки */
230 Часть I Практическая разработка алгоритмов int w; /* Кандидат на следующую вершину */ int weight; /* Вес ребра */ int dist; /* Наилучшее текущее расстояние от начала */ for (i=l; i<=g->nvertices; i++) { intree[i] = FALSE; distance[i] = MAXINT; parent[i] = -1; ) distance[start] = 0; v = start; while (intree[v] == FALSE) { intree[v] = TRUE; p = g->edges[v]; while (p != NULL) ( w = p->y; weight = p->weight; /* ИЗМЕНЕНИЕ */ if (distance[w] > (distance[v]+weight)) { /* ИЗМЕНЕНИЕ */ distance[w] = distance[v]+weight; /* ИЗМЕНЕНИЕ */ parent[w] = v; ) p = p->next; } v = 1; dist = MAXINT; for (i=l; i<=g->nvertices; i++) if ((intree[i] = FALSE) && (dist > distance[i])i ( dist = distance[i]; v = i; ) Этот алгоритм находит не только кратчайший путь от вершины х к вершине /, но и кратчайший путь от вершины s ко всем другим вершинам, что определяет минималь- ное остовное дерево с корнем в вершине s. Для неориентированных графов это будет дерево обхода в ширину, но, в общем, алгоритм предоставляет кратчайший путь от вершины 5 ко всем остальным вершинам. Анализ Какова временная сложность алгоритма Дейкстры? В том виде, как он реализован здесь, временная сложность алгоритма равна (\п\ Это такое же время исполнения, как и для алгоритма Прима, что не удивительно— ведь за исключением дополнитель- ного условия, это точно такой же алгоритм, как и алгоритм Прима. Длина кратчайшего пути от начальной вершины start к заданной вершине / равна точ- но значению distance[t]. Каким образом мы можем найти сам путь с помощью алго- ритма Дейкстры? Точно так же. как в процедуре findpatho (см. листинг 5.13), мы следуем обратным указателям на родительские узлы от вершины /, пока мы не дойдем до начальной вершины (или пока не получим —1, если такого пути не существует).
Гпава 6. Алгоритмы для работы со взвешенными графами 231 Алгоритм Дейкстры работает правильно только на графах, в которых нет ребер с отри- цательным весом. Дело в том, что при построении пути может встретиться ребро с от- рицательным весом настолько большим по модулю, что оно полностью изменит оптимальный путь от вершины х к какой-то другой вершине, которая уже включена в дерево. Образно говоря, самым выгодным путем к соседу по лестничной клетке может оказаться путь через банк на другом конце города, если этот банк выдает за каждое посещение достаточно большое вознаграждение, делающее такой маршрут вы- годным. В большинстве приложений ребра с отрицательным весом не используются, что делает это ограничение на пригодность алгоритма Дейкстры чисто теоретическим. Рассматри- ваемый далее алгоритм Флойда также работает правильно только в том случае, если отсутствуют циклы с отрицательной стоимостью, которые заметно искажают структуру кратчайшего пути. Если банк в нашем примере не ограничивает вознаграждение одно- разовой выплатой, то выгода от его бесконечного посещения заставит вас навсегда от- казаться от идеи навестить соседа. Остановка для размышлений. Кратчайший путь с учетом веса вершин ЗАДАЧА. Допустим, что нам нужно разработать эффективный алгоритм для поиска самого дешевого пути в графе со взвешенными вершинами, а не со взвешенными реб- рами. То есть, стоимостью пути от вершины х к вершине у является сумма весов всех вершин в пути. Решение. Естественным решением будет адаптация алгоритма для графов со взвешен- ными ребрами (например, алгоритма Дейкстры) для этой ситуации. Очевидно, что та- кой подход возможен. Мы просто заменим все упоминания веса ребра на вес его ко- нечной вершины. Эту информацию можно хранить в массиве и обращаться к ней по мерс надобности. Но лично я предпочитаю вместо этого модифицировать граф таким образом, чтобы алгоритм Дейкстры дал желаемый результат. Для этого мы установим в качестве веса ориентированного ребра (/,/) вес вершины /. Применение алгоритма Дейкстры на та- ком графе дает требуемый результат. Эту технику можно распространить на другие задачи, например, такие, в которых вес имеется как у ребер, так и у вершин. 6.3.2. Кратчайшие пути между всеми парами вершин Допустим, что нам нужно найти "центральную" вершину графа, т. е. вершину с крат- чайшими путями ко всем остальным вершинам. Практическим примером такой задачи может быть выбор места для открытия офиса. Или, допустим, вам нужно узнать диа- метр графа, т. е. максимальное кратчайшее расстояние между всеми парами вершин. Практическим примером данной задачи может быть определение максимального вре- менного интервала для доставки письма или сетевого пакета. В этих случаях нужно найти кратчайшие пути между всеми парами вершин графа.
232 Часть I. Практическая разработка алгоритмов Одним из возможных решений этой задачи будет последовательное применение алго- ритма Дейкстры для каждой из возможных п начальных вершин Но лучше будет ис- пользовать алгоритм Флойда-Варшалла. чтобы создать матрицу расстояний размером «хи из исходной матрицы весов графа. Алгоритм Флойда лучше всего использовать на матрицах смежности. В этом нет ниче- го необычного, т. к. нам требуется сохранить все п попарных расстояний в любом слу- чае. В листинге 6. II приводится определение структуры для хранения требуемой ин- формации. Листинг 6.11. Определение типа adjacencyjnatrix typeclef struct int weight [MAXV+1] [MAXV+1] , int nvertices; 1 adjacency matrix; /* Информация о смежности/весе •/ /* Количество вершин в графе *! Под тип adjacency matrix выделяется память для матрицы максимально большого раз- мера. Кроме того, в нем предусмотрено поле для хранения информации о количестве вершин в графе. Критическим вопросом в реализации матрицы смежности является способ обозначения отсутствующих в графе ребер. Обычно для обозначения присутствующих ребер не- взвешенных графов используется I. а отсутствующих — 0. Если эти числа обозначают вес ребер, то такое обозначение дает неверный результат, т. к. отсутствующие ребра рассматриваются как "бесплатный" путь между вершинами. Вместо этого отсутствую- щие ребра нужно инициализировать значением maxint. Таким образом мы можем и проверять наличие ребра, и автоматически игнорировать его при поиске кратчайшего пути, т. к. для этого будут рассматриваться только действительные ребра, при условии, что значение maxint меньше, чем диаметр графа. Кратчайший путь между двумя вершинами графа можно определить несколькими спо- собами. Алгоритм Флойда-Варшалла начинает работу с присвоения вершинам графа номеров от 1 до п. Эти номера используются не для маркировки вершин, а для их упо- рядочивания. Определим Иф/.у]* как длину кратчайшего пути от вершины /' к вершине у, в котором используются в качестве промежуточных вершин только вершины, прону- мерованные 1. 2..к. Что это означает9 Когда к = 0, то промежуточные вершины недопустимы, поэтому единственными разрешенными путями являются первоначальные ребра графа. Таким образом, матрица кратчайших путей для всех пар вершин состоит из первоначальной матрицы смежности. Будет выполнено п циклов, где в к-м цикле разрешается исполь- зовать только первые к вершин в качестве возможных промежуточных вершин на пути между каждой парой вершин х и у. В каждом последующем цикле добавляется одна новая возможная промежуточная вершина, что расширяет набор возможных кратчайших путей. Использование к-й вер- шины в качестве конечной помогает только в том случае, если существует кратчайший путь, который проходит через эту вершину, поэтому И7[/, /]* = min(Hz[i. у]*',»[/, Аг У + И7[£, /]* 1).
Глава 6. Алгоритмы для работы со взвешенными графами 233 Правильность этого выражения не совсем очевидна с первого взгляда, поэтому я реко- мендую хорошо разобраться в нем, чтобы убедиться в этом. Но. как видно из листин- ге 6.12. реализация алгоритма достаточно проста. Листинг 6 12. Реализация алгоритма Флойда-Варшалла floyd(adjacency_marri>: *g) int i,j; /* Счетчики измерений */ int k; /* Счетчик промежуточных вершин */ int through_k; /* Длина пути через вершину к */ for (к=11; k<=g->nvertices; к++) for ii=l; i<=g->nvertices; i++) for (j=l; j<=g->nvertices; j++) I through_k = g->weight[i][k]+g->weight[k] [>]; if (through_k < g->weight[i][j]) g->weight[i][j] = through_k; Время исполнения алгоритма Флойда-Варшалла равно О(и3), что ничем не лучше, чем время исполнения п вызовов алгоритма Дейкстры. Но циклы этого алгоритма такие сжатые, и сама программа такая короткая, что на практике он оказывается более эф- фективным. Алгоритм Флойда-Варшалла примечателен тем. что это один из немногих алгоритмов работы с графами, которые работают лучше на матрицах смежности, чем на списках смежности. По результату работы алгоритма Флойда-Варшалла в его текущей форме нельзя вос- создать фактический кратчайший путь между любой парой вершин. Но эти пути можно получить, если мы сохраним матрицу родителей Р для нашего выбора последней про- межуточной вершины, используемую для каждой пары вершин (х.у). Пусть это будет вершина к. Кратчайшим путем от вершины х к вершине у будет конкатенация крат- чайшего пути от вершины х к вершине к и кратчайшего пути от вершины к к вершине у. которую можно воссоздать рекурсивным способом по матрице Р. Но обратите вни- мание. что в большинстве задач поиска кратчайшего пути между всеми парами точек требуется только матрица расстояний. Для таких приложений и предназначен алгоритм Флойда-Варшалла. 6.3.3. Транзитивное замыкание Алгоритм Флойда-Варшалла имеет еще одну важную область применения — вычисле- ние транзитивного замыкания (transitive closure). При анализе орграфа часто требуется знать, какие вершины достижимы из данного узла. В качестве примера рассмотрим граф шантажиста, в котором наличие ориентированно- го ребра (z.j) означает, что лицо / обладает достаточным компроматом на лицо j и мо- жет заставить его сделать все, что угодно. Для участия в специальном проекте вы хоти- те нанять одного из этих п людей.
234 Часть I. Практическая разработка алгоритмов Простым решением было бы нанять шантажиста, представленного вершиной с наи- большим весом, но лучшим выбором будет шантажист с компроматом на наибольшее число людей, т. е., вершина со связями с наибольшим количеством других вершин. У Стива может быть мощный компромат, но только на Майкла, но если Майкл имеет компромат на всех прочих, тогда именно он будез лучшим выбором для нашего проекта. Вершины, достижимые из любой другой вершины, можно найти посредством обхода в ширину или глубину. Но все варианты можно вычислить с помощью алгоритма крат- чайшего пути между всеми парами вершин. Если после исполнения алгоритма Флойда- Варшалла длина кратчайшего пути между вершиной i и вершиной j равняется maxint, то можно быть уверенным, что между этими вершинами прямого пути не существует. Любые две вершины, у которых вес ребра меньше maxint. достижимы, как и в смысле теории графов, так и в смысле описанного проекта. Транзитивное замыкание рассматривается более подробно в разделе 15 5. 6.4. История из жизни. Печатаем с помощью номеронабирателя В составе группы специалистов я был на экскурсии в компании Periphonics, которая в то время была лидером в области создания телефонных систем речевого взаимодейст- вия. Это более интеллектуальные системы, чем системы типа "Нажмите кнопку 1 для перехода в меню", "Нажмите кнопку 2, если вы не нажали 1", которые так раздражают нас в повседневной жизни. Гид произносил стандартный текст о продукции компании, когда один из членов группы спросил: "Почему бы вам не использовать для ввода дан- ных систему распознавания голоса? Это было бы намного проще, чем вводить текст с помощью номеронабирателя". Реакция гида была отработана. — Нашим клиентам предоставляется опция включения возможности распознавания голоса в наш продукт, но очень немногие из них делают это. Системы распознавания слитной речи для общего класса пользователей недоста- точно точны в большинстве приложений. Наши клиенты предпочитают создавать сис- темы на основе ввода текста с клавиатуры телефона. — Предпочитают вводить текст с помощью номеронабирателя? Не смешите меня! — раздался голос из задних рядов группы.— Я ненавижу такой способ ввода текста. Всегда, когда я звоню в свою маклерскую контору, чтобы узнать биржевой курс, какая- то машина велит мне ввести трехбуквенный код. Что еще хуже, чтобы указать, какую из трех букв, показанных на кнопке, ввести, нужно нажать две кнопки. Я нажимаю кнопку 2, а машина мне говорит: "Нажмите 1 для А, Нажмите 2 для В, Нажмите 3 для С". Это ужасно, если вас интересует мое мнение. — Возможно, не нужно нажимать две кнопки, чтобы ввести одну букву, — подклю- чился я к разговору. — Может быть, система сумеет вычислить требуемую букву по контексту. — Когда вводишь трехбуквенный биржевой код акции, контекста маловато. — Конечно, но если вводить предложения, то контекста будет предостаточно. Спорю, что мы смогли бы правильно восстановить текст, вводимый с панели телефона по одной кнопке на букву.
Гпава 6. Алгоритмы для работы со взвешенными графами 235 Сотрудник Periphonics бросил на меня равнодушный взгляд и продолжил экскурсию. Но мне эта идея запала в голову, и когда я вернулся в свой офис, я решил попытаться воплотить ее в жизнь. При вводе с номеронабирателя не все буквы можно ввести одинаковым способом. Бо- лее того, не все буквы можно даже ввести, т. к. на стандартном американском телефоне буквы Q и Z не обозначены. Поэтому мы условились, что буквы Q, Z и пробел будут на кнопке звездочки (*). Для распознания вводимого с номеронабирателя текста можно было бы воспользоваться частотой использования букв алфавита. Например, если на- жата кнопка 3, то существует большая вероятность, что имеется в виду буква Е, а не находящиеся на этой же кнопке буквы D или F. Таким образом, учитывая частоту ис- пользования каждой буквы в окне из трех букв (триграмме), можно было попытаться предсказать вводимый текст. Вот что получилось, когда я испытал этот метод вводом Геттисбергского послания: enurraore anc reten yeasr ain our ectherr arotght eosti on ugis aootinent a oey oation aoncdivee in liccsty ane eedicatee un uhc rrorosition uiat all oen are arectee e ual ony ye arc enichde in a irect aitil yar uestini yhethes uiat oatioo or aoy oation ro aoncdivee ane ro eedicatee aan loni eneure ye are oet on a irect aattlediele oe uiat yar ye iate aone un eedicate a rostion oe uiat eiele ar a einal restini rlace eor uiore yin icre iate uhdis lives uiat uhe oation ogght live it is aluniethes eittini ane rrores uiat ye rioulc en ugir att in a laries rcore ye aan oou eedicate ye aan oou aoorearatc ye aan oou ialloy ugis iroune the arate oen litini ane eeae yin rustgilec iere iate aoorearatec it ear aante our roor rowes un ade or ceuraat the yople yill little oote oor loni renences yiat ye ray iere att it aan oetes eosiet yiat uhfy eie iere it is eor ur uhe litini rathes un ae eedicatee iere un uhe undiniside yopl yhici uhfy yin entght icre iate uitir ear ro onaky aetancde it is rathes eor ur un ae iere eedicatee un uhc irect uarl rencinini adeore ur uiat eron uhere ioooree eeae ye uale inarearee eeuotion uo tiat aaure eor yhici uhfy iere iate uhe lari eull oearure oe eeuotioo tiat ye iere iggily rerolue uiat uhere eeae rial! oou iate eide io Как можно видеть, метод триграмм оказался довольно хорошим для шифрования тек- ста, но совсем непригодным для его воспроизведения. Причина этому очевидная — алгоритм не знает абсолютно ничего об английских словах. Если применить его совме- стно со словарем, то мы, возможно, получим какой-то результат. Но в словаре может быть несколько разных слов для одной и той же последовательности кнопок номеро- набирателя. В качестве крайнего примера последовательность 22737 может означать одиннадцать разных английских слов, включая cases, cares, cards, capes, caper и bases. В нашей следующей попытке для последовательностей кнопок с несколькими возмож- ными словами мы выводили буквы, в правильности которых не было сомнений, а ос- тальные угадывали методом триграмм. Наградой за это был следующий вывод: eourscore and seven yearr ain our eatherr brought forth on this continent azoey nation conceivee in liberty and dedicatee uo uhe proposition that all men are createe equal ony ye are engagee in azipeat civil yar uestioi whether that nation or aoy nation ro conceivee and ro dedicatee aan long endure ye are oet on azipeat battlefield oe that yar ye iate aone uo dedicate a rostion oe that field ar a final perthni place for those yin here iate their lives that uhe nation oight live it is altogether fittinizane proper that ye should en this aut in a larges sense ye aan oou dedicate ye aan oou consecrate ye aan oou hallow this ground the arate men litioi and deae yin strugglee here iate consecratee it ear above our roor power uo ade or detract the world will little oote oor long remember what ye ray here aut it aan meter forget what uhfy die here it is for ur uhe litioi rather uo ae dedicatee here uo uhe toeioisgee york which uhfy yin fought here iate thus ear ro mocky advancee it is rather for ur uo ae here dedicatee uo uhe great task renagogoi ad fore ur that from there honoree deae ye uale increasee devotion uo that aausc for which uhfy here iate uhe last eull measure oe
236 Часть I. Практическая разработка алгоритмов devotion that ye here highky resolve that there deae shall oou iate fide io vain that this nation under ioe shall late tizoey birth oe freedom and that ioternmenu oe uhe people ay uhe people for uhe people shall oou perish from uhe earth Кто-либо, изучающий американскую историю, возможно и распознал бы этот текст, но для всех других он был бессмысленным. Нам нужно было найти какой-то способ раз- личать слова, набираемые одинаковой последовательностью кнопок. Можно было бы попробовать учитывать относительную частоту использования каждого слова, но здесь также было бы слишком много ошибок. На этом этапе я подключил к проекту Гаральда Pay (Harald Rau), который оказался прекрасным помощником. Прежде всего, он был смышленый и упорный аспирант. Кроме этого, т. к. его родным языком был немецкий, он верил всему, что я говорил ему об английской грамматике. Гаральд создал алгоритм реконструкции слов по нажатым клавишам (рис. 6.9). ВХОД I Распознавание лексем ...#4483*6 3 *2*7 16 1 # ... Лексема Лексема Лексема Лексема Выбор кандидатов Построение осмысленного предложения ВЫХОД GIVE ME A RING. Рис. 6.9. Этапы восстановления слов из последовательностей нажатых кнопок номеронабирателя Программа обрабатывала предложение, идентифицируя все слова, совпадающие с каждой последовательностью введенного кода. Принципиальную трудность пред- ставляло встраивание в нее грамматических условий. — Хорошая информация о грамматике и о частоте использования слов имеется в большой базе данных, называющейся Brown Corpus. Она содержит тысячи типичных
Глава 6. Алгоритмы для работы со взвешенными графами 237 английских предложений, проанализированных по частям речи. Но как мы можем учесть всю эту информацию в нашей программе? — спросил Гаральд. — Давай будем рассматривать это, как задачу на графах, — предложил я. — Задачу на графах? Где же здесь графы? — Предложение можно рассматривать как последовательность лексем, каждая из ко- торых представляет слово в предложении. Для каждой лексемы имеется соответст- вующий список слов, из которого нужно выбрать правильное. Как это сделать? Каждое возможное предложение, т. е. комбинацию слов, можно рассматривать как путь в гра- фе, вершины которого— полный набор возможных слов. Каждое возможное /-е слово будет соединено ребром с каждым возможным (/ + 1)-м словом. Самым дешевым путем по этому графу будет наилучшая интерпретация предложения. — Но все пути выглядят одинаково; они имеют одинаковое количество ребер. Погоди, теперь я вижу! Чтобы сделать пути разными, нам нужно присвоить ребрам вес. — Совершенно верно! Стоимость ребра будет отражать вероятность нашего выбора соединяемой им пары слов. Эту стоимость можно определять по тому, как часто дан- ная пара слов встречается в базе данных. Или вес можно присваивать с учетом частей речи. Возможно, существительные реже соседствуют с другими существительными, чем с глаголами. — Отслеживать статистику пар слов будет трудно, поскольку их так много! Но мы зна- ем частоту использования каждого слова. Как мы можем учесть этот фактор в нашей программе? — Можно сделать так, что цена за проход через определенную вершину будет зависеть от частоты употребления данного слова. В таком случае самый короткий путь через граф будет самым лучшим предложением. — Но как нам определить относительную важность каждого из этих факторов? — Сначала попробуй реализовать то, что тебе кажется естественным, а потом можно экспериментировать. Алгоритм поиска кратчайшего пути, разработанный Гаральдом по этим принципам, показан на рис. 6.10. Рис. 6.10. Путь с наименьшей стоимостью представляет самую лучшую интерпретацию предложения
238 Часть I. Практическая разработка алгоритмов Программа со встроенными грамматическими и статистическими условиями работала прекрасно. Оцените ее воспроизведение ввода Геттисбергского послания с номерона- бирателя телефона теперь: FOURSCORE AND SEVEN YEARS AGO OUR FATHERS BROUGHT FORTH ON THIS CONTINENT A NEW NATION CONCEIVED IN LIBERTY AND DEDICATED TO THE PROPOSITION THAT ALL MEN ARE CREATED EQUAL. NOW WE ARE ENGAGED IN A GREAT CIVIL WAR TESTING WHETHER THAT NATION OR ANY NATION SO CONCEIVED AND SO DEDICATED CAN LONG ENDURE. WF ARE MET ON A GREAT BATTLEFIEl D OF THAT WAS. WE HAVE COME TO DEDICATE A PORTION OF THAT FIELD AS A FINAL SERVING PLACE FOR THOSE WHO HERE HAVE THEIR LIVES THAT THE NATION MIGHT I IVE. IT IS ALTOGE THER FITTING AND PROPER THAT WE SHOULD DO THIS BUT IN A LARGER SENSE WE CAN NOT DEDICATE WE CAN NOT CONSECRATE WE CAN NOT HAL LOW THIS GROUND. THE BRAVE MEN LIVING AND DEAD WHO STRUGGLED HERE HAVE CONSECRA TED IT FAR ABOVE OUR POOR POWER TO ADD OR DETRACT. THE WORLD WILL LI I l l E NOTE NOR LONG REMEMBER WHAT WE SAY HERE BUT IT CAN NEVER FORGET WHAT THEY DID HERE. IT IS FOR US THE LIVING RATHER TO BE DEDICATED HERE TO THE UNFINISHED WORK WHICH THEY WHO FOUGHT HERE HAVE THUS FAR SO NOBLY ADVANCED. IT IS RATHER FOR US TO BE HERE DEDICATED TO THE GREAT TASK REMAINING BEFORE US TIIAT FROM THESE I IGNORED DEAD WE TAKE INCREASED DEVOTION TO THAT CAUSE FOR WHICH THEY HERE HAVE THE LAST FULL MEASURE OF DEVOTION THAI WE HERE HIGHLY RESOLVE THAT THESE DEAD SHALL NOT HAVE DIED IN VAIN THAT THIS NATION UNDER GOD SHALL HAVE A NEW BIRTH OF FREEDOM AND THAT GOVERNMENT OF THE PEOPLE BY THE PEOPLE FOR THE PEOPL E SHALL NOT PERISH FROM THE EARTH. В то время как в выводе имеется несколько ошибок, результат, несомненно, является достаточно хорошим для многих приложений. В компании Periphonics определенно были такого мнения, т. к. они лицензировали нашу программу для использования в своих продуктах. В табл. 6.1 можно видеть, что мы смогли правильно воссоздать свы- ше 99% символов из почти мегабайта, занимаемого речами президента Клинтона. Так что, если бы он отправлял их в виде SMS-сообщений, вводя текст с номеронабирателя, то мы определенно были бы в состоянии понять их. Таблица 6.1. Реконструкция нескольких текстов, вводимых с помощью кнопочного номеронабирателя телефона Текст Количество символов Количество правильных символов Количество правильных непробельпых символов Количество правильных слов Время обработки символа Речи Клинтона 1 073 593 99.04% 98.86% 97.67% 0.97 мс Роман "Herland" 278 670 98,24% 97.89% 97,02% 0.97 мс Роман "Моби Дик" 1 123 581 96.85% 96.25% 94.75% 1.14 мс Библия 3 961 684 96,20% 95,39% 95,39% 1.33 мс Пьесы Шекспира 4 558 202 95.20% 94.21% 92.86% 0.99 мс Скорость воссоздания символов достаточно высокая, более того, быстрее, чем их ввод с клавиатуры номеронабирателя. Условия для многих задач распознавания закономерностей можно сформулировать естественным образом в виде задач поиска кратчайшего пути в графе. Для решения
Гпава 6. Алгоритмы для работы со взвешенными графами 239 этих задач хорошо подходит алгоритм Витерби, который широко применяется в систе- мах распознавания речи и рукописи. Алгоритм Витерби, по сути, решает задачу поиска кратчайшего пути в бесконтурном орграфе. Таким образом, часто бывает полезно по- пытаться сформулировать поставленную задачу в терминах теории графов. 6.5. Потоки в сетях и паросочетание в двудольных графах Графы со взвешенными ребрами можно рассматривать как трубопроводную сеть, в которой вес ребра Я,/) определяет пропускную способность каждого отрезка трубо- провода. В свою очередь, пропускную способность можно рассматривать как функцию поперечного сечения трубы. Широкая труба может пропускать 10 единиц потока за определенное время, в то время как более узкая труба— только 5 единиц за то же са- мое время. В задаче о потоках в сетях требуется определить максимальный объем по- тока, который можно пропустить между вершинами s и / взвешенного графа G, соблю- дая ограничения на максимальную пропускную способность каждого ребра-трубы. 6.5.1. Паросочетание в двудольном графе В то время как задача о потоках в сетях представляет самостоятельный интерес, ее ос- новная ценность состоит в решении других важных задач на графах. Классическим примером является паросочетание в двудольном графе. Паросочетанием графа G = (Г\ Е) называется такое подмножество ребер Е' сЕ, в котором никакие два ребра не имеют общую вершину. При этом вершины группируются попарно таким образом, что каждая вершина принадлежит не более, чем одной такой паре. На рис. 6.11 показа- ны двудольный граф с максимальным паросочетанием (выделено жирными линиями) и соответствующий экземпляр потоков в сети с максимальным потоком s-t. Рис. 6.11. Двудольный граф с максимальным паросочетанием (а) и экземпляр потоков в сети (б) Граф G называется двудольным, если его вершины можно разделить на два множества, L и R, таким образом, что все ребра графа имеют одну вершину в множестве L, а дру- гую— в множестве R. Многие естественные графы являются двудольными. Например, некоторые вершины могут представлять задания, требующие исполнения, а остальные
240 Часть I. Практическая разработка алгоритмов вершины — людей, которые могут выполнить эти задания. Наличие ребра (/,р) означа- ет, что задание j может быть выполнено человеком р. Или же часть вершин может представлять юношей, а другая девушек, а ребра— совместимые гетеросексуальные пары. Паросочетания в этих графах подаются естественному толкованию как рабочее задание или женитьба. Такие паросочетания подробно обсуждаются в разделе 15.6. Максимальное парообразование в двудольном графе можно с легкостью найти, ис- пользуя потоки в сети. Для этого создаем вершину-дс/лок .v. которая соединена со все- ми другими вершинами в подмножестве L взвешенными ребрами, каждое весом 1. По- том создаем вершину-сдгок t, которая соединена со всеми другими вершинами в под- множестве R взвешенными ребрами, каждое весом 1. Наконец, присваиваем каждому ребру двудольного графа G вес 1. Теперь максимальный возможный поток из вершины s в вершину / определяет максимальное паросочетание в графе G. Всегда можно найти поток такого же объема, как и паросочетание, используя только ребра паросочетания и соединенные ими истоки и стоки. Кроме этого, поток большего объема невозможен. Ведь не можем же мы пропускать через какую-либо вершину больше, чем одну едини- цу потока. 6.5.2. Вычисление потоков в сети Традиционные алгоритмы работы с потоками в сети основаны на идее увеличивающих путей, которая состоит в многократном повторении процесса поиска пути с положи- тельной пропускной способностью от вершины $ к вершине t и добавления его к пото- ку. Можно доказать, что поток в сети является оптимальным тогда и только тогда, ко- гда в ней не имеется увеличивающего пути. Так как каждое добавление пути увеличи- вает поток, то в итоге должен быть найден глобальный максимум. Ключевой структурой является граф остаточного потока (residual flow graph) R(G. f'). где G — граф ввода, a f—текущий поток через G. Ориентированный взвешенный граф R(G.f) содержит те же самые вершины, что и граф G. Для каждого ребра (z, j) в графе G, которое обладает пропускной способностью и имеет поток Jli.J). граф R(G.f) может содержать следующие два ребра: ♦ ребро (z, j) с весом c(i,j) -fli.j). если c(i,j) -fli.j) > 0; ♦ ребро (j. i) с весом fii.j'), > 0. Наличие ребра (/,/) в графе остаточного потока указывает на то, что от вершины z к вершине / можно направить положительный поток, точный объем которого опреде- ляется весом ребра. Путь в графе остаточного потока от вершины .<> к вершине t подра- зумевает, что от первой вершины ко второй можно пропустить дополнительный поток, объем которого определяется минимальным весом ребра на этом пути. Для иллюстра- ции этой идеи на рис. 6.12 изображены максимальный поток x-t в графе G и связный граф остаточного потока R(G). Минимальный разрез потока s-t обозначен пунктирной линией возле вершины t. Максимальный поток s-t в графе G равен 7. Этот поток определяется двумя путями в графе остаточного потока R(G). направленными от вершины t к вершине 5 и пропуск- ной способностью 2+5. Эти потоки полностью используют пропускную способность двух ребер, входящих в вершину t. вследствие чего не остается путей, способных что-
Глава 6. Алгоритмы для работы со взвешенными графами 241 либо добавить. Таким образом, поток является оптимальным. Множество ребер, удале- ние которых отделяет вершину s от вершины t (как два ребра, входящие в вершину t на рис. 6.12), называется (з—1)-разрезом (cut). Очевидно, что никакой поток из вершины s в вершину / не может превзойти вес такого минимального разреза. С другой стороны, всегда возможен поток, равный минимальному разрезу. Рис. 6.12. Максимальный поток s-t в графе С и связный граф остаточного потока R(G) Подведение итогов Максимальный поток из s в t всегда равен минимальному весу (л- /)-разреза. Таким обра- зом, алгоритмы для работы с потоками можно применять для решения общих задач связ- ности ребер и вершин графов. Реализация В этой книге мы не можем представить теорию потоков сети в полном объеме. Но мы сможем, по крайней мере, показать, как определять увеличивающие пути и вычислять оптимальный поток. Для каждого ребра в графе остаточного потока необходимо от- слеживать как текущий объем потока, проходящего через него, так и его остаточную пропускную способность. Соответственно, необходимо модифицировать структуру ребра, чтобы учесть дополнительные поля (листинг 6.13). Листинг 6.13. Модифицированная структура реора typedef struct { int v; int capacity; int flow; int residual; struct edgenode ‘next; > edgenode; /* Соседняя вершина */ /* Пропускная способность ребра */ /* Поток через ребро */ /» Остаточная пропускная способность ребра */ /* следующее ребро в списке */ Используя обход в ширину, мы выполняем поиск путей от истока к стоку, которые увеличивают общий поток, и добавляем обнаруженные пути, увеличивая общий поток. Когда все увеличивающие пути обнаружены и добавлены в поток, процедура заверша- ется. возвращая оптимальный поток (листинг 6.14).
242 Часть I. Практическая разработка алгоритмов Листинг 6.14. Процедура поиска оптимального потока netflow(flow_graph *д, int source, int sink) { int volume; /* Вес увеличивающего пути*/ add_residual_edges(g); initialize search(g); bis (g, si urcei ; volume path_volume(g, source, sink, parent); while (volume >0) [ augment_path(g, source, sink, parent, volume); initialize_search(g); bfs(g,source); volume = path_volume(g, source, sink, parent); ) } Увеличивающий путь от истока к стоку повышает объем потока; такой путь можно найти посредством обхода в ширину соответствующего графа. Рассматриваются толь- ко такие ребра, которые имеют остаточную пропускную способность, или, иными сло- вами, положительный остаточный поток. При обходе в ширину насыщенные и нена- сыщенные ребра различаются с помощью процедуры, возвращающей булево значение (листинге. 15). Листинг 6.15. Процедура для различения насыщенных и ненасыщенных ребер bool valid_edge(edgenode *е) if (e-^>residual > 0) return (TRUE) ; eIse return(FALSE); J При увеличении пути максимально возможный объем потока перемещается из ребер с остаточной пропускной способностью в положительный поток. Этот объем ограни- чен ребром с наименьшей пропускной способностью, точно так же, как и объем про- пускаемой по водопроводу воды ограничен наиболее узкой трубой. Соответствующий код показан в листинге 6.16. Листинг 6.16. Добавление увеличивающих путей в поток int path_volume(flow_graph *g, int start, int end, int parents!]) [ edgenode *e; /* Рассматриваемое ребро */ edgenode *find_edge(); if (parents[end] == -1) return(O); e = find_edge(g,parents[end],end); if (start == parents[end]) return(e->residual); else return( min(path_volume(g,start,parents[end],parents), e->residual) );
Гпава 6 Алгоритмы для работы со взвешенными графами 243 edgenode *f ind_edge (1 lowgraph *g, int x, int y) edgenode ’p; /• Временный указатель V p = g->edges[xj; while (p != NULL) if (p->v == y) return(p); p = p->next; return(NULL); Направление дополнительного объема потока по ориентированному ребру (/,/) умень- шает остаточную пропускную способность этого ребра, но увеличивает пропускную способность ребра (/, /). Таким образом, действие увеличения пути требует модифици- рования как прямых, так и обратных ребер для каждого звена пути. Соответствующий код показан в листинге 6.17. Листинг 6.17. Модификация ребер augment_path(flow graph *g, int start, int end, int parents[], int volume) edgenode *e; /* Рассматриваемое peCpo */ edgenode *find_edge(); if (start == end) return; e = find_edge(g, parents[end],end); e->flow +- volume; e->residual -= volume; e = find_edge(g, end, parents [end] ) ; e->residual += volume; augment_path(g, start, parents[end], parents, volume); Для инициализации графа потока требуется создать ориентированные ребра (/,/) и (/, z) для каждого ребра сети е = Все начальные потоки инициализируются в 0. Перво- начальный остаточный поток ребра (/,/) устанавливается равным пропускной способ- ности ребра сети <?, а первоначальный остаточный поток ребра (/', z) устанавливается равным нулю. Анализ Алгоритм поиска увеличивающих путей в конечном счете приходит к оптимальному решению. Но каждый новый увеличивающий путь может добавлять только незначи- тельный объем к общему потоку, поэтому, теоретически, время схождения алгоритма может быть сколь угодно большим. Но Эдмондс (Edmonds) и Карп (Karp) в книге [ЕК72] доказали, что постоянно выбирая кратчайший невзвешенный увеличивающий путь, можно гарантировать, что добавле- ние О(«) увеличивающих путей будет достаточным для получения оптимального по- тока. В действительности, наша реализация поиска оптимального потока и является
244 Часть I. Практическая разработка алгоритмов алгоритмом Эдмондса-Карпа, т. к. для поиска следующего увеличивающего пути ис- пользуется обход в ширину, начинающийся с истока. 6.6. Разрабатывайте не алгоритмы, а графы Правильное моделирование является ключом к эффективному использованию алго- ритмов для работы с графами. Мы определили несколько свойств графов и разработа- ли алгоритмы для их вычисления. В каталоге задач представлено около двух десятков разных задач по графам. Эти классические задачи по графам предоставляют основу для моделирования большинства приложений. Но секрет успешного решения задач на графах заключается в умении разрабатывать не алгоритмы для работы с графами, а сами графы. Мы уже видели несколько примеров этой идеи. В частности: ♦ максимальное остовное дерево можно найти, изменив веса ребер исходного графа G на отрицательные и применив на получившемся графе С алгоритм для построения минимального остовного дерева. Остовное дерево графа С с максимальным по мо- дулю общим весом будет максимальным остовным деревом графа G; ♦ для решения задачи паросочетания в двудольном графе мы создали специальный граф потоков в сети, в котором максимальный поток соответствует паросочетанию максимальной мощности. Рассматриваемые далее примеры дополнительно демонстрируют мощь правильного моделирования. Каждый из этих примеров возник в реальном приложении и каждый можно смоделировать в виде задачи на графах. Некоторые примеры довольно сложны, но они иллюстрируют универсальность графов в представлении взаимоотношений. Прочитав задачу, не спешите заглядывать на страницу с решением, а попытайтесь соз- дать для нее соответствующее графовое представление. Остановка для размышлений. Нить Ариадны ЗАДАЧА. Требуется алгоритм для разработки естественных маршрутов, по которым персонажи видеоигр могут проходить через помещения, наполненные разными препят- ствиями. Решение. Предположительно, искомый маршрут должен соответствовать маршруту, который бы выбрало разумное существо. А поскольку разумные существа склонны к лени и оптимальным действиям, они выберут самый короткий маршрут. Соответствен- но, данную задачу нужно моделировать как задачу поиска кратчайшего пути. Но что у нас будет служить графом? Один из подходов может заключаться в наложе- нии решетки на комнату. Для каждого узла решетки, свободного от предметов, создаем вершину. Между любыми двумя близлежащими вершинами обязательно найдется реб- ро, взвешенное пропорционально расстоянию между ними. Хотя для поиска кратчай- шего пути существуют прямые геометрические методы (см. раздел 15.4), данную зада- чу легче смоделировать дискретно в виде графа.
Гпава 6. Алгоритмы для работы со взвешенными графами 245 Остановка для размышлений. Упорядочивание последовательности ЗАДАЧА. Проект секвенирования ДНК имеет на выходе экспериментальные данные, состоящие из небольших фрагментов. Для каждого фрагмента f известно, что некото- рые фрагменты обязательно расположены слева от него, а другие — справа. Нужно найти непротиворечивое упорядочивание фрагментов слева направо, которое удовле- творяет всем требованиям. Решение. Создаем ориентированный граф, в котором каждый фрагмент представлен вершиной. Вставляем ориентированное ребро (l.f) от любого фрагмента I, который должен быть слева от фрагмента f и ориентированное ребро (/,г) от любого фраг- мента г, который должен быть справа от фрагмента f Нам нужно упорядочить верши- ны таким образом, чтобы все ребра были направлены слева направо. Это будет топо- логическая сортировка получившегося бесконтурного орграфа. Граф должен быть бес- контурным, т. к. при наличии контуров (т. е. замкнутых маршрутов) непротиворечивое упорядочивание невозможно. Остановка для размышлений. Размещение прямоугольников по корзинам ЗАДАЧА. Произвольное множество прямоугольников в плоскости нужно распреде- лить по минимальному количеству корзин таким образом, чтобы ни одно из подмно- жеств прямоугольников в любой корзине не пересекалось с другим. Иными словами, водной и той же корзине не может быть двух перекрывающихся прямоугольников. Решение. Создаем граф, в котором каждая вершина представляет прямоугольник, а перекрывающиеся прямоугольники соединяются ребром. Каждая корзина соответству- ет независимому множеству прямоугольников, поэтому ни один из них не накладыва- ется на другой. Раскраской вершин графа называется разбиение вершин на независи- мые множества, поэтому для нашей задачи требуется минимизировать количество-цве- тов. Остановка для размышлений. Конфликт имен файлов ЗАДАЧА. При переносе кода из операционной системы UNIX в DOS нужно сократить имена нескольких сотен файлов до, самое большее, 8 символов. Просто использовать первые восемь символов имени файла нельзя, т. к. имяфайла! и имя_файла2 будут преобразованы в одно и то же имя. Как рационально сократить имена файлов и при этом избежать конфликтов между получившимися именами? Решение. Создайте двудольный граф, в котором вершины соответствуют каждому первоначальному имени файла f для 1 < i < и, а также набор приемлемых сокращений для каждого имени f\, Соедините ребром каждое первоначальное имя и его со- кращенный вариант. Теперь нужно найти набор из п ребер, не имеющих общих вер- шин, соотнеся, таким образом, первоначальное имя файла с приемлемым индивидуали- зированным сокращением. Задача паросочетаний в двудольном графе, рассматривае- мая в разделе 15.6. как раз является типом задачи поиска независимого множества ребер в графе.
246 Часть I. Практическая разработка алгоритмов Остановка для размышлений. Разделение текста ЗАДАЧА. Найти способ для разделения строчек текста в разрабатываемой системе распознавания текста. Хотя между печатными строчками текста имеется свободный промежуток, но по разным причинам, таким как помехи или перекос страницы, этот интервал трудно выделить. Каким образом можно решить это задачу? Решение. Примем следующее определение графа. Каждый пиксел изображения пред- ставляется вершиной графа, а соседние пикселы соединяются ребром. Вес этого ребра должен быть пропорционален степени черного цвета в пикселах. В этом графе интер- вал между двумя печатными строчками будет путем, направленным от левого края страницы к правому. Нам нужно найти относительно прямой путь, максимально избе- гающий черных точек. Предполагается, что кратчайший путь в графе пикселов будет с большой вероятностью правильным разделителем. Подведение итогов Разработать действительно новый алгоритм работы с графами очень нелегко, поэтому не тратьте на это время. Вместо этого для моделирования стоящей перед вами задачи ста- райтесь разработать графы, которые позволяют применить классические алгоритмы. Замечания к главе Представление задачи в форме потоков в сети является эффективным алгоритмиче- ским инструментом, но для того, чтобы понять, можно ли решить данную задачу с по- мощью этого инструмента, требуется опыт. Для более подробного изучения данной темы рекомендуется ознакомиться с книгами [СС97] и [АМО93]. Метод увеличивающих путей описан Фордом (Ford) и Фулкерсоном (Fulkerson) в книге [FF62]. Эдмондс (Edmonds) и Карп (Karp) в книге [ЕК72] доказали, что постоянно вы- бирая кратчайший невзвешенный увеличивающий путь, можно гарантировать, что до- бавление О(п ) увеличивающих путей будет достаточным для получения оптимального потока. Система распознавания текста, вводимого с помощью кнопочного номеронабирателя телефона, которая рассматривалась ранее в этой главе, описывается более подробно в книге [RS96], 6.7. Упражнения Алгоритмы для эмуляции графов 1. [3] Для графов из задачи 1 главы 5: а) Нарисуйте остовный лес, получаемый после каждой итерации основного цикла алго- ритма Крускала. б) Нарисуйте остовный лес, получаемый после каждой итерации основного цикла алго- ритма Прима. в) Найдите остовное дерево с кратчайшим путем и с корнем в вершине Я. г) Вычислите максимальный поток от вершины А к вершине Н.
Гпава 6. Алгоритмы для работы со взвешенными графами 247 Минимальные остовные деревья 2. [3] Будет ли путь между двумя вершинами в минимальном остовном дереве обязательно самим коротким путем между этими двумя вершинами в полном графе? Если да. пре- доставьте доказательство; если нет. приведите контрпример. 3. [3] Допустим, что все ребра графа имеют разный вес, т. е. нет ни одной пары ребер с одинаковым весом. Будет ли путь между двумя вершинами в минимальном остовном дереве обязательно самым коротким путем между этими двумя вершинами в полном графе? Если да. предоставьте доказательство; если нет, приведите контрпример. 4. [3] Могут ли алгоритмы Прима и Крускала выдавать разные минимальные остовные де- ревья? Аргументируйте свой ответ. 5 [3] Могут ли алгоритмы Прима и Крускала работать с графами, содержащими ребра с отрицательным весом? Аргументируйте свой ответ. 6. [5] Дано минимальное остовное дерево Т графа G (содержащего п вершин и т ребер). К этому графу мы добавим новое ребро е = (и, г) весом и’. Разработайте эффективный алгоритм для построения минимального остовного дерева графа G + е. Чтобы решение было засчитано полностью, алгоритм должен иметь время исполнения О(н). 7. [5] а) Дано минимальное остовное дерево Т взвешенного графа G. Создайте новый граф С', увеличив вес каждого ребра графа G на к. Создают ли ребра минимального остовно- го дерева Т графа G минимальное остовное дерево графа G? Если да, предоставьте до- казательство; если нет, приведите контрпример. б) Пусть Р = {5,..., /} описывает кратчайший взвешенный путь между вершинами и t взвешенного графа G. Создайте новый граф G', увеличив вес каждого ребра графа G на к. Описывает ли Р кратчайший путь от вершины s к вершине / в графе &? Если да, пре- доставьте доказательство; если нет, приведите контрпример. 8. [5] Разработайте и выполните анализ алгоритма, который во взвешенном графе G нахо- дит наименьшее изменение в стоимости ребра, не являющегося ребром минимального остовного дерева, вследствие которого изменится минимальное остовное дерево графа. Алгоритм должен быть корректным и иметь полиноминальное время исполнения. 9. [4] Рассмотрим задачу поиска взвешенного связного подмножества ребер Т с мини- мальным весом во взвешенном связном графе G. Вес Тсостоит из суммы весов всех его ребер. а) Почему эта задача не является задачей поиска минимального остовного дерева? Под- сказка: вспомни те о ребрах с отрицательным весом. б) Разработайте эффективный алгоритм поиска связного подмножества Т с минималь- ным весом. 10. [4] Дан неориентированный граф G = (К, £). Множество ребер Fc Е называется разры- вающим множеством ребер (feedback-edge set), если каждый контур в графе G имеет, по крайней мере, одно ребро в F. а) Дан невзвешенный граф G. Разработайте эффективный алгоритм поиска разрываю- щего множества ребер минимального размера. б) Дан взвешенный неориентированный граф G с ребрами положительного веса. Разра- ботайте эффективный алгоритм поиска разрывающего множества ребер минимального веса.
248 Часть I. Практическая разработка алгоритмов 11. [5] Модифицируйте алгоритм Прима, чтобы он исполнялся за время O(»log£) для графа, у которого есть только к разных весов ребер. Поиск-объединение 12. [5] Разработайте эффективную структуру данных для выполнения таких операций на взвешенных орграфах: а) слияние двух указанных компонентов: б) поиск компонента, содержащего указанную вершину v; в) возвращение минимального ребра из указанного компонента. 13. [5] Разработайте структуру данных, которая может поддерживать выполнение последо- вательности из in операций объединение и поиск на универсальном множестве из п эле- ментов, такой, что вначале выполняются все операции объединения, а за ними - все операции поиска. Последовательность операций должна выполняться за время О(т + п). Поиск кратчайшего пути 14. [3] В задаче о кратчайшем пути в пункт назначения (single-destination shortest path) требуется найти кратчайший путь из каждой вершины графа в указанную вершину. Раз- работайте эффективный алгоритм для решения этой задачи. 15. [3] Дано: неориентированный взвешенный граф G = (Г, £) и его остовное дерево Т с кратчайшим путем и с корнем в вершине v’. Если увеличить вес всех ребер графа G на к, останется ли Т кратчайшим остовным деревом с корнем в вершине v? 16. [3] а) Приведите пример взвешенного связного графа G = (И, £) и вершины г, для кото- рых минимальное остовное дерево и кратчайшее остовное дерево с корнем в вершине v являются одинаковыми. б) Приведите пример взвешенного связного ориентированного графа G= (£, £) и вер- шины v, для которых минимальное остовное дерево существенно отличается от мини- мального остовного дерева с корнем в вершине г. в) Могут ли эти два типа остовных деревьев быть полностью непересекающимися? 17. [3] Докажите, что следующие утверждения верны, в противном случае приведите соот- ветствующий контрпример. а) Будет ли путь между двумя вершинами в минимальном остовном дереве неориенти- рованного графа кратчайшим (имеющим минимальный вес) путем? б) Допустим, что граф имеет единственное минимальное остовное дерево. Будет ли путь между двумя вершинами в минимальном остовном дереве неориентированного графа кратчайшим (имеющим минимальный вес)? 18. [5] В некоторых задачах на графах вес может присваиваться не ребрам (или не только ребрам), а вершинам. Пусть С,, означает вес вершины г, а C(x,v)— вес ребра (х, у). Требуется найти самый дешевый путь между вершинами а и h графа G. Стоимость пути определяется как сумма весов ребер и вершин, через которые проходит путь. а) Все ребра графа имеют нулевой вес, а отсутствующие ребра обозначаются бесконеч- ным весом (со). Вес каждой вершины С,. = 1 (1 < v< п). Разработайте зффективный ал-
Глава 6. Алгоритмы для работы со взвешенными графами 249 горитм поиска самого дешевого пути от вершины а к вершине Ь. Укажите временную сложность этого алгоритма. б) Вершины имеют разный вес (но обязательно положительный), а вес ребер остается прежним. Разработайте эффективный алгоритм поиска самого дешевого пути от верши- ны а к вершине Ь. Укажите временную сложность этого алгоритма. в) Вершины и ребра имеют разный вес (но обязательно положительный). Разработайте эффективный алгоритм поиска самого дешевою пути от вершины а к вершине Ь. Ука- жите временную сложность этого алгоритма. 19. [5] Дан ориентированный взвешенный граф Gen вершинами и т ребрами, в котором все вершины имеют положительный вес. Контуром (directed cycle) называется ориен- тированный путь, который начинается и заканчивается в одной и той же вершине и со- держит, по крайней мере, одно ребро. Предоставьте алгоритм с временем исполнения О(д’) для поиска контура в графе с минимальным общим весом. Чтобы решение было зачтено частично, время исполнения алгоритма может быть равным ()(пт). 20. [5] Можно ли модифицировать алгоритм Дейкстры для решения задачи поиска самого длинного пути из заданного пункта выхода (single-source longest path), изменив minimum на maximum? Если да, предоставьте доказательство: если нет. приведите контрпример. 21. [5] Дан взвешенный бесконтурный орграф G = (Г, £), в котором, возможно, имеются ребра с отрицательным весом. Разработайте алгоритм с линейным временем исполне- ния для решения задачи поиска кратчайшего пути из заданной начальной вершины v. 22. [5] Дан взвешенный ориентированный граф G = (Г, £), в котором все веса положитель- ные. Пусть з- и w — вершины графа G, к — целое число (к < |Р|). Разработайте алгоритм поиска кратчайшего пути от вершины v к вершине м>, содержащего ровно к ребер. Путь не обязательно должен быть простым. 23. [5] Арбитражем называется использование разницы в курсах обмена валют для получения прибыли. Например, в течение короткого периода времени за 1 доллар США можно купить 0,75 фунта стерлингов, за 1 фунт стерлингов— 2 австра- лийских доллара, а за 1 австралийский доллар— 0,50 доллара США. То есть осу- ществление такой сделки принесет прибыль, равную 0.75x2x0.7= 1.05 доллара США или 5%. Имеется п валют щ, ..., с„ и таблица курсов валют R размером п х и. в которой указывается, что за одну единицу валюты с, можно купить R[i,f] единиц валюты с,. Разработайте и выполните анализ алгоритма для определения макси- мального значения с„] • с,2] • 7?[с,д-ь с,*] - R[clt, с,] Подсказка: ищите кратчайший путь между всеми парами вершин. Потоки в сети и паросочетание 24. [3] Паросочетанием в графе называется набор непересекающихся ребер, т. е. ребер, ко- торые не имеют общих вершин. Разработайте алгоритм поиска максимального паросо- четания в дереве. 25. [5] Реперным покрытием (edge cover) неориентированного графа G = (Р, Е) называется набор ребер, для которого в каждую вершину графа входит, по крайней мере, одно реб- ро из этого набора. Разработайте эффективный алгоритм на основе паросочетаний для поиска реберного покрытия максимального размера графа G.
250 Часть I. Практическая разработка алгоритмов Задачи по программированию Эти задачи доступны на сайтах http://www.programming-challenges.com и http:// uva.onlinejudge.org. 1. Freckles. 111001/10034. 2. Necklace. 111002/10054. 3. Railroads. 111004/10039. 4. Tourist Guide. 111006/10199. 5. The Grand Dinner. 111007/10249.
ГЛАВА 7 Комбинаторный поиск и эвристические методы Для многих задач можно найти решение с помощью методов исчерпывающего перебо- ра, хотя временная сложность таких методов может оказаться астрономической. А в других ситуациях время, потраченное на поиск оптимального решения, обязатель- но окупится в дальнейшем. Правильность работы электрической схемы можно дока- зать, проверив все возможные входные контакты и удостоверившись в правильности выходного сигнала. Но такая проверка не всегда стоит затраченных на это усилий. Частота тактового генератора современных компьютеров достигает нескольких гига- герц. что означает, что они могут исполнять несколько миллиардов базовых инструк- ций в секунду. Так как осуществление какой-либо представляющей интерес операции более высокого уровня требует нескольких сот базовых инструкций, то можно ожи- дать, что за секунду мы можем просмотреть несколько миллионов элементов. Но важно иметь представление, насколько велик один миллион. Один миллион пере- становок означает все возможные упорядочивания приблизительно 10 или 11 объектов, но не больше. Один миллион подмножеств означает все возможные комбинации около 20 элементов, но не более. Решение задач существенно большего размера требует тща- тельного сокращения пространства поиска, чтобы обработке подвергались только дей- ствительно имеющие важность элементы. В этой главе представляется метод перебора с возвратом, применяющийся для пере- числения всех возможных решений комбинаторной алгоритмической задачи. Также приводится иллюстрация хитроумных методов отсечения тупиковых решений для ус- корения работы реальных поисковых приложений. Кроме этого, для решения задач, слишком больших для решения методом полного перебора всех комбинаций, рассмат- риваются эвристические методы, такие как имитация отжига. Такие эвристические ме- тоды являются важными инструментами в наборе любого практикующего алгориста. 7.1. Перебор с возвратом Перебор с возвратом позволяет систематически исследовать все возможные конфигу- рации области поиска. Эти конфигурации могут представлять все возможные располо- жения объектов, т. е. перестановки, или все возможные наборы объектов, т. е. подмно- жества. В других ситуациях может потребоваться выполнить перечисление всех де- ревьев графа, всех путей между двумя вершинами или всех возможных способов группирования вершин по цветам. Общий момент в этих задачах— то. что каждую возможную конфигурацию нужно сгенерировать ровно один раз. Запрет как на повторение, так и на пропуск конфшура- ций означает, что нам нужно определить четкий порядок их генерирования. Мы будем моделировать наше решение в виде вектора а = (щ, а2. ..., а,,}, в котором каждый эле-
252 Часть I. Практическая разработка алгоритмов мент а, выбирается из конечного упорядоченного множества S,. Такой вектор может представлять конфигурацию, в которой а, содержит /-й элемент перестановки. Или же этот вектор может представлять заданное подмножество S', где а, истинно тогда и толь- ко тогда, когда /-Й элемент содержится в X. Этот же вектор может также представлять последовательность ходов в игре пли путь в графе, где а, содержит 7-й элемент после- довательности. На каждом этапе алгоритма перебора с возвратом мы пытаемся расширить данное час- тичное решение а = (ui. а?, .... а*), добавляя следующий элемент в конец последова- тельности. После расширения последовательности нам нужно проверить, не содержит ли она полного решения, и если содержит, то можем ли мы вывести или вычислить его. В противном случае нам нужно выяснить, существует ли возможность расширения частичного решения к полному решению. При переборе с возвратом создается дерево, в котором каждая вершина представляет частичное решение. Если узел у создан в результате перехода от узла х. то эти узлы со- единяются ребром. Такое дерево частичных решений предоставляет альтернативный взгляд на перебор с возвратом, т. к. процесс создания решений в точности соответству- ет процессу обхода в глубину дерева перебора с возвратом. Рассматривая перебор с возвратом как обход в глубину неявного графа, мы создаем естественную рекурсивную реализацию базового алгоритма, псевдокод которого показан в листинге 7.1. Листинг 7.1 Перебор с возвратом Backtrack-DFS(А, к) if А - (ai, а,, ..., а>.) является решением, выводим его else к = к + 1 compute Si while Si # 0 do ak = an element in S1; S = S>. ak Backtrack-DFS(A,k) Хотя для перечисления решений можно было бы также применить и обход в ширину, обход в глубину намного предпочтительнее, т. к. он занимает меньше места. Текущее состояние поиска полностью представляется путем от корня к текущему узлу обхода в глубину. Требуемое для этого место пропорционально высоте дерева. А при обходе в ширину в очереди сохраняются все узлы текущего уровня, для чего нужно место, пропорциональное ширине дерева поиска. Для большинства представляющих интерес задач ширина дерева возрастает экспоненциально по отношению к его высоте. Реализация Код алгоритма перебора с возвратом показан в листинге 7.2. Листинг 7.2. Реализация алгоритма перебора с возвратом bool "finished = FALSE; /* Найдены все решения? */ backtrack(int а[], int k, data input)
Гпава 7. Комбинаторный поиск и эвристические методы 253 int с[MAXCANDIDATES] ; /* Кандидаты для следующей позиции */ int ncandidates; /* Количество кандидатов на следующую позицию*/ int i; /* Счетчик ♦/ if (is_a_solution(a, k, input)) process_solution(a, k, input); else k = k+1; construct_candidates (a, k, input, c, &ncandidates); for (i=0; icncandidates; i++) { a[k] c[i]; make _move(a, k, input); backtrack(a, k, input); unmake_move(a, k,input): if (finished) return; /* Досрочное завершение*/ Метод перебора с возвратом обеспечивает правильность результата, перечисляя все возможные комбинации, а эффективность обеспечивается тем, что никакое состояние не исследуется более одного раза. Изучите, как рекурсия позволяет создать легкую и элегантную реализацию алгоритма перебора с возвратом. Так как при каждом рекурсивном вызове создается новый мас- сив кандидатов с, то подмножества еще не рассмотренных кандидатов на расширение решения в каждой позиции не будут пересекаться друг с другом. Алгоритм содержит пять процедур, специфичных для конкретных приложений: ♦ is_a_solution(a,k,input). Эта булева функция проверяет, составляют ли первые к элементов вектора а полное решение данной задачи. Последний аргумент, input, позволяет передавать в процедуру общую информацию. Например, с его помощью можно указать значение п. представляющее заданный размер решения. Эта инфор- мация может быть полезной при создании перестановок или подмножеств из п эле- ментов. но при создании объектов переменного размера, таких как последователь- ности ходов игры, можно передавать другие данные, более соответствующие ситуа- ции; ♦ construct_candidates(a, k,input,с,ncandidates). Эта процедура записывает в мас- сив с полный набор возможных кандидатов на к-ю позицию вектора а, при задан- ном содержимом первых А-1 позиций. Количество кандидатов, содержащихся в этом массиве, заносится в переменную ncandidates. Так же. как в предыдущей функции, аргумент input можно использовать для передачи в процедуру вспомога- тельной информации; ♦ process_solution(a, k, input). Эта процедура выводит, вычисляет или иным образом обрабатывает полное решение после его создания; ♦ make move (а, к, input) И unmake move (а, к, input). Эти процедуры ПОЗВОЛЯЮТ моди- фицировать структуру данных в ответ на последнее перемещение, а также очистить структуру данных, если мы решим отменить это перемещение. При необходимости эту структуру можно было воссоздавать с нуля на основе вектора решений а, но
254 Часть I. Практическая разработка алгоритмов этот подход не является эффективным, когда с каждым перемещением связаны из- менения, которые можно с легкостью отменить. Эти процедуры функционируют в виде заглушек (т. е. ничего не делают) в вызовах процедуры backtrack!) во всех примерах этого раздела, но применяются в про- грамме решения головоломок судоку в разделе 7.3. Для внепланового завершения программы используется глобальный флаг finished, который можно установить в любой прикладной процедуре. Чтобы по-настоящему понять принцип работы алгоритма перебора с возвратом, нужно разобраться, как можно создавать такие объекты, как перестановки и подмножества, определяя правильное пространство состояний. Несколько примеров пространств со- стояний рассматривается в последующих подразделах. 7.1.1. Генерирование всех подмножеств Критическим вопросом при разработке пространств состояний для представления ком- бинаторных объектов является количество объектов, которые нужно представить. Сколько существует подмножеств множества из п элементов, например, множества целых чисел {1, ..., и}? Для п — 1 существует два таких подмножества — и {1}. Для п - 2 существует четыре подмножества, а для п = 2— восемь. Как видим, количество подмножеств удваивается с каждым новым элементом множества; таким образом, для множества из п элементов существует 2" подмножеств. Каждое подмножество описы- вается содержащимися в нем элементами. Чтобы сгенерировать все 2" подмножеств, мы создаем массив (вектор) из п ячеек, в котором булево значение а, указывает, содер- жит ли данное подмножество z-й элемент. В схеме нашего общего алгоритма перебора с возвратом Sk = (true, false), в то время как значение <7 является решением при к = п. Теперь мы можем сгенерировать все подмножества, используя простые реализации процедур is_a_solution(), construct_candidates() И process_solution(), показанные в листинге 7.3. is_a_solution(int а[], int k, int n) { return (k == n); /* k == n? */ 1 construct_candidates(int a[], int k', int n, int c[], int *ncandidates) { C[0] = TRUE; C[l] = FALSE; *ncandidates = 2; ) process_solution(int a[], int k) { int i; /* Счетчик*/ printf("{"I; for (i=l; i<=k; i++) if (a[i] == TRUE) printff %d",i); printf(" }\n”);
Гпава 7. Комбинаторный поиск и эвристические методы 255 Самой сложной из этих трех процедур оказывается процедура для вывода каждого подмножества после его создания! Наконец, при вызове процедуры backtrack ей нужно передать соответствующие аргу- менты. Конкретно, это означает предоставление указателя на пустой вектор решения, установление к в ноль для обозначения того, что вектор пустой, и указание количества элементов в универсальном множестве (листинг 7.4). Листинг 7.4. Вызов процедуры backtrack () для генерирования подмножеств generate_subsets(int n) ( int a[NMAX]; /* Вектор решений */ backtrack(a,0,n); В каком порядке будут генерироваться подмножества множества {1. 2. 3}? Это зависит от порядка перемещений, выполняемых в процедуре construct candidates. Так как true всегда идет перед false, то сначала будут сгенерированы все подмножества для true, а пустое подмножество, состоящее из всех false, генерируется последним: {123}, {12}. {13}, {1}, {23}, {2}, {3}. {}. Изучите внимательно этот пример и убедитесь, что вы понимаете, как работает про- цедура перебора с возвратом. Задача генерирования подмножеств рассматривается более подробно в разделе 14.5. 7.1.2. Генерирование всех перестановок Обязательным предварительным условием генерирования перестановок множества элементов {1..п] является подсчет их количества. Для первого элемента перестанов- ки имеется и вариантов. Вторым элементом может быть любой из оставшихся п — 1 элементов, т. к. в перестановках повторение элементов не допускается. Последователь- ное применение этой процедуры дает нам nl = j-[" / разных перестановок. Этот способ подсчета количества перестановок подсказывает подходящую структуру для их представления. В частности, создаем массив (вектор) а из и ячеек. Набором кандидатов на z-e место будет набор элементов, которые не вошли в (/ — 1) элементов частичного решения, что соответствует первым i - 1 элементам перестановки. В схеме общего алгоритма перебора с возвратом 5* = {1./г} - а, причем значение а является решением, когда к = п. Соответствующая процедура construct_candidates() представлена в листинге 7.5. Г'- ................................ -..... ..................-... - Листинг 7.5. Процедура conetruct_candidates О для генерирования всех перестановок construct_candidates(int а[], int k, int n, int c[], int *ncandidates) int i; /* Счетчик */ bool in_perm[NMAX]; /* Какие элементы в. перестановке? */
256 Часть I. Практическая разработка алгоритмов for (i=l; i<NMAX; i++) in_perm[i] = FALSE; lor (i=0; i<k; i++) in_perm[ a[i] ] = TRUE; ‘ncandidates - 0; lor (i=l; i<=n; i++) if (in_perm[i] == FALSE) { c[ ‘ncandidates] = i; ‘ncandidates = ‘ncandidates + 1; Узнать, является ли i кандидатом на k-e место в перестановке, можно путем перебора всех элементов k- 1 массива а, чтобы убедиться в отсутствии совпадений. Но гораздо лучший способ отслеживания элементов, находящихся в частичном решении, — орга- низовать структуру данных в форме вектора разрядов (см. раздел 12.5). что позволит нам выполнять проверку за постоянное время. Для завершения программы генерирования перестановок необходимо определить ис- пользуемые в ней процедуры process solution и is a solution, а также передать необ- ходимые параметры вызываемой процедуре backtrack. Как определение процедур, так и передача параметров выполняется, по сути, так же, как и для программы генерирова- ния подмножеств (листинг 7.6). Листинг 7.6. Процедуры генерирования перестановок process_solution(int а[], int k) { int i; /* Счетчик */ for (i=l; i<=k; i++) printf(" %d",a[i]); printf("\n"); I is_a_solution(int a[], int k, int n) { return (k == n); 1 generate_permutations(int n) i int a[NMAX]; /* Вектор решений */ backtrack (a, 0, n) ; В результате упорядочения кандидатов эти перестановки генерируются в отсортиро- ванном порядке, т. е. 123, 132. 213. 231. 312 и 321. Задача генерирования перестановок рассматривается более подробно в разделе 14.4. 7.1.3. Генерирование всех путей в графе Задача перечисления всех простых путей от вершины s к вершине / графа является бо- лее сложной, чем перечисление подмножеств или перестановок множества элементов. Не существует явной формулы для определения количества решений в зависимости от количества вершин и ребер, т. к. количество путей зависит от структуры графа.
Гпава 7. Комбинаторный поиск и эвристические методы 257 Начальной точкой любого пути от вершины s к вершине / всегда является вершина 5, т. е. вершина 5 является единственным кандидатом на первое место и = {.$•}. Воз- можными кандидатами на второе место в пути являются такие вершины v, для которых ребро (s, v) находится в графе, т. к. допустимый путь от вершины к вершине идет по ребрам между ними. Вообще говоря, множество Stt, состоит из набора вершин, смеж- ных с вершиной «*, которые не были использованы в частичном решении А. Определе- ние соответствующей процедуры construct candidates () приводится в листинге 7.7. Листинг 7.7. Процедура construct_candidat.es () для перечисления всех путей в графе “onstruct candidates(int а[], int k, int n, int c[], int *ncandidates) int i; /* Счетчики *7 bool in_sol[NMAX]; /‘ Что уже находится в решении? */ edgenode *р; /* временный указатель *7 int last; /* Последняя вершина в текущем пути */ for (i=l; KNMAX; i++) in_sol[i] = FALSE; for (i=l; i<k; i++) in_sol [ a[i] ] = TRUE; if (k==l) { /* Всегда начинаем с вершины 1 */ с[0] = 1; ’ncandidates = 1; else *ncandidates = 0; last = a[k-l] ; p = g.edges[last]; while (p != NULL) { if (!in_sol[ p->y ]) { c[*ncandidates] = p->y; *ncandidates = *ncandidates + 1; ') p = p->next; Условием правильного пути является сд = /. Процедуры для определения решения и его обработки приводятся в листинге 7.8. Листинг 7.8. Процедуры для определения решения и его обработки .s_a_solution(int а[], int k, int t) return (a[k] == t) ; process_solution(int a[], int k) solutioncount ++; /* Подсчитываем все пути от s к t */ 9 Зак 3741
258 Часть I. Практическая разработка алгоритмов Вектор решений А должен иметь достаточно элементов для представления всех п вер- шин, хотя, скорее всего, для большинства путей они не понадобятся. На рис. 7.1 пока- зано дерево поиска с перечислением всех путей между определенной парой вершин графа. Рис. 7.1. Граф (а) и дерево поиска со всеми путями между вершинами s и t (о) 7.2. Отсечение вариантов поиска Метод перебора с возвратом обеспечивает правильность результата, перечисляя все возможные комбинации. Перечисление всех п! перестановок п вершин графа и выбор наилучшей из них дает нам правильный алгоритм для определения оптимального мар- шрута коммивояжера. Для каждой перестановки мы можем видеть, в действительности ли граф G содержит все ребра, указываемые в маршруте, и если содержит, то веса всех ребер суммируются. Но предварительное генерирование всех перестановок для их дальнейшего исследова- ния будет расточительным расходованием ресурсов. Допустим, что наш путь начинает- ся в вершине vh но ребро (vi, v2) не является частью графа G. Тогда рассмотрение сле- дующих (и-2)! перестановок, сгенерированных, начиная с ребра (vb v2), будет бес- смысленной тратой времени. Разумнее отказаться от поиска после (vj. v2) и продолжить с (v,. v3). Ограничение набора следующих элементов таким образом, чтобы остались лишь перемещения, допустимые для текущей частичной конфигурации, позволяет зна- чительно понизить сложность поиска. Отсечением (pruning) называется метод прекращения поиска решения, как только установлено, что данное частичное решение невозможно расширить до полного. Зада- ча коммивояжера состоит в определении самого дешевого маршрута, который прохо- дит через все вершины. Допустим, что мы нашли маршрут / со стоимостью С,. Потом в процессе продолжения поиска мы получаем частичное решение с суммой стоимости вершин С.4 > С,. Нужно ли продолжать исследование этого узла? Нет. так как любой маршрут <2|, ..., с этим префиксом будет стоить больше, чем маршрут /, и поэтому наверняка не является оптимальным. Отсечение таких бесперспективных частичных маршрутов на раннем этапе может существенно улучшить время исполнения.
Гпава 7 Комбинаторный поиск и эвристические методы 259 Другим средством уменьшения вариантов возможных решений комбинаторного поис- ка является применение симметрии. Отсечение частичных решений, идентичных рас- смотренным ранее, требует умения распознавать симметричные области поиска. Возь- мем, например, состояние поиска кратчайшего пути после того, как были рассмотрены все частичные решения, начиная с vi. Есть ли смысл продолжать поиск с частичными решениями, начинающимися с V2? Нет. Любой маршрут, начинающийся и заканчи- вающийся в вершине г?, можно рассматривать как смещенный маршрут, начинающий- ся и заканчивающийся в вершине vb т. к. эти маршруты являются циклами. Таким об- разом, для п вершин существует только (п- I)! разных маршрутов, а не и!. Ограничи- вая первый элемент маршрута вершиной vb мы получаем экономию времени порядка п. не пропуская при этом никаких представляющих интерес решений. Такие законо- мерности могут быть далеко не очевидными, но, когда они обнаружены, их можно экс- плуатировать. Подведение итогов Комбинаторный поиск можно использовать совместно с методами отсечения для решения задач оптимизации небольшого размера. Смысл понятия "небольшой" зависит от кон- кретной задачи, но обычно размер составляет 15 < п < 50 элементов. 7.3. Судоку Головоломки судоку очень популярны во всем мире. Многие газеты печатают их в своих дневных выпусках, выпущены целые сборники этих головоломок. О популярно- сти судоку можно судить по тому факту, что авиакомпания British Airways издала при- каз, запрещающий бортпроводникам решать их во время взлета и посадки. Я даже за- метил. что во время моих лекций по алгоритмам в задних рядах аудитории довольно многие занимаются решением этих головоломок. Что же представляет собой судоку? В наиболее распространенной форме это квадрат размером 9x9 клеточек, некоторые из которых содержат цифры от 1 до 9, а остальные пустые. Решение головоломки состоит в заполнении пустых клеточек таким образом, чтобы каждая строка, каждый столбец и каждый малый квадрат размером 3x3 кле- точки, содержали все цифры от 1 до 9, без повторений и без пропусков. Пример голо- воломки судоку и ее решение показаны на рис. 7.2. 6 3 5 1 7 2 7 3 4 8 1 1 2 8 4 г- О а) б 7 3 8 9 1 5 1 2 9 1 2 7 3 5 4 8 б 8 4 5 6 1 2 9 7 3 7 9 8 9 6 1 3 5 4 5 2 6 4 7 3 8 9 1 1 3 1 5 8 9 2 6 7 4 6 9 1 2 8 7 3 5 2 8 7 3 5 6 J 1 9 3 О 1 9 4 7 ь 2 8 о) Рис. 7.2. Головоломка судоку (а) и ее решение (б)
260 Часть I. Практическая разработка алгоритмов Судоку хорошо поддается решению методом перебора с возвратом. Мы используем головоломку на рис. 7.2 для лучшей иллюстрации этого алгоритмического метода. Пространством состояний будут пустые клеточки, каждая из которых будет в конечном итоге заполнена какой-либо цифрой. Кандидатами на заполнение пустой клеточки (i,y) являются целые числа от 1 до 9, которых еще нет в строке /, столбце /ив малом квад- рате, содержащим клеточку Возврат осуществляется, когда больше нет кандида- тов на заполнение клеточки. Вектор решения а, поддерживаемый процедурой backtrack, может содержать в каждой ячейке только одно целое число. Этого достаточно для хранения содержимого клеточ- ки (числа 1-9). но не для хранения ее координат. Поэтому для хранения позиций ходов мы используем отдельный массив boardtype. Основные структуры данных для под- держки нашего решения определены в листинге 7.9 Листинг 7.9. Определение основных структур данных Udefine DIMENSION 9 /* Доска размером 9*9 */ #define NCELLS DIMENSION*DIMENSION /* На доске 9*9 имеется 81 клеточка */ typedef struct int х, у; /* Координаты х и у клеточки */ ) point; typedef struct t int m[DIMENSION+1][DIMENSION+1]; /* Матрица содержимого доски */ int freecount; /* Количество оставшихся пустых клеточек */ point move [NCEILS+1]; /* Как были заполнены клеточки */ } boardtype; На очередном шаге игры мы должны сначала выбрать открытую клеточку, которую хотим заполнить следующей (процедура next_square), затем определить числа, являю- щиеся кандидатами на заполнение этой клеточки (процедура possible values). Эти процедуры (листинг 7.10), по сути, ведут учет ходов, хотя некоторые детали их работы могут сильно повлиять на производительность. Листинг 7.10. Генерирование кандидатов на заполнение клеточки construct-Candidates(int а[], int k, boardtype *board, int c[],int *ncandidates) { int x,у; /* Позиция следующего хода */ int i; /* Счетчик */ bool possible[DIMENSION+1]; /* Какие числа можно использовать для данной клеточки */ next_square(&х,&у,board); /* Какую клеточку заполнять следующей */ board->move[k].х = х; /* Сохранение выбора следующей клеточки */ board->move[к].у = у; *ncandidates = 0; if ((х<0) SS (у<0)) return; /* Ошибка: нет допустимых ходов */ possible_values(х,у,board,possible); for (i=0; i<=DIMENSION; i++)
Гпава 7. Комбинаторный поиск и эвристические методы 261 if (possible[i] == TRUE) { • [*ncandidates] = i; *ncandidates = *ncandidates - 1; Структуры данных игровой доски необходимо обновлять, чтобы отображать заполне- ние клеточки значением кандидата, а также очищать заполненные клеточки в случае необходимости возврата с данной позиции. Для выполнения этих обновлений исполь- зуются процедуры make move и unmake_move (листинг 7.11), которые вызываются непо- средственно ИЗ процедуры backtrack. Листинг 7.11. Процедуры maxci_move л u._nake_move make_move (int а[], int k, boardtype *board) fill_square(board->move[k].x,board->move[k].y,a[k],board); unmake_move(int a[], int k, boardtype *board) free_square(board->move[k].x,board->move[k].y,board); Одной из важных задач, выполняемых этими процедурами, является отслеживание ко- личества остающихся на доске пустых клеточек. Решение найдено, когда на доске больше нет пустых клеточек (листинг 7.12). Листинг 7.12. Процедура отслеживания пустых клеточек is_a_solution(int а[], int k, boardtype *board) if (board->freecount == 0) return (TRUE); else return(FALSE); Когда решение найдено, устанавливается глобальный флаг finished, что служит сигна- лом к прекращению поиска и выводу решения. Это можно делать, не опасаясь никаких последствий, т. к. классические головоломки судоку могут иметь только одно решение. Головоломки с расширенной интерпретацией могут иметь громадное количество ре- шений. В самом деле, для пустой головоломки (т. е. без начальных цифр) существует 6 670 903 752 021 072 936 960 решений. Чтобы не просматривать все эти решения, мы и прекращаем поиск (листинг 7.13). Листинг 7.13. Завершение поиска и обработка решения process_solution(int а[], int k, boardtype *board) I print_board(board); finished = TRUE;
262 Часть I, Практическая разработка алгоритмов Эта процедура завершает нашу программу, но остается необходимость в написании процедур для определения следующей клеточки для заполнения (next square) и поиска кандидатов на заполнение этой клеточки (possibie_vaiues). Для выбора следующей клеточки для заполнения подходят два способа. ♦ Выбор произвольной клеточки. Выбираем первую попавшуюся пустую клеточку. 11ам все равно, какую выбирать, поскольку нет очевидных оснований считать, что один эвристический метод окажется лучше другого. ♦ Выбор клеточки с наименьшим количеством кандидатов. При этом подходе мы проверяем количество оставшихся кандидатов на заполнение каждой пустой кле- точки (лу), т. е. количество цифр, которые еще не используются ни в строке /, ни в столбце /, ни в малом квадрате, содержащим клеточку (/, /). По результатам этих ис- следований мы выбираем клеточку с наименьшим количеством кандидатов на ее заполнение. Хотя оба подхода дают правильные результаты, второй позволяет найти решение на- много быстрее. Часто имеются пустые клеточки только с одним кандидатом, которые нельзя заполнить иной цифрой, кроме как этим единственным оставшимся кандида- том. Такие клеточки вполне можно заполнить на данном этапе, т. к. это поможет уменьшить количество возможных кандидатов для заполнения других пустых клето- чек. Конечно же, в этом случае выбор каждой следующей клеточки для заполнения будет занимать больше времени, но если головоломка достаточно легкая, то нам. мо- жет быть, никогда не придется выполнять возврат. Если для клеточки с наименьшим количеством кандидатов имеется два кандидата, то вероятность угадать правильный из них с первого раза равна 1/2, по сравнению с веро- ятностью 1/9 угадать правильного кандидата для клеточки без ограничений на количе- ство кандидатов. Уменьшив среднее количество кандидатов для каждой клеточки, на- пример, с трех до двух, мы получим громадное повышение производительности, т. к. это уменьшение дает прогрессивный выигрыш для каждой следующей клеточки. На- пример, если нам нужно заполнить 20 клеточек, то два кандидата на каждую клеточку дают 1 048 576 возможных вариантов заполнения. А уровень ветвления, равный 3, для каждой из 20 клеточек дает в 3 000 раз больше вариантов! Выбрать возможных кандидатов для каждой клеточки можно двумя способами. ♦ Локальный выбор. Наш алгоритм перебора с возвратом выдаст правильный резуль- тат, если процедура генерирования кандидатов на заполнение клеточки (z,j), т. е. процедура possibie_vaiues, действует очевидным образом и предоставляет на вы- бор цифры от 1 до 9, которых еще пет в данной строке, столбце или малом квадрате. ♦ Просмотр вперед. Что будет, если для нашего текущего частичного решения суще- ствует какая-то другая пустая клеточка, для которой локальные критерии не остав- ляют кандидатов? В таком случае данное частичное решение невозможно довести до полного. Получается, что ситуация вокруг какой-то другой клеточки делает дей- ствительное количество кандидатов для клеточки (/, /) равным нулю! Со временем мы подойдем к этой другой клеточке, обнаружим, что для нее нет дей- ствительных кандидатов, и нам придется возвращаться назад. Но зачем вообще ид- ти к этой клеточке, если все затраченные на это усилия будут напрасными? Будет
Глава 7 Комбинаторный поиск и эвристические методы 263 намного выгоднее выполнить возврат к текущей позиции и продолжить поиск в другом направлении1. Для успешного отсечения непродуктивных ветвей поиска требуется просмотр вперед, позволяющий обнаружить, что данный путь решения является тупиковым, и возвра- титься из него на новый как можно раньше. В табл. 7.1 показано количество вызовов процедуры определения решения is_a_soiution для всех четырех комбинаций выбора следующей клеточки и возможных кандидатов для трех разных уровней сложности головоломки судоку. ♦ Головоломка низкого уровня сложности предназначена для решения человеком, а не компьютером. В действительности, моя программа решила ее без единого воз- врата, когда для следующей клеточки выбиралась клеточка с наименьшим количе- ством кандидатов. ♦ Головоломка средней сложности оказалась не под силу ни одному из вышедших в финал участников Мирового чемпионата по судоку в марте 2006 года. Но для про- граммы потребовалось только несколько возвратов, чтобы решить эту головоломку. ♦ Головоломка высокого уровня сложности показана на рис. 7.2 и содержит только 17 заполненных клеточек. Это наименьшее известное количество заполненных кле- точек на всех экземплярах задачи, которое дает только одно полное решение. Таблица 7.1. Количество шагов для получения решения при разных стратегиях отсечения Условие отсечения Уровень сложности next_square possible_values Низкий Средний Высокий произвольное значение локальный выбор 1,904.832 863,305 программа не завершена произвольное значение будущий выбор 127 142 12,507,212 наименьшее количество кандидатов локальный выбор 48 84 1.243.838 наименьшее количество кандидатов будущий выбор 48 65 10.374 Что считается головоломкой высокого уровня сложности, зависит от применяемого для ее решения эвристического алгоритма. Несомненно, среди ваших знакомых есть люди, которым теория кажется труднее, чем практическое программирование, а есть и такие, кто думает иначе. Алгоритм А вполне может считать, что задача 1\ легче, чем задача Ь, в то время как алгоритм В видит трудность этих задач в обратном порядке. 1 Этот подход с просмотром вперед мог бы естественным образом следовать из подхода выбора кле- точки с наименьшим количеством кандидатов, если бы было разрешено выбирать клеточки, для кото- рых нет кандидатов. Но в моей реализации уже заполненные клеточки рассматривались, как не имею- щие ходов, что ограничивает выбор следующей клеточки клеточками, по крайней мере, с одним кан- дидатом.
264 Часть I Практическая разработка алгоритмов Какие выводы мы можем сделать из этих экспериментов? Предварительное исследова- ние дальнейших вариантов решения с целью отсечения тупиковых является самым лучшим способом разрежения пространства поиска. Без применения этой операции мы никогда не решили бы головоломку самого высокого уровня, а более легкие голово- ломки решили в тысячи раз медленнее. Разумный подход к выбору следующей клеточки имел аналогичный эффект, хотя тех- нически мы просто изменяли порядок выполнения работы. Но обработка клеточек с наименьшим количеством кандидатов первыми равнозначна понижению исходящей степени каждого узла дерева, а каждая заполненная клеточка уменьшает количество возможных кандидатов для других клеточек. При произвольном выборе следующей клеточки решение головоломки на рис. 7.2 за- няло почти час. Несомненно, моя программа решила большинство других головоломок быстрее, но головоломки судоку предназначены для решения людьми за гораздо меньшее время. Выбор клеточки с наименьшим количеством кандидатов позволил со- кратить время поиска более чем в 1 200 раз. Вот так проявляется вся мощь отсечения тупиковых вариантов поиска. Использование даже простых стратегий отсечения может значительно уменьшить время исполнения. 7.4. История из жизни. Покрытие шахматной доски Каждый ученый мечтает о решении классической задачи — такой, которая оставалась нерешенной в течение нескольких столетий. Есть что-то романтическое в общении с ушедшими поколениями, участии в эволюционном развитии научной мысли и оказа- нии помощи человечеству на его пути вверх по лестнице технического прогресса. Задача может оставаться нерешенной в течение длительного времени по разным при- чинам. Возможно, она такая трудная, что для ее решения требуется уникальный мощ- ный интеллект. Или же, возможно, еще не были разработаны идеи или методы, тре- буемые для решения данной задачи. Наконец, возможно, что задача никого не заинте- ресовала настолько, чтобы он всерьез занялся ею. Однажды я помог решить задачу, которая оставалась нерешенной свыше сотни лет. Люди увлекаются игрой в шахматы в течение тысяч лет. Эффект комбинаторного взрыва был впервые зафиксирован в легенде, согласно которой изобретатель шахмат запросил у правителя в качестве награды за свое изобретение самую малость — одно зерно пшеницы на первое поле шахматной доски, два— на второе и т. д„ т. е. вдвое больше на каждое следующее (z + 1)-е поле, чем на предыдущее z-e. Но когда правитель узнал, что ему придется раскошелиться на ^”С|2' = 2Ы - 1 - 18 446 744 073 709 551 615 зерен пшеницы, то его ликование по поводу приобретения такой замечательной игры по такой низкой цене сменилось негодованием по поводу алчности и коварства изобре- тателя. Отрубив изобретателю голову, правитель был первым, кто применил метод отсечения в качестве меры контроля комбинаторного взрыва.
Гпава 7. Комбинаторный поиск и эвристические методы 265 В 1849 г. немецкий гроссмейстер Йозеф Клинг (Josef Kling) поставил вопрос, возмож- но ли одновременно держать под ударом все 64 поля шахматной доски восемью основными фигурами— королем, ферзем, двумя конями, двумя ладьями и двумя сло- нами разного цвета. Фигуры не держат под ударом поле, на котором они находятся. Расстановки фигур, которые держат под ударом 63 поля, подобные показанным на рис. 7.3, были известны с далеких времен, но вопрос, являются ли такие расстановки наилучшими возможными, оставался открытым. Казалось, задача решается исчерпывающим комбинаторным перебором, но возмож- ность решить ее таким способом зависела от размера пространства поиска. Посмотрим, сколько существует способов расстановки на шахматной доске восьми основных шахматных фигур (короля, ферзя, двух ладей, двух слонов и двух коней). Очевидный предел таких расстановок равен 64!/(64 - 8)! = 178 462 987 637 760 = 1015. Однако было бы неразумно надеяться на перебор более чем 109 расстановок на обыч- ном компьютере за приемлемое время. Таким образом, чтобы решить задачу расстановок, необходимо выполнить отсечение значительного объема пространства поиска. После удаления ортогональных и диаго- нальных симметричных вариантов для ферзя останется только десять возможных по- зиций (рис. 7.4). После постановки ферзя для размещения пары ладей или коней остается 64'63/2 = = 2 016 положений, для короля — 64, и 32 для каждого из двух слонов. После такого отсечения остается 2 663 550 812 160 = 1013 разных расстановок, что все равно слиш- ком много для исчерпывающего перебора. Все эти расстановки можно было сгенерировать с помощью поиска с возвратом, но пространство поиска нуждалось в дополнительном значительном разрежении. Для это- го нужно было найти способ, чтобы быстро доказать, что для определенной частичной расстановки не существует возможности ее завершения, чтобы поставить под удар все 64 поля. Предположим, что мы уже разместили на доске семь фигур, которые держат под ударом все поля за исключением десяти. Далее допустим, что осталось разместить короля. Существует ли в данной ситуации поле, с которого король может держать под
266 Часть I. Практическая разработка алгоритмов ударом оставшиеся девять полей? Ответ однозначно должен быть отрицательным, т. к. по правилам шахмат король может держать под ударом самое большее восемь полей. Таким образом, нет смысла проверять наличие решения при расположении короля на любом из оставшихся полей. Такая стратегия отсечения расстановок может дать боль- шую экономию, но для ее реализации необходимо внимательно исследовать порядок постановки фигур на доску. Каждая фигура может держать под ударом определенное максимальное количество полей: ферзь— 27, король и конь— 8, ладья— 14, а слон— 13. Хорошей стратегией может быть постановка фигур на доску в порядке убывания их влияния. Таким образом, мы можем выполнять отсечение дальнейших расстановок, когда количество полей, не находящихся под боем, превышает сумму возможностей оставшихся непоставленных фигур. Эта сумма минимизируется поста- новкой фигур в убывающем порядке их возможностей держать поля под ударом. Когда мы реализовали перебор с возвратом, используя эту стратегию отсечения, мы отсекли свыше 95% пространства поиска. После оптимизации метода генерирования постановок фигур наша программа могла исследовать 1 000 полей за секунду. Но и это было слишком медленно, т. к. 10|2/103 = 109 секунд означало свыше 11 000 дней! Хотя мы могли бы настроить программу и ускорить время ее исполнения примерно на поря- док, но в действительности нам было нужно найти способ отсечь еще больше тупико- вых расстановок. Чтобы отсечение было эффективным, нужно устранить большое количество расстано- вок одним приемом, а наши предыдущие попытки в этом направлении были слишком слабыми. А если поставить на доску не восемь фигур, а больше? Очевидно, что чем больше фигур поставлено на доску, тем больше вероятность, что они будут держать под ударом все 64 поля. Но если большее количество фигур не держит под ударом все 64 поля, то и любое из восьмифигурных подмножеств этого множества не способно на это. Такой подход позволяет устранить огромное количество расстановок, удалив всего лишь один узел. Так что в последней версии нашей программы узлы дерева поиска представляли рас- становки, которые могли содержать любое количество фигур и больше, чем одну фигуру на одном и том же поле. Для определенной расстановки мы различали сильную и слабую атаку поля. Сильная атака соответствует обычной атаке по шахматным пра- вилам. Поле считается слабо атакованным, если оно сильно атаковано некоторым под- множеством имеющихся фигур, т. е. если блокировка одних фигур другими игнориру- ется. Как можно видеть на рис. 7.5, все 64 поля можно держать под слабой атакой во- семью фигурами. Наш алгоритм выполнял два прохода. В первом проходе перечислялись расстановки, в которых каждое поле находилось под слабой атакой, а во втором список разрежался, благодаря фильтрации расстановок с блокирующими фигурами. Расстановку со слабой атакой вычислить намного легче, т. к. в этом случае нет необходимости принимать во внимание блокирующие фигуры, а любое множество сильных атак является подмно- жеством множества слабых атак. Любую расстановку, содержащую поле под слабой атакой, можно было отсечь. Эта версия программы была достаточно эффективной, чтобы на медленном IBM PC-R.T 1988 года выпуска завершит > поиск меньше чем за один день. Она не нашла ни
Глава 7. Комбинаторный поиск и эвристические методы 267 одной расстановки, удовлетворяющей первоначальным условиям задачи. Но с ее по- мощью мы смогли доказать, что возможно поставить под удар все поля шахматной доски, используя семь фигур, при условии, что ферзь и конь могут занимать одно и то же поле. На рис. 7.6 показано соответствующее расположение фигур, причем ферзь и конь, расположенные на одном поле, обозначены белым ферзем. Подведение итогов Использование интеллектуальной стратегии отсечения тупиковых решений может позво- лить с удивительной легкостью решить комбинаторные задачи, на первый взгляд кажу- щиеся неразрешимыми. Выполненное должным образом отсечение окажет большее воз- действие на время перебора, чем любой другой фактор. 7.5. Эвристические методы перебора Метод перебора с возвратом позволяет нам найти наилучшее из возможных решений согласно данным условиям. Но любой алгоритм, исследующий все возможные конфи- гурации, обречен на провал на входных экземплярах большого размера. Эвристические методы предоставляют альтернативный подход к оптимизации трудных комбинатор- ных задач. В этом разделе мы рассмотрим такие эвристические методы поиска. Основное внима- ние уделяется методу имитации отжига, который я считаю наиболее надежным для практического применения. Эвристические поисковые алгоритмы поначалу кажутся каким-то шаманством, но если их подвергнуть тщательному исследованию, то выяс- нится, что многое в их работе поддается вполне логическому объяснению. Мы рассмотрим три разных эвристических метода поиска: произвольную выборку (random sampling), градиентный спуск (gradient descent) и имитацию отжига (simulated annealing). Для сравнения этих трех эвристических методов мы испытаем их на задаче коммивояжера. Все три метода имеют два общих компонента. ♦ Представление пространства решений. Это полное, но при этом краткое описание множества возможных решений задачи. Для задачи коммивояжера пространство решений содержит (п— 1)! элементов, т. е. все возможные циклические перестанов-
268 Часть I. Практическая разработка алгоритмов ки вершин. Для представления каждого элемента пространства решений нам требу- ется соответствующая структура данных. Для задачи коммивояжера решения- кандидаты можно представлять с помощью массива 5, содержащего и - 1 вершин, где ячейка S, определяет (/ + 1)-ю вершину маршрута, начинающегося в вершине V|. ♦ Функция стоимости. Поисковые алгоритмы должны иметь функцию стоимости или оценки, чтобы определять качество каждого элемента пространства решений. Наш эвристический алгоритм поиска определяет элемент с наилучшей возможной оценкой — наибольшим или наименьшим значением, в зависимости от природы за- дачи. В случае задачи коммивояжера функция стоимости для оценки данного кан- дидата решения 5 должна лишь сложить вместе все сопутствующие затраты, а именно вес всех ребер (S„ S, J, где S„ , соответствует v,. 7.5.1. Произвольная выборка Самым простым методом организации поиска в пространстве решений является произ- вольная выборка, которая также называется методом Монте-Карло. В этом случае по- следовательно создаются и оцениваются произвольные решения; процесс прекращает- ся. как только будет найдено достаточно хорошее решение или когда нам надоест ждать его получения (что более вероятно). В качестве конечного решения выбирается самое лучшее решение изо всех исследованных в процессе выборки. Для действительно произвольной выборки элементы необходимо выбирать из про- странства решений равномерно произвольным образом. Это означает, что вероятность выбора любого элемента из пространства решений в качестве следующего кандидата должна быть одинаковой. Реализация такой выборки может быть сложнее, чем может казаться с первого взгляда. Алгоритмы для генерирования случайных перестановок, подмножеств, разбиений и графов рассматриваются в разделах 14.4-14.7. А в листин- ге 7.14 предоставлена процедура произвольного выбора решений. Листинг 7.14. Процедура произвольного выбора решений random_sampling(tsp_instance *t, int nsamples, tsp_solution *bestsol) { tsp_solution s; /* Текущее решение задачи коммивояжера */ double best_cost; /* Самая лучшая стоимость на данный момент */ double cost_now; /* Текущая стоимость */ int 1; /* Счетчик */ initialize_solution(t->n,&s); best_cost = solution_cost(&s,t); copy_solution(&s,bestsol); for (i=l; i<=nsamples; i++) { random_solution(&s); cost_now = solution_cost(&s,t); if (cost_now < best_cost) ( best_cost = cost_now; copy_solution(5s,bestsol); ) )
Гпава 7. Комбинаторный поиск и эвристические методы 269 В каких случаях можно успешно использовать произвольную выборку? ♦ Пространство решений содержит высокую долю приемлемых решений. Найти тра- винку в стоге сена намного легче, чем иголку. Когда существует много приемлемых решений, то произвольная выборка должна быстро привести к одному из них. Одним из примеров успешного применения метода произвольной выборки является поиск простых чисел. Построение больших случайных простых чисел для ключей является важным аспектом криптографических систем, таких, как, например, R.SA- кодирование. Приблизительно каждое Inn-е число является простым, поэтому, что- бы найти простое число длиной в несколько сотен цифр, требуется выполнить уме- ренное количество выборок. ♦ Пространство решений не является однородным. Произвольную выборку нужно применять в тех случаях, когда отсутствуют какие бы то ни было признаки прибли- жения к решению. Допустим, что вам нужно выбрать среди своих друзей такого, у которого номер полиса социального страхования заканчивается на 00. Для решения этой задачи не существует другого метода, кроме как спросить произвольно вы- бранного товарища, каков номер его страховки. Возвратимся опять к задаче поиска больших простых чисел. Эти числа разбросаны среди других целых чисел без какой-либо системы. Использование произвольной выборки для их поиска будет не хуже любого другого метода. Но подходит ли метод произвольной выборки для решения задачи коммивояжера? Нет. Самым лучшим решением задачи коммивояжера для 48 столиц континентальных шта- тов, которое я нашел, выполнив 1,5 миллиона произвольных перестановок, было 101 712,8 миль. Это больше чем в три раза превышает протяженность оптимального маршрута! Пространство решений этой задачи почти полностью состоит из посредст- венных и плохих решений, поэтому качество решений растет очень медленно с ростом количества выборок, поделенного на время поиска. Чтобы дать представление о раз- бросе решений между выборками, на рис. 7.7 показаны колебания результатов произ- вольно выбираемых решений (как правило, низкого качества) задачи коммивояжера для столиц американских штатов. Подобно задаче коммивояжера, большинство встречающихся задач имеют сравнитель- но небольшое количество хороших решений, но высокую степень однородности про- странства решений. Для эффективного решения таких задач требуются более мощные эвристические алгоритмы. Остановка для размышлений. Выбор пары Задача. Требуется выбрать две произвольные вершины графа и поменять их местами. Предложите эффективный алгоритм для генерирования произвольных элементов с равномерным распределением из (2) неупорядоченных пар множества {1,..., п}. Решение. Рассмотрим следующую процедуру для генерирования случайных неупоря- доченных пар чисел: i - random_int(1, n-1); ] = random int(i+l, n);
270 Часть I. Практическая разработка алгоритмов Производительность метода произвольных выборок для решения задачи коммивояжера 200000 180000 160000 140000 я & 120000 а о. а 100000 го X ё 80000 60000 40000 20000 0 0 200000 400000 600000 800000 1е+06 1.2е+06 1.4е+06 1.6е+06 Количество итераций Рис. 7.7. Соотношения между временем поиска решений и их качеством для решения задачи коммивояжера методом произвольной выборки Ясно, что эта процедура, на самом деле, генерирует неупорядоченные пары, т. к. i < j. Кроме этого, если допустить, что функция random int генерирует целые числа в диапа- зоне своих двух аргументов однородно, также ясно, что можно сгенерировать все (^неупорядоченные пары. Будет ли распределение этих пар однородным? Ответ на этот вопрос является отрица- тельным. Какова вероятность генерирования пары (1,2)? Вероятность получения 1 равна М(п — 1), а вероятность получения 2 также равна 1/(и — 1), что дает нам вероятность получения пары р(1,2) = 1/(и — 1). Но какова вероятность получения пары чисел (и-1, и)? Вероятность получения первого числа равна Мп, но для второго кандидата пары существует только один возможный вариант! Эта пара будет выпадать в п раз чаще, чем первая. Проблема заключается в том, что количество пар, содержащих первое большое число, меньше, чем пар, содержащих первое малое число. Проблему можно было бы решить, вычислив, сколько именно неупорядоченных пар начинаются с числа i (ровно n-i). и соответствующим образом скорректировав вероятность. Тогда второе число пары можно было бы выбирать произвольным образом в диапазоне от i + 1 до и с равномер- ным распределением. Но вместо того, чтобы заниматься вычислениями, лучше воспользоваться тем обстоя- тельством, что генерирование п~ упорядоченных пар является довольно простой зада- чей. Произвольно выбираем два целых числа независимо друг от друга. Игнорируя упорядоченность (т. е. превращая упорядоченную пару в неупорядоченную пару (х, у). гдех < у), мы получаем вероятность генерирования каждой пары разных цифр, равную 2/w“. В случае генерирования пары (х, х) она отбрасывается и генерируется новая пара.
Глава 7. Комбинаторный поиск и эвристические методы 271 Алгоритм для генерирования равномерно распределенных произвольных пар целых чисел с ожидаемым постоянным временем исполнения приводится в листинге 7.15. Листинг 7.15. Генерирование равномерно распределенных произвольных пар целых чисел а ...«•......................... ....................f........... .^.......л..,... . do i = random_int(1,n); ] = random_int(l,n); if (i > j) swap(si,&j); ) while (i=j) ; 7.5.2. Локальный поиск Допустим, что нам нужно найти эксперта по алгоритмам для решения определенной задачи. Мы можем позвонить по случайному телефонному номеру и спросить отве- тившего, не является ли он таким специалистом. В случае отрицательного ответа мы вешаем трубку и повторяем процесс, пока не найдем требуемого нам профессионала. Возможно, что после многократных звонков мы и найдем нужного человека, но было бы намного эффективнее спросить первого ответившего, нет ли среди его знакомых такого, который, возможно, знает эксперта по алгоритмам, и позвонить ему следую- щему. Такая стратегия исследования прилегающего пространства вокруг каждого элемента пространства решений называется локальным поиском. Каждый элемент х в простран- стве решений можно рассматривать, как вершину с исходящим ребром (х, j), направ- ленным к каждому кандидату на решение у. являющемуся соседом х. Поиск выполня- ется из вершины х по направлению к наиболее перспективному кандидату' вблизи этой вершины. Мы не хотим явно создавать граф этого района для пространства решений большого размера. Представьте себе задачу коммивояжера, которая в таком графе будет иметь (п-1)! вершин. Поэтому мы ищем решение посредством эвристического алгоритма, т. к. не надеемся найти решение в течение приемлемого времени исчерпывающим перебором всех вариантов. На самом деле, мы хотим получить механизм перехода к следующему возможному ре- шению, слегка модифицировав текущее. Типичные механизмы перехода включают обмен местами произвольной пары элементов или изменение (вставка или удаление) одного элемента решения. Наиболее очевидным механизмом перехода для задачи коммивояжера был бы обмен местами произвольной пары вершин S, и S, текущего маршрута, как показано на рис. 7.8. Такой обмен вершин изменяет до восьми ребер маршрута, удаляя ребра, смежные с S, и Sj, и добавляя другие. В идеальном случае влияние этих инкрементальных изменений на возможность оценки качества решения можно вычислить также инкрементным об- разом, вследствие чего время исполнения функции оценки стоимости будет пропор- ционально размеру изменений (обычно постоянное), а не линейное по отношению к размеру решения.
272 Часть I. Практическая разработка алгоритмов Рис. 7.8. Улучшение маршрута коммивояжера за счет обмена местами вершин 2 и 6 Эвристический алгоритм локального поиска начинает работу с произвольного элемен- та пространства решений и сканирует смежную с ним область, пытаясь найти подхо- дящего кандидата на выполнение перехода. Для задачи коммивояжера таким кандида- том будет переход, который понижает стоимость маршрута. А для задачи другого типа, называемой восхождением по выпуклой поверхности (hill-climbing), мы пытаемся най- ти самую высшую (или самую низшую) точку, начав поиск в произвольной точке и выбирая любой промежуточный маршрут, ведущий в требуемом направлении. Эта процедура выбора промежуточного отрезка маршрута повторяется до тех пор. пока не дойдет до точки, в которой все исходящие направления не удовлетворяют нашим тре- бованиям (листинг 7.16). Теперь вы "Царь горы" (или "Король канавы"). ........................ .у.,, Листинг 7.16. Процедура восхождения по выпуклой поверхности hill_climbing(tsp_instance *t, tsp_solution *s) { double cost; /* Самая лучшая стоимость на данный момент *7 double delta; /* Стоимость обмена */ int i, j; /* Счетчики */ bool stuck; /* Это решение лучшее? */ double transition(); initialize_solution(t->n, s); random_solution(s); cost = solution_cost(s, t); do { stuck = TRUE; for (i=l; i<t->n; i++) for (j=i+l; j<=t->n; j++) { delta = transition(s, t, i, j); if (delta < 0) { stuck = FALSE; cost = cost + delta; I else transition(s, t, j, i) ; 1 ) while ((stuck);
Гпава 7. Комбинаторный поиск и эвристические методы 273 Но, скорее всего, вы не "Царь горы", И вот почему. Допустим, что вы решили начать свое восхождение на гору, проснувшись утром в отеле на горнолыжном курорте. Бли- жайшей высшей точкой будет верхний этаж отеля, а потом его крыша. А дальше вам идти некуда. Чтобы взобраться на вершину горы, вам нужно сначала выйти на улицу, для чего придется спуститься на первый этаж отеля. Но это нарушает требование, что- бы каждый шаг повышал ваше местонахождение. Алгоритм восхождения по выпуклой поверхности и подобные эвристические алгоритмы, такие как жадный поиск или поиск методом градиентного спуска, позволяют быстро получить оптимальные локальные результаты, но часто не справляются с поиском наилучшего глобального решения. В каких случаях локальный поиск приводит к успеху? ♦ Пространство решений является высокооднородным. Метод восхождения по вы- пуклой поверхности лучше всего работает с выпуклым пространством решений, т. е. пространством, содержащим ровно одно возвышение. Таким образом, независимо от местонахождения в пространстве решений точки начала поиска, у нас всегда имеется направление, в котором нужно продолжать поиск до тех пор, пока мы не дойдем до глобального максимума. Этим свойством обладают многие естественные задачи. В частности, можно счи- тать, что двоичный поиск начинается в середине пространства решений. В этой точ- ке существует только одно из двух возможных направлений, в котором нужно идти, чтобы приблизиться к целевому элементу. Симплексный алгоритм для линейного программирования (см. раздел 13.6) представляет собой не что иное, как восхожде- ние по выпуклой поверхности правильного пространства решений, но, тем не менее, гарантирует получение оптимального решения для любой задачи линейного про- граммирования. ♦ Стоимость оценки изменения намного ниже стоимости глобальной оценки. Стои- мость оценки произвольного решения из п вершин для задачи коммивояжера равна 0(и), т. к. нам нужно суммировать стоимость каждого ребра в циклической пере- становке, описывающей маршрут. Но когда эта стоимость известна, то стоимость маршрута после обмена местами данной пары вершин можно определить за посто- янное время. Если у нас имеется очень большое значение п и ограниченное время для поиска, то лучше потратить это время на выполнение нескольких инкрементальных оценок, чем на несколько произвольных выборок, даже если мы ищем иголку в стоге сена. Основным недостатком локального поиска является то, что как только мы нашли ло- кальный оптимум, то не остается больше никаких вариантов для поиска глобального решения. Конечно же, если у нас имеется время, мы можем начать новый поиск с дру- гой произвольно выбранной точки, но в поисковом пространстве, содержащем много небольших возвышенностей, маловероятно найти оптимальное решение. Насколько хорошим является метод локального поиска для решения задачи комми- вояжера? Намного лучшим, чем произвольная выборка, при сходных временных затра- тах Выполнив около 1,5 миллиона оценок маршрута задачи коммивояжера для столиц 48 континентальных штатов, мы получили самый лучший результат длиной 40 121,2 мили, что всего лишь на 19,5% больше, чем оптимальный маршрут длиной
274 Часть I. Практическая разработка алгоритмов 33 523,7 мили. На рис. 7.9 показаны результаты применения метода локального поиска: многократные переходы от случайных маршрутов к удовлетворительным решениям примерно одинакового качества. Производительность метода восхождения по выпуклой поверхности для задачи коммивояжера Количество итераций Рис. 7.9. Отношение времени поиска к качеству решений задачи коммивояжера для метода восхождения по выпуклой поверхности Хотя этот результат намного лучше, чем полученный методом произвольных выборок, действительно хорошим его назвать трудно. К примеру, были бы вы довольны, если бы вам пришлось платить на 19,6% больше налогов, чем вы в действительности должны? Вывод: нам нужны более мощные методы для получения решения, близкого к опти- мальному. 7.5.3. Имитация отжига Имитация отжига (simulated annealing) представляет собой эвристическую процедуру поиска, которая допускает случайные переходы, что ведет к более дорогостоящим (и, соответственно, худшим) решениям. Это выглядит как шаг назад, но позволяет удержать поиск от зацикливания на оптимальном локальном решении. Если вернуться к нашей аналогии с покорением горных вершин, начиная с комнаты горнолыжного отеля, можно сказать, что теперь нам разрешается сойти вниз по лестнице или выпрыг- нуть в окно, а затем продолжить восхождение в правильном направлении. Идея имита- ции отжига является аналогией физического процесса остывания расплавленных мате- риалов и сопутствующего перехода в твердое состояние. В теории термодинамики энергетическое состояние системы определяется энергетическим состоянием каждой составляющей его частицы. Частица переходит из одного энергетического состояния в другое произвольным образом, при этом переходы между состояниями обуславлива- ются температурой системы.
Глава 7 Комбинаторный поиск и эвристические методы 275 В частности, вероятность перехода Р(е,. е„ Т) из энергетического состояния е, в состоя- ние е, при температуре Т определяется формулой: P(e,,erT) = e‘'''e'y,W }. где кц—постоянная Больцмана. Каков смысл этой формулы? Чтобы ответить на этот вопрос, рассмотрим значение по- казателя степени при разных условиях. Вероятность перехода из высокоэнергетическо- го состояния в низкоэнергетическое очень велика. Тем не менее, существует ненулевая вероятность перехода в высокоэнергетическое состояние, причем переходы малой ам- плитуды вероятнее, чем значительные. Кроме этого, чем выше температура, тем выше вероятность энергетических переходов. Какое отношение все это имеет к комбинаторной оптимизации? При охлаждении фи- зическая система стремится достичь состояния с минимальной энергией. Минимизация полной энергии представляет собой задачу комбинаторной оптимизации для любого набора дискретных частиц. Посредством произвольных переходов, генерируемых со- гласно данному распределению вероятностей, мы можем эмулировать физические процессы, чтобы решать произвольные задачи комбинаторной оптимизации. Псевдо- код алгоритма для такой эмуляции приводится в листинге 7.17. Листинг 7.17. Алгоритм имитации отжига Simulated-Annealing() Создаем первоначальное решение S Инициализируем температуру t repeat for i = 1 to iteration-length do Генерируем произвольный переход из S в ST If (C (S) > C (Si) ) then S = Sc else if (e‘ > random[0, i] ; then S - S. Понижаем температуру t until (больше нет изменений в С(S) Return S Подведение итогов Имитация отжига является эффективным средством, потому что она больше времени уде- ляет "хорошим" элементам пространства решений, чем "плохим", а также потому, что она не зацикливается на одном локальном оптимальном решении. Так же. как и в случае с локальным поиском, представление задачи состоит из пред- ставления пространства решений и задания несложной функции стоимости C(.v) для определения качества конкретного решения. Новым компонентом является график ох- лаждения (cooling schedule), параметры которого управляют вероятностью приемлемо- сти "плохого" перехода в зависимости от времени. В начале поиска мы стремимся использовать фактор случайности, чтобы исследовать пространство поиска, поэтому вероятность приемлемости перехода в низкоэнергетиче-
276 Часть I Практическая разработка алгоритмов ское состояние должна быть высокой. В процессе поиска мы стремимся ограничить переходы переходами к локальным улучшениям и оптимизациям. График охлаждения можно регулировать с помощью следующих параметров: ♦ первоначальная температура системы. Обычно I, = 1: ♦ функция понижения температуры. Обычно t/< = а-/* |, где 0,8 < a < 0,99. Это означает экспоненциальное понижение температуры, а не линейное; ♦ количество итераций перед понижением температуры. Обычно разрешается провес- ти от 100 до 1 000 итераций; ♦ критерии приемлемости. В типичном случае приемлем любой переход от состояния S, к состоянию S, + |. если C(S, 11) < C(S,), а также отрицательный переход, если где г— случайное число в диапазоне 0< r< 1. Константа к нормализует функцию стоимости, чтобы при начальной температуре принимались почти все переходы; ♦ критерии остановки. Если за последнюю итерацию (или несколько последних ите- раций) значение текущего решения не изменилось или не улучшилось, то поиск прекращается и выводится текущее решение. Создание правильного графика охлаждения фактически выполняется методом проб и ошибок в процессе подстановки разных значений констант и наблюдения за результа- тами. Я рекомендую начинать работу с методом имитации отжига с ознакомления с уже существующими его реализациями. В частности, мою реализацию этого метода можно найти на сайте http://www.algorist.com, а другие реализации — в разделе 13.5. Сравните профили выполнения всех трех рассмотренных эвристических алгоритмов. Облако точек решений методом произвольной выборки существенно хуже, чем множе- ство решений, полученных другими эвристическими методами. Очевидно, что полосы решений, полученные методом восхождения по выпуклой поверхности, намного лучше. Но самым лучшим изо всех является профиль выполнения алгоритма имитации отжи- га, показанный на рис. 7.10. Все три прогона алгоритма имитации отжига дают намного лучшие решения, чем са- мое лучшее решение метода восхождения по выпуклой поверхности. Более того, чтобы получить максимальное улучшение результата, требуется сравнительно небольшое ко- личество итераций, что видно по трем резким переходам к оптимальному решению. При том же количестве выборок, что и для других методов (1 500 000), метод имита- ции отжига выдал решение стоимостью 36 617,4 мили, что всего лишь на 9,2% больше оптимального. Если вы готовы подождать несколько минут, то можно получить еще более качественное решение. В частности, выполнение 5 000 000 итераций понижает стоимость решения до 34 254,9 мили, что на 2,2% выше оптимального. Но дальнейшее увеличение количества итераций до 10 000 000 не принесло последующего улучшения результатов. При умелом обращении самые лучшие эвристические алгоритмы, настроенные под решение задачи коммивояжера, могут выдать чуть более удачное решение, чем метод
Гпава 7. Комбинаторный поиск и эвристические методы 277 Производительность метода имитации отжига для решения задачи коммивояжера 200000 1В0000 160000 140000 го & 120000 л го 5 100000 й 80000 60000 40000 20000 о О 200000 400000 600000 800000 1е+06 1.2е+06 1.4е+06 1 6е+06 Количество итераций Рис. 7.10. Соотношения между временем поиска решений и их качеством для решения задачи коммивояжера методом имитации отжига имитации отжига. Однако метод имитации отжига работает превосходно, не требуя специальной настройки. Я предпочитаю пользоваться им. Реализация Реализация эвристического алгоритма имитации отжига показана в листинге 7.18. Ристин; 7.18. Реализация метода имитации отжига anneal(tsp_instance *t, tsp_solution *s) int i1, 12; /* Пара элементов для обмена местами*/ int i,j; /* Счетчики */ double temperature; /* Текущая температура системы */ double current_value; /* Значение текущего состояния */ double start_value; /* Значение в начале цикла */ double delta; /* Значение после обмена */ double merit, flip; /* Условия принятия обмена*/ double exponent; /* Показатель степени для функции энерг. состояния*/ double random_float(); double solution_cost(), transition(); temperature = INITIAL_TEMPERATURE; initialize_solution(t->n, s) ; current_value = solution_cost(s, t); for (i=l; i<=COOLING_STEPS; i++) { temperature *= COOLING_FRACTION; start_value = current_value;
278 Часть I. Практическая разработка алгоритмов for (j=l; j<=STEPS_PER_TEMP; j++) { /* Выбираем индексы элементов для обмена местами *7 il = random_int(1, t->n); i2 = random_int(1, t->n,; flip = random_float(0,1); delta = transitions, t, il, 12); exponent = (-delta/current_value)/(K*temperature); merit = pow(E, exponent); if (delta < 0) /* Принять удачный результат */ current_value = current_value+delta; else { if (merit > flip) /* Принять неудачный результат */ current_value = current_value+delta; else /* Отклонить */ transition(s,t,il,i2) ; I } /* Восстанавливаем температуру в случае успеха */ if ((current_value-start_value) < 0.0) temperature = temperature/COOLING FRACTION; 7.5.4. Применение метода имитации отжига Теперь рассмотрим несколько примеров, демонстрирующих, как метод имитации от- жига можно использовать для решения реальных задач комбинаторного поиска. Задача максимального разреза В задаче максимального разреза требуется разделить вершины взвешенного графа G на множества f7i и И2 таким образом, чтобы максимизировать в каждом множестве вес (или количество) ребер с одной вершиной. Для графов, представляющих электронные схемы, максимальный разрез графа определяет наибольший поток данных, который может одновременно протекать в схеме. Задача максимального разреза является NP-.полной (см. раздел 16.6). Как можно сформулировать задачу максимального разреза для решения методом ими- тации отжига? Пространство решений состоит из всех 2" 1 возможных разбиений вер- шин. Мы получаем двойную экономию по всем подмножествам вершин, т. к. можно полагать, что вершина v( зафиксирована на левой стороне разбиения. Подмножество вершин, сопровождающее эту вершину, можно представить с помощью битового век- тора. Стоимостью решения является сумма весов разреза в текущей конфигурации. Механизм естественного перехода выбирает произвольным образом одну вершину и перемещает ее в другую часть разбиения, просто изменив на обратное значение соот- ветствующего бита в битовом векторе. Изменение в функции стоимости составит раз- ность между весом старых соседей вершины и весом ее новых соседей. Она вычисля- ется за время, пропорциональное степени вершины. На практике именно к такому простому и естественному моделированию следует стре- миться при поиске эвристического алгоритма.
Гпава 7. Комбинаторный поиск и эвристические методы 279 Независимое множество Независимым множеством графа G называется подмножество вершин 5". не содержа- щее ребер, у которых обе конечные точки являются членами X. Максимальным незави- симым множеством графа является такой наибольший пустой порожденный подграф. Задача поиска независимых множеств возникает в задачах рассеивания, связанных с календарным планированием и теорией шифрования (см раздел 16.2). Естественное пространство состояний для решения задачи методом имитации отжига включает в себя все 2" подмножества вершин, представленные в виде битового векто- ра. Так же, как и в случае с максимальным разрезом, простой механизм переходов до- бавляет или удаляет одну вершину из множества 5. Одной из наиболее естественных функций стоимости для подмножества X будет функ- ция, возвращающая 0, если X содержит ребро, и |Х|, в случае действительно независи- мого множества. Эта функция гарантирует, что мы непрерывно приближаемся к полу- чению независимого множества. Но это условие настолько жесткое, что существует большая вероятность остаться в пределах лишь небольшой части возможного про- странства поиска. Большую степень гибкости в доступе к пространству поиска и уско- рение работы функции стоимости можно получить, разрешив непустые графы на ран- них этапах охлаждения. На практике будет лучше использовать функцию стоимости наподобие С(Х) = [X] - /:тГ. где 2 является постоянной, Т представляет температуру, arn.s-— количество ребер в подграфе, порожденным X. Зависимость С(Х) от Т обеспе- чивает ускорение вытеснения ребер по мере охлаждения системы. Размещение компонентов на печатной плате Задача разработки печатных плат заключается в размещении на них должным образом компонентов, обычно интегральных схем. Заданными критериями компоновки могут быть минимизация площади или отношения длины платы к ее ширине, чтобы она по- мещалась в отведенное место, и минимизация длины соединяющих дорожек. Компо- новка печатных плат является типичным примером запутанной задачи оптимизации со многими критериями. Для решения таких задач идеально подходит метод имитации отжига. При формальной постановке задается коллекция прямоугольных модулей г\,.... г„ с соответствующими размерами h, * Кроме этого, для каждой пары модулей (г„ /;) задается количество соединяющих их дорожек w4. Требуется найти такое размещение прямоугольников, которое минимизирует площадь печатной платы и длину соединяю- щих дорожек, при условии, что прямоугольники не могут накладываться, частично или полностью, друг на друга. Пространство состояний для данной задачи должно описывать расположение каждого прямоугольника. Чтобы сделать задачу дискретной, можно наложить ограничение, при котором прямоугольники могут размещаться только на вершинах решетки целых чи- сел Подходящим механизмом переходов может быть перемещение одного прямо- угольника в другое место или обмен местами двух прямоугольников. Естественной функцией стоимости будет Л II С(Х)= ющаоь (S'.™onia ширина} \ ("^аоражкаина юж сине (=| /=|
280 Часть I. Практическая разработка алгоритмов где л,П(,и Лшюжгаие являются постоянными, представляющими влияние этих факторов на функцию стоимости. По всей видимости, значение должно быть обратно пропорционально температуре, чтобы после предварительного размещения прямоугольников их позиции корректировались во избежание наложения прямоуголь- ников друг на друга. Подведение итогов Метод имитации отжига является простым, но эффективным методом получения хотя и не оптимальных, но достаточно хороших решений задач комбинаторного поиска 7.6. История из жизни. Только это не радио — Считайте, что это радио. — мой собеседник тихо рассмеялся. — Только это не радио. Меня срочно доставили на корпоративном реактивном самолете в научно-иссле- довательский центр одной большой компании, расположенный где-то к востоку от штата Калифорния. Они были настолько озабочены сохранением секретности, что мне даже никогда ни пришлось увидеть разрабатываемое ими устройство. Тем не менее, они отличнейшим образом абстрагировали задачу. Задача была связана с технологией производства, называемой селективной сборкой. Эли Уитни (Eli Whitney) считается изобретателем системы взаимозаменяемых компо- нентов. которая сыграла важную роль в промышленной революции. Принцип взаимо- заменяемости состоит в изготовлении деталей механизма с определенной степенью точности, называемой допуском на обработку, что позволяет использовать любую комбинацию деталей для сборки механизма. Таким образом существенно ускоряется процесс производства, т. к. детали теперь можно просто складывать вместе, а не под- гонять каждую индивидуально с помощью напильника. Взаимозаменяемость также значительно облегчила ремонт изделий, позволяя выполнять замену вышедших из строя деталей без особых трудностей. Все это было очень хорошо, за исключением одного обстоятельства. В частности, если размеры детали слегка не соответствовали допускам, то такую де- таль нельзя было использовать. Однако нашелбя сообразительный сотрудник, который предложил использовать детали с нарушениями в допусках совместно с деталями, из- готовленными точно по требуемому размеру. Таким образом, использование плохих деталей совместно с хорошими дает приемлемый результат. В этом и заключается суть селективной сборки. — Каждый прибор состоит из п деталей разных типов, — продолжал представитель компании. — Для /-го типа детали (скажем, фланцевой прокладки) имеется s, экземп- ляров этой детали, для каждой из которых указана степень отклонения от эталонного размера. Нам необходимо подобрать детали друг к другу таким образом, чтобы полу- чить максимально возможное количество работающих приборов. — Предположим, каждый прибор состоит из трех деталей, а сумма отклонений от нор- мы всех деталей работающего прибора не должна превышать 50. Умело распределяя хорошие и плохие детали в каждом устройстве, мы можем использовать все детали и получить три работающих прибора. Данная ситуация иллюстрируется на рис. 7.11.
Гпава 7 Комбинаторный поиск и эвристические методы 281 Рис. 7.11. Распределение деталей между тремя приборами, чтобы сумма отклонений для каждого не превышала 50 Я немного поразмышлял над задачей. Проще всего было бы использовать для сборки каждого прибора самые лучшие оставшиеся детали каждого типа, повторяя процесс до тех пор, пока собранный таким способом прибор не будет работать. Но таким образом мы получаем небольшое количество приборов с большим разбросом качества, в то время, как компании требуется максимальное количество приборов высокого качества. Целью было совместить хорошие и плохие детали друг с другом таким образом, чтобы общее качество собранного устройства было бы приемлемым. Определенно, задача выглядела как паросочетание в графах (см. раздел 15.6). Допустим, мы создадим граф, в котором вершины представляю!' экземпляры деталей, и соединим ребрами все пары экземпляров с допустимой общей погрешностью в размерах. В паросочетании в графах мы ищем наибольшее количество ребер, в котором никакие два ребра не опираются на одну и ту же вершину. Это аналогично максимальному количеству сборок из двух де- талей. которые можно получить из данного набора деталей. — Вашу задачу можно решить методом паросочетания,— объявил я.— При условии, что все приборы состоят из двух деталей. Мое заявление было встречено общим молчанием, за которым последовал дружный смех. — Все знают, что в приборе более двух деталей. Таким образом, данный алгоритмический подход был отвергнут. Расширение задачи до сопоставления более чем двух деталей превращало ее в задачу паросочетания в гиперграфах1, которая является NP-полной. Более того, только само время построения графа может быть экспоненциально зависимым от количества типов деталей, т. к. каж- дое возможное гиперребро (сборку) нужно будет создавать явным образом. Пришлось начинать разработку алгоритма сначала. Итак, нужно собрать детали таким образом, чтобы общая сумма отклонений от нормы не превышала определенного до- пустимого значения. Это выглядело, как задача упаковки контейнеров. В задаче этого типа (см. раздел /7.9) требуется разложить набор элементов разного размера в наи- меньшее количество контейнеров с ограниченной емкостью к. В данной ситуации кон- гейнеры представляли сборки, каждая из которых допускала общую сумму отклонений Гиперграф содержит ребра, каждое из которых может иметь более двух вершин. Их можно предста- вить в виде общих коллекций вершин.
282 Часть I. Практическая разработка алгоритмов от нормы < к. Пакуемые элементы представляли отдельные детали, размер которых отражал качество изготовления. Однако полученная задача не является чистой задачей разложения по контейнерам в силу неоднотипности деталей. Данное приложение накладывало ограничение на до- пустимое содержимое контейнеров. Требование создать наибольшее количество при- боров означало, что нужно было найти метод разложения элементов, который макси- мизирует количество контейнеров, содержащих ровно одну деталь каждого типа. Задача разложения по контейнерам является NP-полной, однако естественно подходит для решения методом эвристического поиска. Пространство решений состоит из вари- антов разложения деталей по контейнерам. Для каждого контейнера мы сначала выби- раем по произвольной детали каждого типа, чтобы получить начальную конфигурацию для поиска. Потом мы выполняем локальный поиск, перемещая детали из одного контейнера в другой. Можно было бы перемещать по одной детали, но будет более эффективным обменивать детали определенного типа между двумя произвольно выбранными кон- тейнерами. При таком обмене оба контейнера остаются полностью собранными прибо- рами, предположительно с более приемлемым значением общего отклонения от нор- мы. Для операции обмена требовалось три произвольных целых числа— одно для вы- бора типа детали (в диапазоне от 1 до т) и два для выбора контейнеров, межд\ которыми нужно выполнять обмен (например, в диапазоне от 1 до Ь). Ключевым решением был выбор функции стоимости. Для каждой сборки был установ- лен жесткий общий предел отклонений от нормы, равный к. Но что может быть наи- лучшим способом оценки нескольких сборок? Можно было бы в качестве общей оцен- ки просто возвращать количество приемлемых сборок— целое число в диапазоне от 1 до Ь. Хотя эта величина и была той, которую мы хотели оптимизировать, она не была чувствительна к продвижению в направлении правильного решения. Допустим, что в результате одного из обменов деталей между сборками допустимое отклонение одной из неработающих сборок стало намного ближе к пределу отклонений к для сборки Это была бы значительно более удачная отправная точка для дальнейшего поиска решения, чем первоначальная. В итоге, я остановился наследующей функции стоимости. Каждой работающей сборке я давал оценку в 1 пункт, а каждой неработающей— значительно меньшую оценку, в зависимости от того, насколько близка она была к предельному значению к. Оценка неработающей сборки понижалась экспоненциально, в зависимости оттого, насколько ее общее отклонение от нормы превышало к. Таким образом, оптимизатор будет пы- таться максимизировать количество работающих приборов, после чего станет подго- нять очередную сборку как можно ближе к пределу. Я воплотил этот алгоритм в коде, после чего обработал с его помощью набор деталей, взятых непосредственно из производственного цеха. Оказалось, что данные приборы состоят из 8 деталей важных типов. Детали некоторых типов были более дорогими, чем другие, поэтому для них было меньше рассматриваемых вариантов. Для детали с наименьшими разрешенными отклонениями было предоставлено только восемь эк- земпляров, поэтому из данного набора деталей можно было собрать только восемь приборов.
Гпава 7. Комбинаторный поиск и эвристические методы 283 Я наблюдал за работой программы имитации отжига. Первые четыре сборки она выда- ла сразу же без особых усилий, и лишь после этого поиск немного замедлился и сборки 5 и 6 были предоставлены с небольшой задержкой. Потом наступила пауза, после ко- торой была выдана седьмая сборка. Но. несмотря на все свои усилия, программа не смогла сложить восемь работающих приборов, прежде чем мне надоело наблюдать за ее действиями. Я позвонил в компанию и хотел признаться в поражении, но они и слышать ничего об этом не хотели. Оказалось, что самостоятельно они смогли получить только шесть ра- ботающих приборов, так что мой результат был значительным прогрессом! 7.7. История из жизни. Отжиг массивов Вразделе 3.9 рассказывалось, как мы использовали структуры данных высшего уровня для эмулирования нового метода секвенирования ДНК. Для нашего метода, интерак- тивного секвенирования методом гибридизации, нужно было создавать по требованию массивы специфичных олигонуклеотидов. Наш метод заинтересовал одного биохимика из Оксфордского университета и, что бо- лее важно, в его лаборатории было необходимое оборудование для испытания этого метода. Автоматическая система подачи реактивов Southern Array Maker (производства компании Beckman Instruments) выкладывает дискретные последовательности олиго- нуклеотидов в виде шестидесяти четырех параллельных рядов на полипропиленовой подложке. Устройство создает массивы, добавляя отдельные элементы в каждую ячей- ку вдоль определенных рядов и столбцов массива. В табл. 7.2 показан пример создания массива всех 24 = 16 пуриновых (Л или G) 4-меров путем создания префиксов вдоль рядов и суффиксов вдоль столбцов. Таблица 7.2. Массив всех пуриновых 4-меров Префикс Суффикс АА AG GA GG АА АААА AAAG AAGA AAGG AG AGAA AGAG AGGA AGGG GA GAAA GAAG GAGA GAGG GG GGAA GGAG GGGA GGGG Эта технология предоставляла идеальную среду для проверки возможности примене- ния интерактивного секвенирования методом гибридизации в лаборатории, т. к. она позволяла создавать с помощью компьютера широкий набор массивов олигонуклео- тидов. Вот мы и должны были предоставить соответствующее программное обеспечение. Для создания сложных массивов требовалось решить трудную комбинаторную задачу. В качестве входа задавался набор п строк (представляющих олигонуклеотиды), из ко-
284 Часть I. Практическая разработка алгоритмов торых нужно было создать массив размером т * т (где т — 64 на устройстве Southern Array Maker). Нам нужно было запрограммировать создание рядов и столбцов, чтобы реализовать множество строк 5". Мы доказали, что задача разработки плотных массивов является NP-полной, но это не имело большого значения, т. к. решать ее в любом слу- чае пришлось мне и моему студенту Рики Брэдли (Ricky Bradley). — Нам нужно будет использовать эвристический метод, — сказал я ему. — Как смоде- лировать эту задачу? — Каждую строку можно разбить на составляющие ее префиксные и суффиксные па- ры. Например, строку АСС можно создать четырьмя разными способами: из префикса" (пустая строка) и суффикса АСС, префикса А и суффикса СС, префикса АС и суффик- са С или префикса ЛСС и суффикса". Нам нужно найти наименьший набор префиксов и суффиксов, которые совместно позволяют получить все данные строки,— сказал Рики. — Хорошо. Это дает нам естественное представление для метода имитации отжига. Пространство состояний будет содержать все возможные подмножества префиксов и суффиксов. Естественные переходы между состояниями могут включать вставку или удаление строк из наших подмножеств или обмен пар местами. — Какую функцию стоимости можно использовать? — спросил он. — Нам требуется массив как можно меньшего размера, который охватывает все стро- ки. Попробуем взять максимальное количество рядов (префиксов) или столбцов (суф- фиксов), используемых в нашем массиве, плюс те строки из множества 5". которые еще не были задействованы. Рики отправился на свое рабочее место и реализовал программу имитации отжига в соответствии с этими принципами. Программа выводила текущее состояние решения после каждого принятия перехода, и за ее работой было интересно наблюдать. Она бы- стро отсеяла ненужные префиксы и суффиксы и размер массива начал стремительно сокращаться. Но после нескольких сотен итераций прогресс начал замедляться. При переходе программа удаляла ненужный суффикс, выполняла вычисления в течение некоторого времени, а потом добавляла другой суффикс. После нескольких тысяч ите- раций не наблюдалось никакого реального улучшения. — Кажется, что программа не может определить, когда она получает правильное ре- шение,— предположил я.— Функция стоимости оценивает только минимизацию большей стороны массива. Давай добавим в нее выражения для оценки действий с дру- гой стороной. Рики внес соответствующие изменения в функцию стоимости, и мы снова запустили программу. На этот раз она уверенно обрабатывала и другую сторону. Более того, на- ши массивы стали выглядеть как тонкие прямоугольники, а не квадраты. — Ладно. Давай добавим в функцию стоимости еще одно выражение, чтобы массивы стали квадратными. Рики опять изменил программу. На этот раз массивы на выходе имели правильную форму и поиск двигался в нужном направлении. Но происходило это по-прежнему медленно.
Гпава 7 Комбинаторный поиск и эвристические методы 285 — Многие операции вставки затрагивают недостаточное количество строк. Может быть, нам следует сделать уклон в пользу случайной выборки, чтобы важные префиксы и суффиксы выбирались чаще? Рики внес очередные изменения. Теперь программа работала быстрее, но все равно иногда возникали паузы. Мы изменили график охлаждения. Результаты улучшились, но были ли они действительно хорошими? Не зная нижнего оптимального предела, нельзя было сказать, насколько удачным является наше решение. Мы продолжали на- страивать разные аспекты нашей программы, пока все дальнейшие модификации не перестали приводить к улучшению. Конечная версия программы улучшала первоначальный массив, используя следующие операции: ♦ swap— обменять префикс/суффикс в массиве с префиксом/суффиксом вне массива; ♦ add— добавить в массив случайный префикс/суффикс; ♦ delete — удалить из массива случайный префикс/суффикс: ♦ useful add—добавить в массив префикс/суффикс с наибольшей пригодностью; ♦ useful delete — удалить из массива префикс/суффикс с наименьшей пригодностью; ♦ string add— выбрать произвольную строку вне массива и добавить наиболее при- годный префикс и/или суффикс, чтобы покрыть эту строку. Мы использовали стандартный график охлаждения с экспоненциально убывающей температурой (зависящей от размера задачи) и зависящим от температуры критерием Больцмана, чтобы выбирать состояния, имеющие более высокую стоимость. Наша ко- нечная функция стоимости определялась таким образом: _ . , (max-min)2 cost = 2 х max + min + --------4(5/r(,„o/ - str,,,} где max— максимальный размер чипа, min — минимальный размер чипа, strMai = |5|, as(r,„— количество строк из множества5, находящихся в настоящее время в чипе. Насколько хороший результат мы получили? На рис. 7.12 показано четыре этапа сжа- тия специального массива, состоящего из 5 716 однозначных 7-меров вируса иммуно- дефицита человека (ВИЧ). Рис. 7.12. Сжатие массива ВИЧ метолом имитации отжига после 0. 500. 1 000 и 5 750 итераций Черные пикселы массива представляют первое появление 7-мера ВИЧ. Конечный размер чипа— 130х 132, что является прогрессом по сравнению с первоначальным размером 192x192. Чтобы завершить оптимизацию, программе потребовалось около
286 Часть I. Практическая разработка алгоритмов 15 минут, что было приемлемо для данного приложения. Насколько хорошим был этот результат? Так как имитация отжига— всего лишь эвристический метод, то мы, по сути, не знаем, насколько близким к оптимальному является наше решение. Лично я думаю, что результат довольно хороший, но не могу быть в этом полностью уверен- ным. Метод имитации отжига является хорошим инструментом для решения сложных задач оптимизации. Однако, если вы хотите получить отличные результаты, будьте готовы потратить больше времени на дополнительную настройку и оптимизацию про- граммы. чем ушло на создание ее первоначального варианта. Это тяжелая работа, но зачастую у вас нет иного выхода. 7.8. Другие эвристические методы поиска Для поиска хороших решений задач комбинаторной оптимизации было предложено несколько эвристических методов. Подобно методу имитации отжига, многие эвристи- ческие методы основаны на аналогии с реальными физическими процессами. Одними из популярных методов являются генетические алгоритмы, нейронные сети и алго- ритм муравейника. Интуитивные представления, лежащие в основе этих методов, привлекательны своей понятностью, но скептики считают такие подходы шаманством, основанным на краси- вых аналогиях с природными явлениями, а не на результатах исследования задач, дос- тигнутых с помощью других методов. Вопрос заключается не в том, возможно ли, приложив достаточные усилия, получить приемлемые решения задач с помощью этих методов. Ясно, что возможно. Но настоя- щий вопрос состоит в том. позволяют ли эти методы получить лучшие решения при меньшей сложности реализации, чем другие рассмотренные методы. В общем, я так не думаю. Но в духе непредвзятого исследования мы кратко рассмот- рим генетические алгоритмы, которые пользуются наибольшей популярностью. Более подробную информацию см. в разделе "Замечания к главе". Генетические алгоритмы Идея генетических алгоритмов возникла как отражение теории эволюции и естествен- ного отбора. Посредством естественного отбора организмы приспосабливаются к вы- живанию в определенной среде. В генетическом коде организма происходят случайные мутации, которые передаются его потомкам. Если данная мутация окажется полезной то тогда у потомков будет больше шансов выжить. Если же мутация вредная, то веро- ятность выживания потомков и передачи этого качества по наследству уменьшается. Генетические алгоритмы содержат "популяцию" кандидатов на решение данной зада- чи. Из этой популяции случайным образом выбираются элементы, которые "размно- жаются" так, что в потомке комбинируются свойства двух родителей. Вероятность вы- бора элемента для размножения основывается на его "физической форме", которой, по существу, является стоимость предоставляемого им решения. Элементы, имеющие плохую физическую форму, вымирают, уступая место потомкам элементов, пребы- вающих в более хорошей форме.
Гпава 7. Комбинаторный поиск и эвристические методы 287 Идея генетических алгоритмов чрезвычайно привлекательна. Но эти алгоритмы не мо- гут работать так эффективно над практическими задачами комбинаторной оптимиза- ции. как метод имитации отжига. На то есть две причины. Во-первых, моделирование приложений посредством генетических операторов, таких как мутация и пересечение набитовых строках, является в высшей степени неестественным. Псевдобиология вно- сит дополнительный уровень сложности в решаемую задачу. Во-вторых, решение нетривиальных задач посредством генетических алгоритмов занимает очень много времени. Операции пересечения и мутации обычно не используют структуры данных, специфичных для данной задачи, вследствие чего большинство переходов дает низко- качественные решения и процесс поиска наилучшего решения протекает медленно. В этом случае аналогия с эволюцией (которой для осуществления важных изменений требуются миллионы лет) оказывается уместной. Чтобы отговорить вас от использования генетических алгоритмов в своих прило- жениях. мы не будем продолжать рассматривать их. Но если вы все же очень хотите поэкспериментировать с ними, то подсказки по их реализации можно найти в разделе 13.5. Подведение итогов Я лично никогда не сталкивался ни с одной задачей, для решения которой генетические алгоритмы оказались бы самым подходящим средством. Более того, я никогда не встречал никаких результатов вычислений, полученных посредством генетических алгоритмов, ко- торые бы производили на меня положительное впечатление. Пользуйтесь методом имита- ции отжига для своих экспериментов с эвристическим поиском. 7.9. Параллельные алгоритмы Ум хорошо, а два лучше, а, более обобщенно, п умов лучше, чем п- 1 умов. Такой подход кажется легким способом решения трудных задач. В самом деле, для решения некоторых задач параллельные алгоритмы являются наиболее эффективным средст- вом. Например, для реализации реалистичной анимации графические приложения вы- сокого разрешения реального времени должны выводить около тридцати кадров в се- кунду. Единственным способом справиться с такой задачей может быть выделение от- дельного процессора для обработки каждого кадра или для обработки одной части кадра. Большие системы линейных уравнений для научных целей регулярно решаются параллельным методом. Но параллельные алгоритмы имеют несколько скрытых недостатков, о которых нужно знать. ♦ Потенциальный выигрыш часто невелик. Допустим, что у вас имеется исключи- тельный доступ к двадцати процессорам. Теоретически, используя все эти процес- соры, можно в двадцать раз повысить скорость работы самой быстрой последова- тельной программы. Это, конечно же, прекрасно, но, возможно, производитель- ность можно повысить, использовав более удачный последовательный алгоритм. Время, требуемое на переработку кода для исполнения на параллельных процессо- рах, возможно, лучше потратить на улучшение последовательной версии програм-
288 Часть I. Практическая разработка алгоритмов мы. Также нужно иметь в виду, что средства отладки и повышения производитель- ности кода, такие как профилировщики, лучше работают на обычных компьютерах, чем на многопроцессорных. ♦ Простое ускорение работы еще ничего не означает. Допустим, что ваша парал- лельная программа работает в 20 раз быстрее на 20-процессорной машине, чем на компьютере с одним процессором. Это просто замечательно, не так ли? Конечно же, если вы всегда получаете линейное повышение скорости и имеете доступ к сколь угодно большому количеству процессоров, то со временем вы перекроете ре- зультаты любого последовательного алгоритма. Но тщательно разработанный по- следовательный алгоритм часто может оказаться эффективнее параллелизируемого кода, выполняемого на типичном многопроцессорном компьютере. Скорее всего, причиной низкой производительности вашей параллельной программы на однопро- цессорной машине является недостаточно хороший последовательный алгоритм. Поэтому измерение ускорения работы такой программы при использовании не- скольких процессоров является некорректной оценкой преимуществ параллелизма. Классическим примером такой ситуации является минимаксный алгоритм поиска в дереве игры, применяющийся в компьютерных программах игры в шахматы. Поиск методом исчерпывающего перебора в этом дереве удивительно легко поддается па- раллелизации: просто для каждого поддерева выделяется отдельный процессор. Но при таком подходе большая часть работы этих процессоров выполняется впустую, т. к. одни и те же позиции исследуются на нескольких процессорах. Использование же более интеллектуального алгоритма альфа-бета-отсечений может с легкостью уменьшить объем работы на 99,99%, по сравнению с чем любые достоинства поис ка методом исчерпывающего перебора на параллельных процессорах выглядят ни- чтожными. Алгоритм альфа-бета-отсечений можно распараллелить, но это задача не из легких. Кроме этого, увеличение количества используемых процессоров дает не- ожиданно малое повышение производительности. ♦ Параллельные алгоритмы плохо поддаются отладке. Если только вашу задачу не- возможно разбить на несколько независимых подзадач, то при ее исполнении на не- скольких процессорах разные процессоры должны взаимодействовать друг с дру- гом. чтобы выдать правильный конечный результат. К сожалению, вследствие неде- терминистического характера этого взаимодействия параллельные программы трудно отлаживать. Пожалуй, самым лучшим примером является шахматный су- перкомпьютер Deep Blue. Хотя последняя версия этого компьютера в конце концов победила чемпиона мира по шахматам Гарри Каспарова, в предыдущих соревнова- ниях машина терпела поражения из-за программных ошибок, по большому счету вызванных обширным параллелизмом. Я рекомендую рассматривать параллельные вычисления только в тех случаях, когда все другие попытки решить задачу последовательным способом оказываются слишком медленными. Но даже в таких случаях я бы ограничился использованием алгоритмов, которые параллелизируют задачу, разбивая ее на несколько подзадач, для исполнения которых разные процессоры не должны общаться друг с другом, кроме как для объ- единения конечных результатов. Такой примитивный параллелизм может быть доста- точно простым как для реализации, так и для отладки, т. к. его конечным результатом.
Гпава 7. Комбинаторный поиск и эвристические методы 289 по сути, является хорошая последовательная реализация. Но даже здесь есть свои под- водные камни. 7.10. История из жизни. "Торопиться в никуда" В разделе 2.8 я описал наши усилия по созданию быстрой программы для исследова- ния проблемы Уоринга. Код был достаточно быстрым и мог решить задачу за несколь- ко недель, работая в фоновом режиме на настольном компьютере. Но моему коллеге этот вариант не понравился. — Давайте запустим его на многопроцессорном компьютере,— предложил он.— В конце концов, в этом коде внешний цикл выполняет одни и те же вычисления для каждого целого числа от 1 до 1 000 000 000. Я могу разбить этот диапазон чисел на одинаковые интервалы и обработать каждый из них на отдельном процессоре. Увиди- те, как это будет легко. Он приступил к работе по выполнению нашей программы на компьютере Intel 1PSC- 860 с 32 узлами, каждый из которых имел свои собственные 16 Мбайт памяти — (очень мощная машина для того времени). Но в течение нескольких следующих недель я постоянно получал от него известия о проблемах. Например: ♦ программа работает прекрасно, но только прошлой ночью один процессор вышел из строя; ♦ все шло хорошо, но компьютер случайно перезагрузили, так что мы потеряли все результаты; ♦ согласно корпоративным правилам никто ни при каких обстоятельствах не может использовать все процессоры более чем 13 часов. Но, в конце концов, обстоятельства сложились благоприятно. Он дождался стабильной работы компьютера, захватил 16 процессоров (половину компьютера), разбил целые числа от 1 до 1 000 000 000 на 16 одинаковых диапазонов и запустил вычисление каж- дого диапазона на отдельном процессоре. Следующий день он провел, отбивая атаки разъяренных пользователей, которые не могли попасть на компьютер из-за его про- граммы. Как только первый процессор закончил обработку своего диапазона (числа от 1 до 62 500 000), он объявил негодующей толпе, что скоро и остальные процессоры закончат свою работу. Но этого не произошло. Наш коллега не принял во внимание то обстоятельство, что время для обработки каждого целого числа увеличивалось пропорционально увеличе- нию чисел. В конце концов, проверка представимости числа 1 000 000 000 в виде сум- мы трех пирамидальных чисел требует больше времени, чем такая же проверка для числа 100. Таким образом, последующие процессоры сообщали об окончании работы через все более длительные интервалы времени. Из-за особенностей устройства супер- компьютера освободившиеся процессоры нельзя было задействовать снова, пока не была завершена вся работа. В конце концов, половина машины и большинство ее поль- зователей оказались в заложниках у одного, последнего процессора. Какие выводы можно сделать из этой истории? Если вы собираетесь распараллелить вычисления, позаботьтесь о том, чтобы должным образом распределить работу между 10 Зак. 3741
290 Часть I. Практическая разработка алгоритмов процессорами. В приведенном примере правильный баланс нагрузки, достигнутый по- средством несложных вычислений или с помощью алгоритма разбиения, рассматри- ваемого в разделе 8.5, мог бы значительно сократить время исполнения задачи, а также время, в течение которого нашему коллеге пришлось выносить нападки других пользо- вателей. Замечания к главе Обсуждение перебора с возвратом в значительной степени основано на моей книге "Programming Challenges" [SR03]. В частности, рассмотренная здесь процедура перебо- ра с возвратом является обобщенной версией процедуры, представленной в главе 8 этой книги. В данной главе также имеется мое решение знаменитой задачи о восьми ферзях, где требуется найти все расположения на шахматной доске восьми ферзей, ни один из которых не находится под ударом любого другого. Метод имитации отжига был первоначально рассмотрен в статье [K.GV83J. в которой также обсуждалось его применение для решения задачи размещения на печатных пла- тах сверхбольших интегральных схем. Приложения в разделе 7.5.4 основаны на мате- риале из книги [АК.89]. В представленных здесь эвристических решениях задачи коммивояжера в качестве ло- кальной операции используется обмен вершин. В действительности, намного более мощной операцией является обмен ребер. При каждой операции обмениваются макси- мум два ребра в маршруте, по сравнению с четырьмя ребрами при обмене местами вершин. Это повышает вероятность локального улучшения. Но чтобы эффективно поддерживать порядок полученного маршрута, требуются более сложные структуры данных, которые рассматриваются в книге [FJMO93]. Разные эвристические методы поиска подробно представлены в книге [AL97], которую я рекомендую всем, кто хочет углубить свои знания в области эвристических методов поиска. В этой книге описан алгоритм поиска с запретами (tabu search), который явля- ется разновидностью метода имитации отжига, использующего дополнительные струк- туры данных, чтобы избежать переходов в недавно рассмотренные состояния. Алго- ритм муравейника рассматривается в книге [DT04]. Более доброжелательную точку зрения на генетические и им подобные алгоритмы, чем та, что представлена в этой гла- ве, см. в книге [MFOOJ. Подробную информацию по комбинаторному поиску оптимальных позиций на шах- матной доске можно найти в статье [RHS89], Наша работа по использованию метода имитации отжига для сжатия массивов ДНК была представлена в журнале [BS97]. До- полнительную информацию по селективной сборке см. в журнале [Pug86] и журнале [CGJ98]. Наша работа по параллельным вычислениям с пирамидальными числами бы- ла представлена в журнале [DY94]. 7.11. Упражнения Перебор с возвратом 1. [3] Беспорядком (derangement) называется такая перестановка р множества {1, ..., и}, в которой ни один элемент на находится на своем правильном месте, т. е., р, * z для всех
Гпава 7. Комбинаторный поиск и эвристические методы 291 1 < / < п. Разработайте программу перебора с возвратом с применением отсечений для создания всех беспорядочных перестановок л элементов. 2. [4] Мультимножества (multiset) могут содержать повторяющиеся элементы Таким об- разом, мультимножество из п элементов может иметь меньше, чем п\ разных перестано- вок. Например, мультимножество {I, I, 2, 2} имеет только шесть разных перестановок: {1, 1, 2, 2}, {1,2, 1,2}, {I, 2, 2, I}, {2, 1, 1. 2}, {2, 1,2, 1} и {2, 2, 1, 1}. Разработайте и реализуйте алгоритм для создания всех перестановок мультимножества. 3. [5] Разработайте и реализуйте алгоритм для проверки, являются ли два графа изоморф- ными по отношению друг к другу. Задача изоморфизма графов рассматривается в разде- w 16.9. При правильном отсечении можно без проблем выполнять проверку графов, со- держащих сотни вершин. 4. [5] Анаграммой называется перестановка букв слова или фразы таким образом, чтобы получилось другое слово или фраза. Иногда результаты таких перестановок поражают Например, фраза MANY VOTED BUSH RETIRED (многие проголосовали за отставку Буша) является анаграммой фразы TUESDAY NOVEMBER, THIRD (вторник, третье но- ября)'. Разработайте и реализуйте алгоритм создания анаграмм, используя комбинатор- ный поиск и словарь. 5. [8] Разработайте и реализуйте алгоритм для решения задачи изоморфизма подграфов. Даны графы G и Н. Существует ли такой подграф Н' графа Н, для которого граф G явля- ется изоморфным графу //'? Как ваша программа работает в таких особых случаях изо- морфизма подграфов, как гамильтонов цикл, клика, независимое множество? 6. [8] В проекте по реконструкции автострады дано п(п- 1)/2 упорядоченных по длине рас- стояний. Задача заключается в том, чтобы найти положения п точек на линии, которые порождают эти расстояния. Например, расстояния {1, 2, 3, 4, 5, 6} можно определить, расположив вторую точку на расстоянии I единицы от первой, третью — на расстоянии 3 единиц от второй, а четвертую— на расстоянии 2 единиц от третьей. Разработайте и реализуйте алгоритм для вывода всех решений этой задачи. Чтобы минимизировать по- иск, используйте по мере возможности аддитивные ограничения. При правильном отсе- чении можно без проблем решать задачи, содержащие сотни узлов. Комбинаторная оптимизация Для каждой из приведенных далее задач реализуйте программу комбинаторного поиска для оптимального решения входного экземпляра небольшого размера и/или разработайте и реа- лизуйте эвристический алгоритм имитации отжига для получения приемлемых решений Дайте оценку практической работе вашей программы. 7. [5] Разработайте и реализуйте алгоритм для решения задачи минимизации ширины по- лос, рассматриваемой в разделе 13.2. 8. [5] Разработайте и реализуйте алгоритм для решения задачи максимальной приемлемо- сти, рассматриваемой в разделе 14.10. 9. [5] Разработайте и реализуйте алгоритм для решения задачи поиска максимальной клики, рассматриваемой в разделе 16.1. 1 День президентских выборов в США в 1992 г., когда действующий президент Буш проиграл Биллу Клин- тону. — Прим, иерее.
292 Часть I. Практическая разработка алгоритмов 10. [5] Разработайте и реализуйте алгоритм для решения задачи минимальной вершинной раскраски, рассматриваемой в разделе 16.7. 11. [5] Разработайте и реализуйте алгоритм для решения задачи реберной раскраски, рас- сматриваемой в разделе 16.8. 12. [5] Разработайте и реализуйте алгоритм для решения задачи поиска разрывающего множества вершин, рассматриваемой в разделе 16.11. 13. [5] Разработайте и реализуйте алгоритм для решения задачи покрытия множества, рас- сматриваемой в разделе 18.1. Задачи, предлагаемые на собеседовании 14. [4] Напишите функцию поиска всех перестановок букв в данной строке. 15. [4] Реализуйте алгоритм перечисления всех A-элементных подмножеств множества, со- держащего л элементов. 16. [5] Анаграммой называется перестановка букв слова или фразы таким образом, чтобы получилось другое слово или фраза. Например, анаграммой строки Steven Skiena явля- ется строка Vainest Knees. Предложите алгоритм для создания всех анаграмм данной строки. 17. [5] Каждая кнопка телефонного номеронабирателя содержит несколько букв. Напишите программу для генерирования всех возможных слов, которые можно получить из дан- ной последовательности нажатых кнопок (например, 145345). 18. [7] Имеется пустая комната и п человек снаружи. На каждом этапе можно впустить од- ного человека в комнату или выпустить одного человека из комнаты. Можете ли вы упорядочить последовательность из 2п этапов таким образом, чтобы каждая возможная комбинация людей возникла только один раз? 19. [4] Используя генератор случайных чисел, который генерирует случайные целые числа в диапазоне от 0 до 4 с одинаковой вероятностью, напишите генератор случайных чисел (rng07), который генерирует целые числа в диапазоне от 0 до 7 с одинаковой вероят- ностью. Укажите ожидаемое количество вызовов функции rng04 для каждого вызова функции rng07. Задачи по программированию Эти задачи доступны на сайтах http://vvvvvv.programming-challenges.com и http:// uva.onlinejudge.org. 1. Little Bishops. 110801/861. 2. 15-Puzzle Problem. 110802/10181. 3. Tug of War. 110805/10032. 4. Color Hash. 110807/704.
ГЛАВА 8 Динамическое программирование Наиболее трудными алгоритмическими задачами являются задачи оптимизации, в ко- торых требуется найти решение, максимизирующее или минимизирующее определен- ную функцию. Классическим примером задачи оптимизации является задача комми- вояжера. в которой требуется найти маршрут с минимальной стоимостью для посеще- ния всех вершин графа. Как мы видели в главе 1, для решения задачи коммивояжера можно легко предложить несколько алгоритмов, которые выдают кажущиеся удовле- творительными решения, но не всегда выдают маршрут с минимальной стоимостью. Для алгоритмов задач оптимизации требуется доказательство, что они всегда возвра- щают наилучшее возможное решение. "Жадные" алгоритмы, которые принимают наи- лучшее локальное решение на каждом шаге, обычно эффективны, но не гарантируют глобальное оптимальное решение. Алгоритмы поиска методом исчерпывающего пере- бора всегда выдают оптимальный результат, но обычно временная сложность таких алгоритмов чрезмерно высока. Динамическое программирование сочетает лучшие возможности обоих подходов. Этот метод предоставляет возможность разрабатывать алгоритмы специального назначения, которые систематически исследуют все возможности (таким образом гарантируя пра- вильность решения) и в то же самое время сохраняют ранее полученные промежуточ- ные результаты (таким образом обеспечивая эффективность работы). Сохранение по- следствий всех возможных решений и систематическое использование этой информа- ции минимизируют общий объем работы. Если вы понимаете суть динамического программирования, эта технология разработки алгоритмов, возможно, является для вас самой удобной для практического примене- ния. Я считаю, что алгоритмы динамического программирования часто легче разрабо- тать заново, чем искать их готовую реализацию в какой-либо книге. Однако, пока вы не понимаете динамическое программирование, оно кажется вам каким-то шаман- ством. Динамическое программирование— это технология для эффективной реализации ре- курсивных алгоритмов посредством сохранения промежуточных результатов. Секрет ее применения заключается в определении, выдает ли простой рекурсивный алгоритм одинаковые результаты для одинаковых подзадач. Если выдает, то вместо повторения вычислений ответ каждой подзадачи можно сохранять в таблице для использования в дальнейшем, что дает возможность получить эффективный алгоритм. Начинаем разра- ботку с определения и отладки рекурсивного алгоритма. Только добившись правиль- ной работы нашего рекурсивного алгоритма, мы переходим к поиску мер по ускоре- нию его работы, сохраняя результаты в матрице. Динамическое программирование обычно является подходящим методом решения за- дач оптимизации в случае комбинаторных объектов, которые имеют естественный по-
294 Часть I. Практическая разработка алгоритмов рядок организации компонентов слева направо. К таким объектам относятся строки символов, корневые деревья, многоугольники, а также последовательности целых чи- сел. Изучение динамического программирования лучше всего начинать с исследования готовых примеров. Для демонстрации практической пользы от динамического про- граммирования в этой главе приводится несколько историй из жизни, в которых оно сыграло решающую роль в решении поставленной задачи. 8.1. Кэширование и вычисления По сути, динамическое программирование является компромиссом, при котором по- вышенный расход памяти компенсируется экономией времени. Многократное вычис- ление некого значения безвредно само по себе, пока затраченное на это время не ока- зывает отрицательного влияния на производительность. В таком случае для повыше- ния производительности будет разумнее не повторять вычисления, а сохранять результаты первоначальных вычислений и потом обращаться к ним в случае надоб- ности. Экономия времени исполнения за счет повышенного расхода памяти в динамическом программировании лучше всего проявляется при рассмотрении рекуррентных соотно- шений, таких как числа Фибоначчи. В последующих разделах мы рассмотрим три про- граммы для вычисления этих последовательностей. 8.1.1. Генерирование чисел Фибоначчи методом рекурсии Числа Фибоначчи были впервые исследованы в XIII веке итальянским математиком Фибоначчи для моделирования размножения кроликов. Фибоначчи предположил, что количество пар кроликов, рождающихся в данный год, равно сумме пар кроликов, ро- жденных в два предыдущие годы, начиная с одной пары кроликов в первом году. Для подсчета количества кроликов, рожденных в n-м году, он определил следующее рекур- рентное соотношение: F„ = F„-i + F„_2 для которого Fq = 0 и Д| = I. Таким образом. F2 = 1, /л = 2. Ряд продолжается в виде последовательности {3, 5, 8, 13, 21, 34, 55, 89, 144,...}. Как потом выяснилось, формула Фибоначчи не очень хорошо подходит для описания размножения кроликов, но зато оказалось, что она обладает множеством интересных свойств. Так как числа Фибонач- чи определяются рекурсивной формулой, то для вычисления и-го числа Фибоначчи легко создать рекурсивную программу. Пример такого рекурсивного алгоритма на языке С приводится в листинге 8.1. ... ........................................................... г—--..........,.>., ..,, Листинг 8.1. Рекурсивная функция для вычисления л-го числа Фибоначчи uirtr/iiwHb . , ... ................................................ long fib_r(int n) { if (n == 0) return(O); if (n == 1) return(l); return(fibr(n-l) + fib_r(n-2));
Гпава 8. Динамическое программирование 295 Ход выполнения этого рекурсивного алгоритма можно проиллюстрировать его рекур- сивным деревом, как показано на рис. 8.1. Рис. 8.1. Дерево для рекурсивного вычисления чисел Фибоначчи Как и все рекурсивные алгоритмы, это дерево обрабатывается посредством обхода в глубину. Я настоятельно рекомендую проследить выполнение этого примера вручную, чтобы освежить ваши знания о рекурсии. Обратите внимание, что число /Д4) вычисляется на обеих сторонах дерева, а число F(2) вычисляется в этом небольшом примере целых пять раз. Полный вес этих избыточных вычислений становится ясным при исполнении программы. Для вычисления первых 45 чисел Фибоначчи моей программе потребовалось больше 7 минут. Скорее всего, используя правильный алгоритм, эти числа можно было бы быстрее вычислить вруч- ную. Сколько времени этот алгоритм будет вычислять число F(n)2 Так как IFn^<p = (\ + 45)/2= 1,61803 , это означает, что F„ > 1,6". Так как листьями наше- го дерева являются только 0 и 1, то такая большая сумма означает, что у нас должно быть, по крайней мере, 1,6" листов или вызовов процедуры! Иными словами, эта не- большая простенькая программа имеет экспоненциальную временную сложность. 8.1.2. Генерирование чисел Фибоначчи посредством кэширования Но в действительности мы можем решить эту задачу намного эффективнее. Для этого мы явно сохраняем (или кэшируем) результаты вычисления каждого числа Фибоначчи F(k) в таблице. Теперь, прежде чем вычислять значение, мы сначала проверяем его на- личие в таблице, таким образом избегая повторных вычислений. Соответствующий код приводится в листинге 8.2. ((define MAXN 45 ((define UNKNOWN -1 long f[MAXN+l]; long fib_c(int n) /* Наибольшее представляющее интерес число n*/ /* Пустая ячейка */ /* Массив для хранения вычисленных значений fib */
296 Часть I. Практическая разработка алгоритмов if (f[n] == UNKNOWN) I[n] = fib_c(n-l) * fib_c(n-2); return(f[n]); 1 long frb_c_drrver(int n) ( int 1; /* Счетчик */ f[0] = 0; f [1] = 1; for (i=2; i<=n; i++) f[i] = UNKNOWN; return(fib_c(n)); ) Чтобы вычислить F(n). мы вызываем процедуру fib_c_driver(n). Первым делом эта процедура инициализирует кэш двумя известными значениями (/ДО) и HI)) и флагом unknown для остальных, неизвестных, значений. Потом вызывается рекурсивный алго- ритм вычисления, модифицированный для предварительной проверки на наличие вы- численного значения данного числа Фибоначчи. Версия с кэшированием очень быстро доходит до максимального целого числа, кото- рое можно представить типом long integer. Причины этого становятся понятными по- сле изучения дерева рекурсии, показанного на рис. 8.2. Н(1) F(0) Рис. 8.2. Дерево вычисления чисел Фибоначчи методом кэширования В данном случае отсутствует сколь-либо значительное ветвление, т. к. вычисления вы- полняются только в левой ветви. Вызовы в правой ветви находят нужные значения в кэше и немедленно возвращают управление. Какова временная сложность этого алгоритма? Дерево рекурсии является более ин- формативным, чем код. Значение F(n) вычисляется за линейное время (т. е. за время О(п)), т. к. рекурсивная функция fib_c(k) вызывается ровно два раза для каждого зна- чения к, где 0 < к< п. Данный общий метод явного кэширования результатов вызовов рекурсивной функции для того, чтобы избежать повторных вычислений, позволяет максимально использо- вать преимущества динамического программирования, что делает его заслуживающим
Гпава 8 Динамическое программирование 297 более внимательного рассмотрения. В принципе, кэширование можно применять с лю- бым рекурсивным алгоритмом. Но сохранение частичных ре1ультатов не принесет ни- какой пользы при работе таких рекурсивных алгоритмов, как быстрая сортировка, пе- ребор с возвратами и обход в глубину, т. к. все рекурсивные вызовы в этих алгоритмах имеют разные значения параметров. Нет смысла сохранять то. что больше не будет использовано. Кэширование результатов вычислений имеет смысл только в случае достаточно не- большого пространства значений параметров, когда мы можем себе позволить расходы на хранение данных. Так как аргументом рекурсивной функции fib_c(k) является це- лое число из диапазона от 0 до п, то кэшировать нужно только О(и) значений. Нам вы- годно пойти на линейные затраты памяти вместо экспоненциальных затрат времени. Но, как мы увидим далее, полностью избавившись от рекурсии, можно получить еще лучшую производительность. Подведение итогов Явное кэширование результатов рекурсивных процедур позволяет максимально использо- вать преимущества динамического программирования, главным достоинством которого является такое же время выполнения, что и у более элегантных решений. Если вы предпо- читаете бесхитростное написание большого объема кода поиску красивого решения, вы можете не читать последующий материал. 8.1.3. Генерирование чисел Фибоначчи посредством динамического программирования Число Фибоначчи F„ можно вычислить за линейное время с большей эффективностью, если явно задать последовательность рекуррентных вычислений, как показано в лис- тинге 8.3. Листинг 8.3. Вычисления числа Фибоначчи без рекурсии long fib_dp(int п) int i; /* Счетчик */ long f[MAXN+l]; /* Массив для хранения вычисленных значений fib */ f[0] = 0; f[l] 1; for (i=2; ion; i++) f[i] = f[i-1]+f[i-2]; return (f [n]) ; Как можно видеть, в данном варианте процедуры рекурсия совсем не используется. Мы начинаем вычисление последовательности Фибоначчи с ее наименьшего значения до заданного числа и сохраняем все промежуточные результаты. Таким образом, когда нам нужно вычислить число F,, мы уже имеем требуемые для этого числа F, । и F,2- Линейная временная сложность этого алгоритма должна быть очевидной. Вычисление каждого из и значений выполняется как простое суммирование двух целых чисел, сложность которого, как по времени, так и по памяти, равна О(п).
298 Часть I. Практическая разработка алгоритмов Но более внимательное исследование процесса решения задачи показывает, что совсем не обязательно хранить все промежуточные результаты в течение вычисления всей по- следовательности. Так как вычисляемое значение зависит от двух аргументов, то толь- ко их и нужно сохранять. Соответствующая реализация показана в листинге 8.4. ................—---------------------..-------------.....и...,.,....,-....,................... ..............---------- Листинг 8.4. Окончательная версия процедуры вычисления чисел Фибоначчи long fib_ultimate(int n) { int i; long back2=0, backl=l; long next; if (n == 0) return (0); for (i=2; i<n; i++) { next = backl+back2; back2 = backl; backl = next; } return(backl+back2); /* Счетчик ★/ /* Последние два значения f[n] */ /* Промежуточная сумма */ Данная реализация понижает сложность по памяти до постоянной, при этом никак не ухудшая сложность по времени. 8.1.4. Биномиальные коэффициенты В качестве другого примера устранения рекурсии указанием порядка вычислений рас- смотрим вычисление биномиальных коэффициентов. Биномиальные коэффициенты являются наиболее важным классом натуральных чисел. В комбинаторике биномиаль- ный коэффициент (£) представляет количество возможных подмножеств из к элемен- тов множества из п элементов. Каким образом находятся биномиальные коэффициенты? Так как (") = я!/((л- к)\к\\ то их значения можно получить, вычисляя факториалы. Но этот метод имеет серьезный недостаток. Промежуточные вычисления могут вызвать арифметическое переполне- ние, даже если конечный результат не превышает максимальное допустимое целое число в компьютере. Более надежным способом вычисления биномиальных коэффициентов является ис- пользование неявного рекуррентного соотношения, используемого для построения треугольника Паскаля: 1 1 1 1 2 1 13 3 1 1 4 6 4 1 1 5 10 10 5 1
Гпава 8. Динамическое программирование 299 Каждый элемент строки является суммой двух элементов слева и справа от него в предшествующей строке. Это подразумевает следующее рекуррентное соотношение: Почему эта формула дает правильный результат? Рассмотрим, входит ли и-й элемент в одно из подмножеств (£) из к элементов. Если входит, то мы можем завершить созда- ние подмножества, выбрав другие к- 1 элементов из оставшихся n- 1 элементов мно- жества. Если же не входит, то нам нужно выбрать все к элементов из оставшихся п - 1 элементов множества. Эти два случая не пересекаются и включают в себя все возмож- ности, поэтому в сумме учитываются все к подмножеств. Для любого рекуррентного соотношения требуется база. Какие значения биномиаль- ных коэффициентов нам известны без вычислений? Левая часть формулы легко приво- дится к ("о*). Сколько можно создать подмножеств, содержащих 0 элементов, из не- кого множества? Ровно одно— пустое. Если это не кажется вам убедительным, с рав- ным успехом в качестве базы можно принять С") = т. Правая часть формулы легко приводится к (£). Сколько можно создать подмножеств, содержащих к элементов, из множества, содержащего к элементов? Ровно одно— первоначальное множество. Эти базовые экземпляры и рекуррентное соотношение определяют все интересующие нас биномиальные коэффициенты. Самым лучшим способом вычисления такого рекуррентного соотношения будет созда- ние таблицы всех возможных значений, имеющей требуемый размер, как показано на рис. 8.3. П1 / п 0 1 2 3 4 5 0 А 1 В G 2 С 1 Н 3 D 2 3 I 4 Е 4 5 6 J 5 F 7 8 9 10 К а) m / п 0 1 2 3 4 5 0 1 1 1 1 2 1 1 1 3 1 3 3 1 4 1 4 6 4 1 5 1 5 10 10 5 1 б) Рис. 8.3. Порядок вычисления биномиальных коэффициентов (а), содержимое матрицы после вычислений (б) Инициализированные ячейки таблицы помечены буквами от А до К. обозначающими порядок, в котором им были присвоены значения. Каждой неинициализированной ячейке присваивается значение, равное сумме значений двух ячеек из предыдущего ряда: той. что непосредственно над ней. и той. что сверху и слева. Цифры от 1 до 10. маркирующие треугольник ячеек, обозначают порядок вычисления подмножества (4) = 5 е помощью кода, представленного в листинге 8.5.
300 Часть I. Практическая разработка алгоритмов Листинг 8.5. Вычисление биномиального коэффициента long binomial_coefficient(n, m) Int n, m; /* Вычислить n, выбрать m */ f int i, j; /* Счетчики */ long be[MAXN][MAXN]; /* Таблица биномиальных коэффициентов */ for (1=0; =n; i++) bc[i][0] 1; for (j=0; j<=n; j++) bc(j] [j] 1; for (1=1; i<=n; i++) for (j=l; j<i; j++) = bc[i-l] [j-1] + bc[i-l][j]; return bc[n][m] Внимательно изучите эту функцию, чтобы понять, как она работает. Далее в этой главе мы будем уделять больше внимания вопросам формулирования и анализа соответст- вующей рекуррентности, чем технике работы с таблицами. 8.2. Поиск приблизительно совпадающих строк Поиск совпадающих комбинаций символов в текстовых строках является задачей, важность которой не подлежит сомнению. В разделе 3.7.2 были представлены алго- ритмы для поиска точных совпадений комбинаций символов, т. е. выяснения, где именно в текстовой строке Т находится искомая подстрока Р. Но, к сожалению, жизнь не всегда так проста. Например, слова, как в тексте, так и в искомой строке, могут быть написаны с ошибками, вследствие чего точного совпадения не получится. Эволюцион- ные изменения в геномных последовательностях или языковых конструкциях приводят к тому, что мы вводим в строке поиска "Не убей", когда нужно найти "Не убий". Каким образом можно нейтрализовать орфографические ошибки, чтобы можно было найти подстроку, наиболее близкую к искомой? Чтобы можно было выполнять поиск неточно совпадающих строк, нам нужно сначала определить функцию стоимости, с помощью которой мы будем выяснять, насколько далеко две строки находятся друг от друга, т. е. измерять расстояние между парами строк. Для этого нам нужно учиты- вать количество операций, которые придется выполнить, чтобы преобразовать одну строку в другую. Возможны такие действия: ♦ замена— заменяет один символ строки Р другим символом из текста Т. Например, "shot" превращается в "spot"; ♦ вставка— вставляет один символ в строку Р так, чтобы она совпадала с подстро- кой в тексте Т. Например, "ago" превращается в "agog"; ♦ удаление— удаляет один символ из строки Р так, чтобы она совпадала с подстро- кой в тексте Т. Например, "hour" превращается в "our". Для правильной постановки вопроса сходства строк нам нужно установить стоимость каждой из этих операций преобразования строк. Присваивая каждой операции стои- мость равную 1. мы определяем расстояние редактирования между двумя строками.
Гпава 8 Динамическое программирование 301 Задача нечеткого сравнения строк возникает во многих приложениях и рассматривает- ся в разделе 18.4. Нечеткое сравнение строк может казаться трудной задачей, т. к. нам нужно решить, где именно в строке-образце и тексте нужно вставить или удалить символы и сколько сим- волов нужно вставить или удалить. Попробуем подойти к рассмотрению задачи с дру- гого конца. Какая информация нам понадобится для принятия окончательного реше- ния? Что может случиться при сравнении с последним символом каждой строки? 8.2.1. Применение рекурсии для вычисления расстояния редактирования При создании рекурсивного алгоритма можно использовать то обстоятельство, что ли- бо последний символ в строке совпадет с искомым, либо нам придется его заменить, удалить или добавить. В результате отсечения символов, участвующих в этой послед- ней операции редактирования, останется пара более коротких строк. Пусть i и j будут последними символами префиксов строк Р и Т соответственно. В результате последней операции образуются три пары более коротких строк, соответствующих совпадающим строкам либо строкам, полученным после замены, добавления или удаления. Если бы нам была известна стоимость редактирования этих более коротких строк, мы могли бы решить, какая опция дает наилучшее решение, и соответственно выбрать эту опцию. Оказывается, с помощью рекурсии мы може.м узнать эту стоимость. Пусть £>[/.у]— минимальное количество различий между образцами Р\, Р,, .... Р, и фрагментом текста Т, оканчивающимся на j. Иными словами, D[i,j] представляет са- мый дешевый из трех возможных способов расширения более коротких строк: ♦ если (Р, = ГД тогда D[i- l,j - 1], в противном случае D[i- l,j - 1] + 1. Это означа- ет. что в зависимости от того, одинаковы ли последние символы строк, мы получа- ем совпадение /-го и j-го символов или заменяем их; ♦ Z)[z-l,j]+ 1. Это означает, что в строке образца имеется дополнительный символ, который нужно учесть, поэтому указатель позиции в тексте не продвигается и вы- полняется вставка символа: ♦ D[i,j- 1]+ 1. Это означает, что в тексте имеется лишний символ, который нужно удалить, поэтому указатель позиции в строке-образце не продвигается и выполняет- ся удаление символа. В листинге 8.6 представлена программа вычисления стоимости редактирования. Листинг 8.6. Вычисление стоимости редактирования методом рекурсии ♦define MATCH О ♦define INSERT 1 ♦define DELETE 2 /* Символ перечислимого типа для /* Символ перечислимого типа для /* Символ перечислимого типа для совпадения */ вставки */ удаления */ int string_compare (char *s. char *t, int i, int j) int k; int opt [3]; int lowest_cost; /* Счетчик*/ /* Стоимость трех опций */ /* Самая низкая стоимость */
302 Часть I. Практическая разработка алгоритмов if (i == 0) return!] * indel(' ')); if (j == 0) return(i + indel(' ')); opt(MATCH) = string_compare(s, t, i-1, j-1) + match(s[i],t [j ]); opt[INSERT] = string_compare(s,t,i,j-1) + indel (t[j]); opt[DELETE] = string_compare(s,t,i-1, j) + indel(s[i]); lowest_cost = opt[MATCH]; for (k=INSERT; k<=DELETE; k++) if (opt[k] < lowest_cost) lowest_cost = opt[k]; return( lowest_cost ); > Легко убедиться, что эта программа абсолютно корректна. Однако работает она недо- пустимо медленно. На моем компьютере сравнение двух строк длиной в 11 символов занимает несколько секунд, а более-менее длинные строки обрабатываются целую веч- ность. Почему этот алгоритм работает так медленно? Потому что он вычисляет одни и те же значения по нескольку раз, в результате чего имеет экспоненциальную временную сложность. В каждой позиции строки происходит тройное ветвление рекурсии, вслед- ствие чего количество вызовов растет со скоростью 3", а на самом деле даже быстрее, т. к. при большинстве вызовов сокращается только один из двух индексов, а не оба. 8.2.2. Применение динамического программирования для вычисления расстояния редактирования Так как же можно сделать этот алгоритм пригодным для использования на практике? В решении этого вопроса важным является то обстоятельство, что при большинстве рекурсивных вызовов выполняются вычисления, которые уже были выполнены. Отку- да нам это известно? Возможно только |Р|*|7] однозначных рекурсивных вызовов, т. к. именно столько имеется разных пар передаваемых в качестве параметров. Сохра- нив вычисленные значения для каждой из этих (/, j) пар в таблице, мы можем при не- обходимости извлечь их из этой таблицы, вместо того, чтобы вычислять их снова. Таблица для хранения вычисленных значений представляет собой двумерную матрицу т, каждая из ячеек которой содержит значение стоимости оптимального решения подзадачи, а также указатель на родительский элемент, объясняющий, как мы попали в эту ячейку. Объявление структуры таблицы показано в листинге 8.7. typedef struct { int cost; /* Стоимость попадания в данную ячейку */ int parent; /* Родительская ячейка */ ) cell; cell m[MAXLEN+l][MAXLEN+1]; /* Таблица динамич. программирования */ Между динамической и рекурсивной версиями реализации алгоритма поиска неточно совпадающих строк имеются три различия. Во-первых, промежуточные значения по- лучаются из таблицы, а не в результате рекурсивных вызовов. Во-вторых, обновляется содержимое поля родительского указателя каждой ячейки, что в дальнейшем позволит
Гпава 8. Динамическое программирование 303 восстановить последовательность редактирования. В-третьих, в реализации исполь- зуется более общая функция goai_ceil(), а не просто возвращается значение ш[ IрI ] [ 1 тI ] .cost. Как следствие, мы можем применять данную процедуру для решения более обширного класса задач. Реализация алгоритма представлена в листинге 8.8. Листинг 8.8. Вычисление стоимости редактировании int string_compare(char *s, char *t) int i, j, k; /* Счетчики ’/ int opt[3]; /* Стоимость трех вариантов */ for (i=0; i<MAXLEN; i++) I row_init(i); column_init(i); for (i=l; i<strlen(s); i++) { for (j=l; j<strlen(t); j++) { opt[MATCH] = m[i-l][j—1].cost + match(s[i],t[j]); opt[INSERT] = m[i][j-1].cost + index(t[j]); opt[DELETE] = m[i-l][j].cost + indel(s[i]); m[i][j].cost = opt[MATCH]; m[i][j].parent = MATCH; for (k=INSERT; k<=DELETE; k++) if (opt[k] < m[i] [j] .cost) { m[i][j].cost =opt[k]; m[i] [j] .parent = k; ) ) goal_cell(s, t, &i, &j); return( m[i][j].cost ); Здесь строки и индексы используются несколько необычным образом. В частности, полагается, что в начало каждой строки был добавлен пробел, вследствие чего первый значащий символ строки s находится в позиции з[1]. Это позволяет нам синхронизи- ровать индексы матрицы m с индексами строк. Вспомните, что нам нужно выделить нулевую строку и столбец матрицы ш для хранения граничных значений, совпадающих с пустым префиксом. В качестве альтернативы мы бы могли оставить входные строки без изменений и просто откорректировать должным образом индексы. Для определения значения ячейки (z,j) требуются три готовых значения из ячеек (»- I,/- 1), (i,j- 1) и (i — 1, j). Подойдет любой порядок вычислений, удовлетворяю- щий этому условию, в том числе построчный, используемый в этой программе1. 1 Допустим, мы создадим граф с вершиной для каждой ячейки матрицы и ориентированным ребром (т.г). означающим, что для вычисления значения ячейки у нужно значение ячейки .г. Любая топологическая сор- тировка на получившемся бесконтурном орграфе (кстати, почему это будет бесконтурный орграф?) опреде- ляет приемлемый порядок вычислений.
304 Часть I. Практическая разработка алгоритмов В качестве примера приводятся матрицы стоимости преобразования строки р = "thou shall not" в строку t = "you should not" за пять шагов (табл. 8.1). Таблица 8.1. Пример матрицы динамического программирования T У 0 ll - s h 0 Ll 1 d - П О t р ПОЗ. 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 0 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 Г. 1 1 1 2 3 4 5 6 7 8 9 10 11 12 13 13 h: 2 2 2 2 3 4 5 5 6 7 8 9 10 11 12 13 о: 3 3 3 2 3 4 5 6 5 6 7 8 9 10 11 12 и: 4 4 4 3 2 3 4 5 6 5 6 7 8 9 10 11 5 5 5 4 3 2 3 4 5 6 6 7 7 8 9 10 s: 6 6 6 5 4 3 2 3 4 5 6 7 8 8 9 10 h: 7 7 7 6 5 4 3 2 3 4 5 6 7 8 9 10 a: 8 8 8 7 6 5 4 3 3 4 5 6 7 8 9 10 1: 9 9 9 8 7 6 5 4 4 4 4 5 6 7 8 9 t 10 10 10 9 8 7 6 5 5 5 5 5 6 7 8 8 11 11 11 10 9 8 7 6 6 6 6 6 5 5 7 8 n: 12 12 12 11 10 9 8 7 7 7 7 7 6 5 6 7 o: 13 13 13 12 11 10 9 8 7 8 8 8 7 6 5 6 t: 14 14 14 13 12 11 10 9 8 8 9 9 8 7 6 5 8.2.3. Восстановление пути Функция сравнения строк возвращает стоимость оптимального выравнивания, но не выполняет собственно выравнивание. Нам полезно знать, что для преобразования строки "thou shall not" в строку "you should not" требуется только пять операций редак- тирования. но было бы еще полезнее знать последовательность этих операций. Возможные решения определенной задачи динамического программирования описы- ваются путями в матрице динамического программирования, начинающимися с перво- начальной конфигурации (пары пустых строк (0, 0)) и заканчивающимися конечным требуемым состоянием (парой заполненных строк (|Р|,|Т|)). Ключом к созданию реше- ния задачи является реконструкция решений, принимаемых на каждом шаге оптималь- ного пути, ведущего к целевому состоянию. Эти решения записаны в поле указателя на родителя каждой ячейки массива. Нужные решения можно воспроизвести, выполнив проход по решениям в обратном направлении от целевого состояния, следуя указателям на родительские ячейки масси- ва, пока не придем к начальной ячейке пути решения задачи. Поле указателя на роди- теля ячейки содержит информацию о типе операции, выполненной в этой ячей- ке— MATCH, INSERT или DELETE. Обратная трассировка пути решения в табл. 8.2 преобразования строки "thou-shalt-not" в строку "you-should-not" выдает нам последова- тельность операций редактирования DSMMMMM1SMSMMMM. Это означает, что мы удаляем первую букву "t", заменяем букву "h" буквой "у", оставляем без изменений
Гпава 8 Динамическое программирование 305 следующие пять букв, после чего вставляем букву "о", заменяем букву "а" буквой "и" и, наконец, заменяем букву "t" буквой "d". Таблица 8.2. Матрица указателей на родителей T У 0 u - s h 0 u 1 cl - n О t p 1103. 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 0 -/ 1 1 1 1 1 1 1 1 1 1 1 1 1 1 t: 1 2 0 0 0 0 0 0 0 0 0 0 0 0 0 0 h: 2 2 0 0 0 0 0 0 I 1 1 1 1 1 1 1 o: 3 2 0 0 0 0 0 0 0 1 1 1 1 1 0 1 u: 4 2 0 2 0 1 1 ; 1 0 1 1 1 1 1 1 5 • 2 0 2 2 0 1 I 1 1 0 0 - 0 1 1 1 s: 6 2 0 2 2 2 0 i 1 1 1 0 0 0 0 0 h: 7 2 0 2 2 2 2 0 / 1 1 1 1 1 0 0 a: 8 2 0 2 2 2 2 2 0 0 0 0 0 0 0 0 1: 9 2 0 2 2 2 2 2 0 0 0 1 1 1 1 1 t: 10 2 0 2 2 2 2 2 0 0 0 0 0 0 0 0 11 2 0 2 2 0 2 2 0 0 0 0 0 1 1 1 n: 12 2 0 2 2 2 2 2 0 0 0 0 2 0 1 1 o: 13 2 0 0 2 2 2 2 0 0 0 0 2 2 0 1 t: 14 2 0 2 2 2 2 2 2 0 0 0 2 2 2 0 Трассировка указателей на родительские ячейки восстанавливает решение в обратном порядке. Но с помощью рекурсии мы можем восстановить прямой порядок решения. Соответствующий код представлен в листинге 8.9. : Листинг 8.9. Восстановление решения в прямом порядке reconstruct_path(char *s, char *t, int i, int j) ( if (m[i][j].parent == -1) return; if (m[i][j].parent == MATCH) ( reconstruct_path(s, t, i-1, j-1); match_out(s, t, i, j); return; ) if (m[i][j].parent == INSERT) { reconstruct_path(s, t, i, j-1); insert_out(t, j); return; ) if (m[i][j].parent == DELETE) { reconstruct_path(s, t, i-1, j); deleteout(s, i); return;
306 Часть I. Практическая разработка алгоритмов Для многих задач, включая задачу вычисления расстояния редактирования, путь реше- ния можно восстановить из матрицы стоимости, не прибегая к явному сохранению ука- зателей на предыдущий элемент. Для этого нам нужно выполнить трассировку назад, начиная со значений стоимости трех возможных родительских ячеек и соответствую- щих символов строки, чтобы восстановить операцию, в результате выполнения кото- рой мы оказались в текущей ячейке при данной стоимости. 8.2.4. Разновидности расстояния редактирования Процедуры сравнения строк (string compare) И восстановления пути (reconstruct path) вызывают несколько функций, которые еще не были определены. Эти функции можно разбить на четыре категории. ♦ Инициализация таблицы. Функции row inito и column init () инициализируют ну- левые строку и столбец таблицы соответственно. В задаче вычисления расстояния редактирования ячейки (/, 0) и (0, /) соответствуют сравнению строк длиной i с пус- той строкой. Для этого требуется ровно / вставок/удалений. поэтому код этих функ- ций очевиден (листинг 8.10). Листинг 8.10. Процедуры инициализации строк и столбцов таблицы row_imt (int i) I m[0] Li].cost = i; if (i>0) m[0][ij.parent = INSERT; else m[0][i].parent = -1; column_init(int i) ( m[i][0]-cost i; if (i>0) mlij[0].parent = else m[i][0].parent = DELETE; -1; ♦ Стоимость операций. Функции match (c, d) и indei(c) возвращают стоимость пре- образования символа с в символ d и вставки/удаления символа с соответственно. Для стандартной задачи вычисления расстояния редактирования функция match)) возвращает 0, если символы одинаковые, и 1 в противном случае. Функция indeio всегда возвращает 1, независимо от аргумента. Но можно также использовать функции стоимости, специфичные для конкретных приложений. Такие функции мо- гут быть менее взыскательными к заменяемым символам, расположенным друг воз- ле друга на стандартной раскладке клавиатуры, или символам, которые выглядят или звучат похоже. Общая реализация функций стоимости показана в листинге 8.11.
Гпава 8. Динамическое программирование 307 Листинг 8.11. Функции стоимости int match (char с, char d) if (c == d) return(O); else return (1); I int indel (char c) return(1); ♦ Определение целевой ячейки. Функция goal cell (j возвращает индексы ячейки, обо- значающей конечную точку решения. В случае задачи вычисления расстояния ре- дактирования это значение определяется длиной двух входных строк. Но другие приложения, которые мы вскоре рассмотрим, не имеют фиксированного располо- жения решений. Общая реализация функции определения местонахождения целевой ячейки приво- дится в листинге 8.12. Листинг 8.12. Функция определения местонахождения целевой ячейки goal_cell (char * *s, char *t, int *i, int * j) *1 = strlen(s) — 1; = strlen(t) 1; ♦ Операция обратной трассировки. Функции match_out(>, insert_outo и delete out () выполняют соответствующие действия для каждой операции редакти- рования при трассировке решения. В случае задачи вычисления расстояния редак- тирования это означает вывод названия операции или обрабатываемого символа, в зависимости от требований приложения. Общая реализация функций трассировки решения приводится в листинге 8.13. Листинг 8.13. Функции трассировки решения insert_out (char *t, int j) ( printf("I"); delete_out (char *s, int i) printf("D”); I match_out (char *s, char *t, int i, int j) if (s[i]==t[jl) printf("M"); else printf("S");
308 Часть I. Практическая разработка алгоритмов Для задачи вычисления расстояния редактирования эти функции довольно простые. Но следует признать, что правильное выполнение операций получения граничных состоя- ний и манипулирования индексами является трудной задачей. Хотя при должном по- нимании применяемого метода алгоритмы динамического программирования легко разрабатывать, для правильной реализации деталей требуется внимательно продумать и всесторонне протестировать предлагаемое решение. На первый взгляд кажется, что для такого простого алгоритма потребовалось слишком много вспомогательной работы. Но теперь, только слегка модифицировав эти общие функции, мы можем решить несколько важных задач, как особые случаи задачи вы- числения расстояния редактирования. ♦ Поиск подстроки в тексте. Допустим, что мы хотим найти в тексте Т подстроку, приблизительно совпадающую с подстрокой Р. Например, будем искать строку "Skiena" и возможные ее варианты (Skienna, Skena, Skeena, Skina и т. п.). Поиск с помощью нашей первоначальной функции вычисления расстояния редактирова- ния будет малочувствительным, т. к. большая часть стоимости любого редактиро- вания будет определяться удалением фрагментов текста, не совпадающих со стро- кой "Skiena". В данном случае оптимальным решением будет поиск всех разбросан- ных в тексте совпадений с "...S...k...i...e...n...a" и удаление всего остального. Нам требуется такой способ вычисления расстояния редактирования, в котором стоимость начала совпадения не зависит от местонахождения в тексте, чтобы учи- тывались и совпадения в середине текста. Теперь целевое состояние находится не обязательно в конце строк, а в таком месте текста, где совпадение со всем образцом имеет минимальную стоимость. Модифицировав эти две функции, мы получим тре- буемое правильное решение, как показано в листинге 8.14. ♦ Самая длинная общая подпоследовательность. Допустим, что нам нужно найти самую длинную последовательность разбросанных символов, общую для обеих строк. (Данная задача рассматривается в разделе 18.8). Общая подпоследовательность представляет собой последовательность всех раз- бросанных совпадающих символов в обеих строках. Например, самой длинной об- Листинг 8.14. Модифицированные функции для поиска неточно совпадающих строк row_init(int i) { m[0][i].cost = 0; /* ИЗМЕНЕНИЕ */ m[0][i].parent = -1; /* ИЗМЕНЕНИЕ */ } goal_cell(char *s, char *t, int *i, int *j) , ( int k; /* Счетчик ’/ *i = strlen(s) — 1; + j = 0; for (k=l; k<strlen(t); k++) if (m[*i][k].cost < m(*i][*j].cost) *j = k;
Гпава 8. Динамическое программирование 309 щей подпоследовательностью строк "democrat" и "republican" является "еса". Чтобы максимизировать количество таких совпадений, требуется предотвратить замену несовпадающих символов. Когда замена символов запрещена, избавиться от несов- падающих символов можно только с помощью операций вставки и удаления. Для получения самого дешевого выравнивания будет выполнено наименьшее количест- во таких операций, поэтому в ней должна сохраниться самая длинная общая под- последовательность. Чтобы получить требуемое выравнивание, мы модифицируем функцию стоимости совпадений, чтобы повысить стоимость замены символов, как показано в листинге 8.15. Листинг 8.15. Модифицированная функция стоимости совпадений int match (char с, char d) if (c == d) return(O); else return(MAXLEN); В действительности, чтобы сделать замену полностью непривлекательной операци- ей редактирования, будет достаточным сделать ее стоимость выше, чем совместная стоимость вставки и удаления. ♦ Максимальная монотонная подпоследовательность. Числовая последовательность является монотонно возрастающей, если каждый следующий элемент этой после- довательности превышает предыдущий. Задача максимальной монотонной подпос- ледовательности состоит в удалении наименьшего количества элементов из вход- ной строки 5 с целью получения монотонно возрастающей подпоследовательности. Например, для последовательности 243517698 максимальной возрастающей под- последовательностью является 23568. По сути, это просто задача поиска самой длинной общей подпоследовательности, где второй строкой являются элементы строки 5, отсортированные в возрастающем порядке. Любая из этих двух общих последовательностей должна, во-первых, со- держать символы в соответствующем порядке в строке S’ и, во-вторых, содержать только символы с возрастающими номерами позиций в последовательности упоря- дочивания. так что более длинная из этих последовательностей является решением. Конечно же, этот подход можно модифицировать для поиска самой длинной убы- вающей последовательности, просто изменив порядок сортировки на обратный. Как можно видеть, базовую процедуру вычисления расстояния редактирования можно с легкостью приспособить для решения многих интересных задач. Секрет заключается в определении, что конкретная задача является всего лишь частным случаем общей задачи нечеткого сравнения строк. Внимательный читатель может заметить, что для вычисления затрат на выравнивание не требуется содержание всех О(тп) ячеек. Если мы будем вычислять рекуррентное соотношение, заполняя столбцы матрицы слева направо, то нам никогда не потре- буется больше двух столбцов для хранения всей требуемой для этого вычисления ин- формации. Таким образом, для вычисления рекуррентного соотношения достаточно
310 Часть I. Практическая разработка алгоритмов объема памяти О(т), при этом сложность по времени остается прежней. Но, к сожале- нию. не имея полной матрицы, мы не можем восстановить выравнивание. Экономия памяти является важным моментом динамического программирования. Так как объем памяти на любом компьютере ограничен, то сложность но памяти, равная О(пт), является более узким местом, чем такая же сложность по времени. К счастью, эта проблема решается с помощью алгоритма "разделяй и властвуй", который выпол- няет выравнивание за время ()(пт), требуя при этом память объемом О(т). Этот алго- ритм рассматривается в разделе 18.4. 8.3. Самая длинная возрастающая последовательность Решение задачи посредством динамического программирования состоит из трех шагов: 1. Сформулировать решение в виде рекуррентного соотношения или рекурсивного ал- горитма. 2. Показать, что количество разных значений параметра, принимаемых рекуррент- ностью, ограничено полиномиальной функцией (будем надеяться, небольшой сте- пени). 3. Указать порядок вычисления рекуррентного соотношения, с тем, чтобы частичные результаты были всегда доступными, когда они требуются. Чтобы разобраться в этих деталях, рассмотрим разработку алгоритма поиска самой длинной монотонно возрастающей подпоследовательности в последовательности из п чисел. По правде говоря, эта задача была описана, как частный случай задачи вычисле- ния расстояния редактирования в предшествующем разделе, где она называлась зада- чей поиска максимальной монотонной подпоследовательности. Тем не менее, будет полезно рассмотреть разработку се решения с самого начала. В действительности, ал- горитмы динамического программирования часто легче разработать сначала, чем ис- кать существующее решение. Возрастающая последовательность отличается от серии, в которой элементы физиче- ски находятся рядом друг с другом. Выбранные элементы обеих структур должны быть отсортированы в возрастающем порядке слева направо. Рассмотрим, например, после- довательность 5= {2, 4. 3, 5, I, 7. 6. 9, 8}. Самая длинная возрастающая подпоследовательность последовательности .S' состоит из пяти символов: {2. 3, 5, 6, 8}. В действительности, подпоследовательностей этой длины в последовательности 5 имеется восемь. (Попробуйте найти их.) А самых длинных серий, длиной в два символа, имеется четыре: (2. 4). (3. 5). (1.7) и (6, 9). Поиск самой длинной возрастающей серии в числовой последовательности является простой задачей. По сути, разработка алгоритма с линейным временем исполнения не должна вызвать особых трудностей. Но задача поиска самой длинной возрастающей подпоследовательности значительно сложнее. Как мы можем определить, какие раз- бросанные элементы пропустить? Чтобы применить динамическое программирование, нам нужно создать рекуррентное соотношение, которое вычисляет длину самой длин- ной последовательности. Чтобы найти подходящее рекуррентное соотношение, задайте
Гпава 8. Динамическое программирование 311 себе вопрос, какая информация о первых n- I элементах последовательности \ помог- ла бы найти решение для всей последовательности? ♦ Длина самой длинной возрастающей подпоследовательности последовательности $1. 52....5„_| кажется полезной информацией. Более того, это будет самая длинная возрастающая подпоследовательность в .S’. если только s„ не предоставит возрас- тающую последовательность такой же длины. К сожалению, длина этой последовательности не является достаточной информаци- ей для получения полного решения. Допустим, что каким-то образом мы узнали, что самая длинная возрастающая подпоследовательность последовательности sj. 52, ..., 5,,_| содержит пять символов и что 5„ = 9. Будет ли длина последней самой длинной возрастающей подпоследовательности последовательности .S’ равняться 5 или 6? ♦ Нам необходимо знать количество символов в самой длинной последовательности, которую расширит s„. Чтобы быть уверенным в том, что мы знаем это. нам в дейст- вительности нужно знать количество символов в самой длинной последовательно- сти. которую может расширить иобое возможное значение для х„. Это предоставляет нам базовую идею для создания рекуррентного соотношения. Опре- деляем /„ как количество символов самой длинной последовательности, заканчиваю- щейся на 5,. Самая длинная возрастающая подпоследовательность, содержащая н-е число, получа- ется в результате добавления этого числа в конец самой длинной возрастающей после- довательности слева от п и оканчивающейся числом, меньшим, чем х„. Длина /, вычис- ляется с помощью следующего рекуррентного соотношения: /, = max /, +1, где s, < х„ 0</« ' /о= О Эти значения определяют количество символов в самой длинной возрастающей после- довательности, заканчивающейся данным числом. Количество символов в самой длин- ной возрастающей подпоследовательности полной перестановки можно выразить фор- мулой maxi < ,< „/„ т. к. самая лучшая последовательность должна когда-нибудь закон- читься. В табл. 8.3 приводятся данные для предыдущего примера. Таблица 8.3. Данные для задачи поиска самой длинной возрастающей подпоследовательности Последовательность х, 2 4 3 5 1 7 6 9 8 Длина 1, 1 2 3 3 1 4 4 5 5 Предшественник р, - 1 1 2 - 4 4 6 6 Какую вспомогательную информацию нам следует сохранить, чтобы восстановить са- му последовательность, а не только ее длину? Для каждого элемента х, сохраняется его предшественник, а именно индекс р, элемента, непосредственно предшествующего х, в самой длинной возрастающей последовательности, заканчивающейся на s,. З ак как все эти указатели направлены влево, то самую длинную последовательность можно
312 Часть I. Практическая разработка алгоритмов с легкостью восстановить, начав с се последнего значения и следуя указателям на дру- гие ее члены. Какова временная сложность этого алгоритма? Если каждое из п значений /, вычисля- ется путем сравнения s, с п значениями слева от него (где n>i- I), то общее время это- го анализа будет равно О(гГ). На самом деле, умело используя словарные структуры данных, это рекуррентное соотношение можно вычислить за время O(nlgn). Поскольку простое рекуррентное соотношение легче поддается программированию, лучше начать с его реализации. Подведение итогов Когда у вас достаточно опыта динамического программирования, то такие алгоритмы бу- дет легче создавать с нуля, чем искать готовое подходящее решение. 8.4. История из жизни. Эволюция омара Придя утром на работу, я застал возле моего кабинета двух аспирантов. Это были два будущих кандидата наук, работающих в области высокопроизводительной компьютер- ной графики. Они занимались исследованием новых методов для создания красивых компьютерных изображений, но картина, которую они нарисовали мне в то утро, кра- сивой не была. — Видите ли. мы хотим создать программу для морфинга одного изображения в дру- гое, — начали объяснять они. — Что вы понимаете под морфингом? — спросил я. — Для реализации спецэффектов в фильмах мы хотим создавать промежуточные сце- ны при преобразовании одного изображения в другое. Допустим, мы хотим превратить вас в актера Хамфри Богарта. Чтобы такое превращение выглядело правдоподобным, нам нужно создать множество промежуточных кадров, таких, что в первых вы выгля- дите сами собой, а в последних, как он. — Если вы вправду можете превратить меня в Богарта, то вы действительно нашли что-то стоящее. — согласился я. — Но у нас возникают проблемы, т. к. процесс выглядит не очень правдоподобно.— Они показали мне морфинг ужасного качества. — Проблема состоит в том. что нам нужно найти правильное соответствие между чертами двух изображений. Результат никуда не годится, когда мы получаем неправильное соответствие и пытаемся преобра- зовать, скажем, губу в ухо. — Еще бы. Значит, вы хотите, чтобы я вам дал алгоритм, чтобы сопоставлять губы с губами? — Даже проще, чем это. Мы преобразуем каждую строку первоначального изображе- ния в такую же строку конечного изображения. Нам нужно найти оптимальное соот- ветствие между темными пикселами в линии объекта А и темными пикселами соответ- ствующей линии объекта В. Вот как здесь,— показали они мне пример успешных со- поставлений (рис. 8.4). — Понятно, — сказал я. — Вы хотите сопоставить крупные темные области с крупны- ми, а небольшие — с небольшими.
Гпава 8 Динамическое программирование 313 Сегменты объекта А Сегменты объекта В Рис. 8.4. Успешное сопоставление двух линий пикселов — Да, при условии, что такое сопоставление не смещает их слишком сильно влево или вправо. В таком случае, вероятно, лучше слить области или разбить область на две, чем смещать их слишком далеко, т. к. это может привести к сопоставлению подбородка с бровями. Как нам лучше всего сделать это? — Последний вопрос. Вам нужно будет сопоставлять два интервала таким образом, чтобы они пересекались? — Думаю, что нет. Пересекающиеся интервалы нельзя сопоставлять. Это было бы рав- носильно обмену местами правого и левого глаза. Я попытался изобразить задумчивость, но я не такой хороший актер, как Богарт. Идея решения этой задачи возникла у меня, лишь только они начали говорить о линиях пик- селов. Они хотели преобразовать один массив пикселов в другой массив с минималь- ными изменениями. Это выглядело как редактирование строки пикселов, что является классическим применением динамического программирования. (Нечеткое сравнение строк обсуждается в разделах 8.2 и 18.4. ) То обстоятельство, что интервалы не могли пересекаться, окончательно утвердило эту идею в качестве решения. Это означало, что любая задача сопоставления полосы тем- ных пикселов объекта А с полосой темных пикселов объекта В разбивалась на две меньшие подзадачи, т. е. на полосу пикселов слева и справа от сопоставляемого пиксе- ла. Конечной стоимостью глобального сопоставления будет стоимость данного сопос- тавления плюс сумма значений стоимости сопоставлений всех пикселов справа и сле- ва. Создание оптимального сопоставления левой части является меньшей и соответст- венно более легкой задачей. Кроме этого, может быть только О(гГ) возможных подзадач для левой части, т. к. каждая полностью описывается парой из и верхних пик- селов и п нижних пикселов. — Для решения вашей задачи мы используем алгоритм динамического программиро- вания.— заявил я. — Но решить ее можно несколькими способами, в зависимости от того, что вы хотите редактировать, пикселы или серии пикселов. Я бы. наверное, пре- образовал каждую строку пикселов в список отсортированных по правой крайней точ- ке серий черных пикселов. Каждая серия помечается ее начальной позицией и длиной. Поддерживается стоимость самого дешевого сопоставления между самыми левыми I сериями и самыми левыми j сериями для всех I и J. Используются следующие операции редактирования: ♦ сопоставление всей серии. Выполняется сопоставление верхней серии i с нижней серией /, при этом стоимость сопоставления будет зависеть от разницы в длине се- рий и их позиций;
314 Часть I. Практическая разработка алгоритмов ♦ слияние серий. Несколько верхних серий сопоставляются с одной нижней серией. Стоимость этой операции будет зависеть от количества сливаемых серий, их отно- сительных позиций и их длины; ♦ разбиение серии. Одна верхняя серия сопоставляется с несколькими последователь- ными нижними сериями. Это всего лишь операция, обратная операции слияния. По- этому стоимость этой операции также будет зависеть от количества серий, их отно- сительных позиций и их длины. — Для каждой пары серий (/, /') и всех применимых случаев мы вычисляем стоимость операции редактирования и добавляем ее к уже вычисленной и сохраненной стоимости редактирования части, расположенной слева от начальной точки редактирования. Са- мый дешевый из этих случаев и будет определять стоимость операции редактирования Ф./]- Аспиранты все это записали в свои блокноты, а потом поинтересовались: — Значит, стоимость сопоставления двух серий будет функцией их длины и позиции. Но как нам определять относительную стоимость? — Это ваше дело. Динамическое программирование можно применить для оптимиза- ции сопоставлений только тогда, когда мы знаем функции стоимости. Вы сами должны решить, какую стоимость устанавливать для операций изменений длины линий и сме- щения их позиций. Я рекомендую реализовать динамическое программирование и по- пробовать разные варианты стоимости на нескольких разных изображениях и выбрать из них такие, которые кажутся отвечающими вашим требованиям. Они переглянулись, улыбнулись и помчались в лабораторию. Применив динамическое программирование для сопоставления частей изображения, они создали свою про- грамму морфинга. Результаты работы этой программы можно видеть на рис. 8.5 на примере превращения омара в человеческое лицо. К сожалению, они так и не удосужились превратить меня в Хамфри Богарта. Рис. 8.5.1 !ревращение омара в человеческое лицо с помощью динамического программирования
Гпава 8 Динамическое программирование 315 8.5. Задача разбиения Допустим, что нужно просмотреть несколько книг на полке, чтобы найти определен- ную информацию. Для выполнения этого задания выделено трое сотрудников, каждо- му из которых будет дана определенная часть книг для просмотра. Чтобы сохранить начальный порядок книг, проще всего разделить полку на три части и назначить по части каждому сотруднику для просмотра. Но как правильно распределить книги среди сотрудников? Если все книги имеют оди- наковое количество страниц, то тогда это сделать очень легко: разделить все книги на три равные части. Например: 100 100 100 | 100 100 100 | 100 100 100 Таким образом, каждому сотруднику достанется три книги с общим количеством в триста страниц. А если количество страниц в книгах не одинаковое? Допустим, что мы распределили таким же образом следующие книги: 100 200 300 | 400 500 600 | 700 800 900 Я лично предпочел бы просматривать первую часть, размером только в 600 страниц, но не последнюю, размером в 2 400 страниц. В этом случае самое справедливое распреде- ление книг было бы следующим: 100 200 300 400 500 | 600 700 | 800 900 Таким образом, самая большая часть будет содержать 1 700 страниц, а самая мень- шая — 1 300. В общем виде задачу можно сформулировать так: ЗАДАЧА. Разделение множества целых чисел на подмножества без перестановок. Вход. Множество S положительных чисел {5|.л,,} и целое число к. Выход. Разделить множество 5 на к или меньше подмножеств таким образом, чтобы максимальная из сумм подмножеств была как можно меньше. Данная задача, называемая задачей линейного разбиения (linear partition), часто возни- кает в параллельных процессах, когда нам нужно распределить нагрузку среди несколь- ких процессоров таким образом, чтобы минимизировать общее время исполнения. Узким местом в таких вычислениях является процессор, которому назначено больше всего работы. В разделе 7.10 было описано неудачное решение такой задачи. Отложите книгу на несколько минут и попробуйте разработать алгоритм для решения задачи линейного разбиения. Для начинающего алгориста наиболее естественным подходом к решению этой задачи может казаться использование эвристического алгоритма. Например, он может вычис- лить средний размер раздела, равный / к. после чего попытаться выполнить разбиение как можно ближе к этому среднему. Но такие эвристические методы реше- ния обречены на провал при некоторых входных экземплярах, т. к. они не выполняют систематическую оценку всех возможных решений.
316 Часть I. Практическая разработка алгоритмов Вместо этого рассмотрим рекурсивный подход исчерпывающего поиска. Обратите внимание на то. что k-й раздел начинается после {к - 1)-го разделителя. Где же мы мо- жем поставить этот последний разделитель? Между z-м и (/ + 1)-м элементами для i та- кого, что 1 < / < п. Каковы будут наши затраты на эту операцию? Общей стоимостью будет большая из двух величин: стоимость последнего раздела или стоимость наибольшего раздела, созданного слева от i. Каким будет размер этого левого раздела? Чтобы минимизировать общую стоимость, нам нужно разделить {s,, .... .v,' элементов, по возможности, на одинаковые части, используя оставшиеся к —2 разделителей. Но ведь это та же самая задача, только в меньшем экземпляре, и. значит, ее можно решить рекурсивно! Исходя из этого, определим М[п, к\ как минимально возможную стоимость по всем операциям разбиения множества {si, .... s„} на к подмножеств, где стоимость разбиения определяется как наибольшая сумма элементов в одной из ее частей. Определив таким образом эту функцию, мы можем ее вычислить: п П М[п,к] = min тах(Лф,£-1]. sj /=<+| Для рекуррентного соотношения необходимо указать граничные условия. Эти гранич- ные условия всегда устанавливают наименьшие возможные значения для всех аргу- ментов. Для данной задачи наименьшим значением первого аргумента будет п = 1, что означает, что первый раздел состоит из одного элемента. Независимо от количества применяемых разделителей, первый раздел нельзя создать меньшим, чем st. Наимень- шим значением второго аргумента будет к = 1, что означает, что множество 5 вообще не разбивается на подмножества. Таким образом: п ЛД1, к] =si для всех£>0и Л/[и, I] = J's, По определению, это рекуррентное соотношение должно возвратить значение опти- мального размера раздела. Сколько времени уйдет на вычисление, если сохранять час- тичные результаты? Всего в таблице имеется кт ячеек. Сколько времени займет вы- числение результата М[п', к']? Для вычисления этого значения требуется найти мини- мальную из п' величин, каждая из которых является максимальной в таблице поиска и суммой, самое большее, п' элементов. Если заполнение каждой из кп ячеек занимает время и", то всю рекуррентность можно вычислить за время О(кп\ Меньшие значения вычисляются раньше больших, с тем, чтобы на каждом шаге вы- числений имелись необходимые данные. Реализация алгоритма приводится в листин- ге 8.16. Листинг 8.16. Алгоритм решения задачи линейного разбиения partition(int s[), int n, int k) { int m[MAXN+l][MAXK+1]; int d[MAXN+1][MAXK+1]; int p[MAXN+1]; /* Таблица значений */ /* Таблица разделителей */ /* Массив префиксных сумм*/
Гпава 8 Динамическое программирование 317 mt cost; /* Стоимость тестового разбиения */ int i,j, х; /* Счетчики */ р[0] - 0; /* Создаем префиксные суммы */ for (1=1; i<=n; i++) р[i]=р[i-1]+s[i]; for (i=l; i<=n; i++) m[i][1] = p[i]; /* Инициализируем границы */ for (j=l; j<=k; j++) m[l] [j] = s[l]; for (i=2; i<=n; i++) /* Вычисляем основное рекуррентное соотношение */ for (j=2; j<=k; j++) ( m[i] [j] = MAXINT; for (x=l; x<=(i-l); x++) { cost = max (m[x] [j-1] , p[i]-p[x]); if (m[i] [j] > cost) { m[i] [j] = cost; d[i] [j] = x; reconstruct_partition(s, d, n, k) ; /* Выводим решение разделения ★/ В действительности реализация из листинга 8.15 работает быстрее, чем мы рассчитали. При первоначальном анализе предполагалось, что обновление каждой ячейки матрицы занимает время Это предположение основано на том обстоятельстве, что мы вы- бираем наилучшую из, самое большее, п возможных точек для размещения разделите- ля, для каждого из которых требуется сумма из, самое большее, п возможных членов. Но в действительности можно с легкостью избежать вычисления этих сумм, сохраняя набор из и префиксных сумм /?[;] = Хл-i5*’ т' \ = ДлНтЯЛ]- Это позволяет вычислять каждую ячейку за линейное время, что дает время исполнения O(kn~). Изучив рекуррентное соотношение и матрицы динамического программирования, представленные в табл. 8.4, вы должны убедиться в том, что конечное значение Л/(», к) будет стоимостью наибольшего диапазона в оптимальном разбиении. Таблица 8.4. Матрицы динамического программирования для двух входных экземпляров м к п 1 2 1 1 1 1 1 2 1 1 1 3 2 1 1 4 2 2 I 5 3 2 1 6 3 2 1 7 4 3 1 8 4 3 1 9 5 3 D к п 1 2 3 1 1 1 1 1 - 1 2 1 - 2 2 1 - 2 3 1 - 3 4 1 - 3 4 1 - 4 5 1 - 4 6 М к п 1 2 3 1 1 1 1 2 3 2 2 3 6 3 3 4 10 6 4 5 15 9 6 6 21 И 9 7 28 15 11 8 36 21 15 8 45 24 17 D к 11 1 2 3 1 2 1 1 3 - 2 2 4 - 3 3 5 - 3 4 6 - 4 5 7 - 5 6 8 - 5 6 9 - 6 7
318 Часть I. Практическая разработка алгоритмов Но для большинства приложений нам требуется информация о том, как фактически выполнять само разбиение. Без этой информации мы, образно выражаясь, имеем лишь купон с большой скидкой на товар, которого нет на складе. Для восстановления оптимального решения используется вторая матрица, D. При каж- дом обновлении значения Лф, /] мы записываем позицию разделителя, которая требо- валась для получения этого значения. Чтобы восстановить путь, который привел к оптимальному решению, мы идем назад от D[n, £] и добавляем разделитель в каждой указанной позиции. Эту обратную трассировку лучше всего выполнять с помощью ре- курсивной процедуры, показанной в листинге 8.17. Листинг 8.17, Рекурсивная процедура восстановления решения reconstruct_partition(int s[],int d[MAXN+1][МАХК+1], int n, int k) I if (k=L) Г int_books(s, 1, n); else reconstruct partition(s, d, dfn][k],k-1); print_books (s, d[n] (k]+l,n); t print_books(int s[], int start, int end) { int 1; /* Счетчик */ for (i=start; i<=end; i++) printf(" d ",s[i]l; printf("\n"); ) 8.6. Синтаксический разбор При компилировании исходного кода программы компилятор определяет, отвечает ли структура программы требованиям синтаксиса данного языка программирования. В случае полного соответствия программа компилируется; в противном случае компи- лятор выдает сообщение об ошибке. Для этого требуется точное описание синтаксиса языка, которое обычно дается контекстно-свободной грамматикой, как показано в примере на рис. 8.6, а. sentence ::= noun-phrase verb-phrase noun-phrase ::= article noun verb-phrase ::= verb noun-phrase article ::= the, a noun cat. milk verb ::= drank a) sentence article noun the cat drank the milk £ Рис. 8.6. Кошекстно-свободная грамматика (а) и дерево синтаксического разбора (б)
Глава 8. Динамическое программирование 319 Каждое правило (production) грамматики определяет интерпретацию именованного символа в левой части правила в виде последовательности символов в левой части правила. Правая сторона правила может содержать сочетание нетерминальных симво- лов, или просто нетерминалов, (которые также определяются правилами) или терми- нальных символов, которые определяются просто как строки, например "the", "a", "cat", "milk" и "drank". Синтаксический разбор (parsing) текстовой строки S согласно правилам контекстно- свободной грамматики G представляет собой алгоритмическую задачу создания дерева синтаксического разбора правил замены, определяющего строку 5 в виде единого не- терминального символа грамматики G. На рис. 8.6, б показано дерево синтаксического разбора простого предложения согласно правилам грамматики, представленной на рис. 8.6. а. В студенческие времена синтаксический разбор казался мне ужасно сложным предме- том. По несколько лет тому назад один приятель без труда объяснил мне его за ланчем. Правда теперь я понимаю динамическое программирование намного лучше. Мы полагаем, что длина текстовой строки S равна п символам, а сама грамматика G имеет постоянный размер. Такое предположение является справедливым, т. к., незави- симо от размера компилируемой программы, грамматика определенного языка про- граммирования (например, С или Java) имеет фиксированный размер. Кроме этого, мы полагаем, что определения каждого правила представлены в нор.мачь- ной форме Хомского (Chomsky normal form). Это означает, что правая сторона каждого нетривиального правила состоит из двух нетерминальных символов, т. е. A"—» YZ. или одного терминального символа, т. е. X—♦ а. Любую контекстно-свободную грамматику можно механически преобразовать в нормальную форму Хомского, многократно со- кращая длинные правые стороны за счет добавления дополнительных нетерминальных символов и правил. Таким образом, данное предположение не приводит к потери общ- ности. Как можно выполнить эффективный синтаксический разбор строки используя кон- текстно-свободную грамматику, в которой каждое представляющее интерес правило состоит из двух нетерминальных символов? Здесь ключевым является то обстоятельст- во. что правило, применяемое в корне дерева синтаксического разбора (скажем. А—> )’Z), разделяет строку X в позиции I таким образом, что левая часть строки (S[l. /]) генерируется нетерминальным символом У, а правая часть (S[/ + I, и]) генерируется элементом Z. Это обстоятельство наводит на мысль об использовании динамического программиро- вания. в котором мы отслеживаем все нетерминальные символы, генерируемые каждой подстрокой строки S. Определяем булеву функцию Аф‘, j. А], которая во «вращает зна- чение ИСТИНА тогда и только тогда, когда подстрока 5[/.у] генерируется нетерми- нальным символом .V. Это происходит тогда, когда существует правило X —♦ KZ и такая точка разрыва к между i и /, что левая часть генерирует Y, а правая — Z. Иными словами, M[i,j.X] = V (V M[i,k.Y]M[k + \,j.Z]'), (.V->E?)eG ,=*
320 Часть I. Практическая разработка алгоритмов где символ V означает логическое ИЛИ по всем правилам и позициям разделения, а символ означает операцию логического И над двумя булевыми значениями. Однобуквенные терминальные символы определяют граничные условия рекуррентно- сти. В частности, M[i, i, .¥] определяется как ИСТИНА тогда и только тогда, когда су- ществует такое правило X—>а, что Л’[/] = а. Какова временная сложность этого алгоритма? Размер пространства состояний равен О(п ), т. к. существует п(п + 1)/2 подстрок, определяемых парами Умножение это- го значения на количество нетерминальных символов (скажем, v) не влияет на асим- птотический верхний предел, т. к. грамматика определена постоянного размера. Чтобы вычислить значение M[i, j. А], нужно протестировать все промежуточные значения к, так что вычисление каждой из О(п2} ячеек занимает время О(п). Отсюда получаем ку- бическую, т. е. О(п}, временную сложность алгоритма синтаксического разбора. Остановка для размышлений. Экономичный синтаксический разбор ЗАДАЧА. Нередко в программах имеются синтаксические ошибки, из-за которых про- грамма не компилируется. Для данной контекстно-свободной грамматики G и строки ввода S определить наименьшее количество замен символов, которые нужно выпол- нить в строке S. чтобы конечная строка была приемлемой для грамматики G. Решение. Когда я впервые столкнулся с этой задачей, она казалась чрезвычайно труд- ной. Но после некоторых размышлений я пришел к выводу, что она является общим случаем задачи вычисления расстояния редактирования, для решения которой динами- ческое программирование подходит самым естественным образом. Синтаксический разбор также вначале казался трудной задачей, но потом поддался решению тем же методом. Действительно, мы можем решить объединенную задачу, обобщив отноше- ние рекуррентности, которое мы использовали для выполнения простого синтаксиче- ского разбора. Определим целочисленную функцию M'[i, j, А], которая возвращает количество мини- мальных изменений в строке Л’[/, /], позволяющих создать ее нетерминальным симво- лом X. Этот символ будет создаваться правилом х —► yz. Некоторые изменения под- строки .у могут находиться слева от точки раздела, а другие — справа, но нас интересу- ет только минимизация суммы. Иными словами, A/'[z, j.X] = min (mmM'[i,k,Y] +M'[k + 1, /’,Z]) (A'->)'Z)eG i=k Также слегка изменяются граничные условия. Если существует правило X —♦ а. то стоимость сравнения в позиции i зависит от содержимого <S[z], где *$”[/] = a, M[i, i, А] = 0. В противном случае требуется одна замена, поэтому M[i, i. А] = 1, если 5[/] а. Но если в грамматике отсутствует правило X —► а, то не существует способа выполнить замену в односимвольной строке, чтобы получить что-либо, создающее X, поэтому M[i, i. А] = со для всех /. 8.6.1. Триангуляция с минимальным весом То же самое рекуррентное соотношение, используемое в алгоритме синтаксического разбора, можно также использовать для решения интересной задачи вычислительной
Гпава 8. Динамическое программирование 321 геометрии. Триангуляцией многоугольника Р = {vH v„, V|} называется набор непере- секающихся диагоналей, которые разделяют данный многоугольник на треугольники. Весом триангуляции называется сумма длин всех ее диагоналей. Как показано на рис. 8.7, для любого многоугольника может существовать несколько разных триангу- ляций. Рис. 8.7. Лис разные триангуляции для выпукло! о семиугольника Нам нужно найти для данного многоугольника р триангуляцию с минимальным весом. Триангуляция является базовым компонентом большинства геометрических алгорит- мов (ем. раздел 17.3). Чтобы применить динамическое программирование для решения этой задачи, нам нужно найти способ разбить многоугольник на меньшие части. Обратите внимание на то, что каждая сторона многоугольника должна входить ровно в один треугольник. Чтобы создать треугольник на основе этой стороны, необходимо определить третью вершину этого треугольника, как показано на рис. 8.8. Рис. 8.8. Выбор вершины А для стороны (i.J) многоугольника Когда мы определим правильную объединяющую вершину, многоугольник будет раз- бит на два меньших многоугольника, для каждого из которых нужно будет выполнить оптимальную триангуляцию. Пусть T[i, j] обозначает стоимость триангуляции от вер- шины v, к вершине v, без учета длины хорды d„ между v, и vy. Последнее обстоятельство позволяет избежать двойного подсчета этих внутренних хорд в следующем рекуррент- ном соотношении: Г[/, 7] = rnin(7’[z, £] + T[k, j] + d,k + dk)) Это основное условие применимо, когда вершины i и j являются непосредственными соседями, т. к. 7]7, i + 1] = 0. Поскольку количество вершин в каждом поддиапазоне правой части отношения мень- ше. чем в левой части, вычисление может выполняться на интервале между i и j. Псев- докод соответствующего алгоритма приводится в листинге 8.18. 11 Зак .3741
322 Часть I. Практическая разработка алгоритмов < ..................... .......... ................ Листинг 8.18. Алгоритм триангуляции .............................. ........ Л. Minimum-Weight-Triangulation (Р) for 1 = 1 to n-1 do T[i, i + l] = 0 for gap = 2 to n 1 for j = 1 to n - gap do j = i + gap T[i, j] = min,. '+I(T[1, k] + T[k, j] + dp p + dp) return T[l, n] Триангуляция T имеет (2) значений, вычисление каждого из которых занимает время O{j - z), если вычислять секции в порядке возрастания размера. Так как/- / = (?(«). то сложность по времени полного вычисления будет О(п\ а по памяти — О(п~). Что произойдет, если точки имеются внутри многоугольника? В таком случае динами- ческое программирование нельзя применить таким же образом, т. к. ребра триангуля- ции не обязательно разбивают многоугольник на две разные части. Теперь вместо только возможных поддиапазонов их количество возрастет экспоненциально. Бо- лее того, известно, что общий случай данной задачи является NP-полной. Подведение итогов Для любой задачи оптимизации объектов, упорядоченных слева направо, например, сим- волов строки, элементов перестановки, вершин многоугольника или листьев дерева поис- ка, применение динамического программирования, скорее всего, позволит создать эффек- тивный алгоритм для получения оптимального решения. 8.7. Ограничения динамического программирования. Задача коммивояжера Динамическое программирование подходит не для всех задач. Важно понимать, поче- му его применение может не дать желаемого результата, и уметь избегать ситуаций, чреватых появлением неправильного или неэффективного алгоритма. Материалом для наших алгоритмических экспериментов снова будет задача комми- вояжера, в которой мы ищем самый короткий маршрут для посещения всех городов. Но мы ограничимся следующим случаем: ЗАДАЧА. Самый длинный простой путь. Вход. Взвешенный граф G, в котором указаны начальная (,v) и конечная (/) вершины. Выход. Самый длинный маршрут от 5 до t, в котором все вершины посещаются только один раз. Между этой задачей и задачей коммивояжера имеется два несущественных различия. Во-первых, требуется найти путь, а не замкнутый маршрут. Эта разница несуществен- на, т. к. мы можем получить замкнутый маршрут, просто включив в путь ребро (/, s). Во-вторых, требуется найти наиболее длинный путь, а не наиболее короткий маршрут.
Гпава 8. Динамическое программирование 323 Опять же. разница не является существенной: нам просто требуется посетить как мож- но больше вершин (в идеале, все), точно так же. как и в задаче коммивояжера. Ключе- вым словом в постановке данной задачи является слово простой, означающее, что лю- бую вершину мы можем посетить только один раз. Для невзвешенных графов (в которых вес каждого ребра равен единице) самый длин- ный путь от \ к t будет равным п- 1. Задача поиска таких Гамильтоновых путей (если они существуют) является важной задачей теории графов, которая рассматривается в разделе 16.5. 8.7.1. Вопрос правильности алгоритмов динамического программирования Алгоритмы динамического программирования правильны лишь настолько, насколько правильны рекуррентные соотношения, на которых они основаны. Допустим, мы опре- делим функцию LP[i,j] как длину максимального простого пути от вершины i к вер- шине j. Обратите внимание на то, что самый длинный простой путь от вершины i к вершине j перед тем. как попасть в вершину j, должен пройти через некую вершину х. Таким образом, последним ребром пути должно быть ребро (х, j). Здесь напрашивается следующее рекуррентное соотношение для вычисления длины максимального пути, где с(х, j) — вес ребра (х, /): LP[i,j] = max £P[z,x] + c(x, j) (x.J)el- Идея кажется разумной, однако я вижу в ней, по крайней мере, два недостатка. Прежде всего, в этом рекуррентном соотношении не предусмотрено ничего для обес- печения простоты. Откуда мы знаем, что вершина j не использовалась ранее в самом длинном простом пути от вершины i к вершине х? Если использовалась, то добавление ребра (х, /) создаст цикл. Чтобы предотвратить такое развитие событий, мы должны определить такое рекуррентное соотношение, которое помнит пройденный путь. Воз- можно. мы могли бы достичь этого, определив функцию LP'\i,j, А:],как самый длинный путь от вершины i к вершине j, не включающий вершину А:? Это был бы шаг в пра- вильном направлении, но он все равно не даст нам работающее должным образом ре- куррентное соотношение. Вторая проблема затрагивает порядок вычисления. Какой элемент мы вычисляем пер- вым? Так как вершины графа не упорядочены слева направо или в порядке возраста- ния. то неясно, какими должны быть меньшие подпрограммы. При отсутствии такого упорядочивания мы легко окажемся в бесконечном цикле. Динамическое программирование можно использовать для решения любой задачи, в которой соблюдается принцип оптимальности. Иными словами, это означает, что множество частичных решений может быть оптимально расширено с учетом состоя- ний после частичных решений, а не специфики самих частичных решений. Например, когда мы решали, расширять ли нечеткое сравнение строк за счет замены, вставки или удаления, нам не требовалось знать, какая последовательность операций была выпол- нена до этого момента. В действительности может существовать несколько разных по- следовательностей редактирования, которые выдают стоимость С на первых р симво-
324 Часть I. Практическая разработка алгоритмов лах строки образца Р и / символах строки Т. Очередные решения принимаются на основе последствий предыдущих решений, а не на основе самих предыдущих ре- шений. Когда важна не стоимость выполняемых операций, а их специфика, задача не удовле- творяет принципу оптимальности. В качестве примера такой задачи можно привести разновидность задачи вычисления расстояния редактирования, в которой не разреша- ется использовать комбинации операций в определенных последовательностях. Но при правильной постановке задачи принцип оптимальности соблюдается во многих комби- наторных задачах. 8.7.2. Эффективность алгоритмов динамического программирования Время исполнения любого алгоритма динамического программирования зависит от двух факторов: количества частичных решений, которые необходимо отслеживать, и длительности вычисления каждого частичного решения. Обычно более важным явля- ется первый аспект, а именно, размер пространства состояний. Во всех приведенных здесь примерах частичные решения полностью описываются по- средством указания точек остановки при вводе. Это объясняется тем. что элементы обрабатываемых комбинаторных объектов (строк, числовых последовательностей или многоугольников) неявно упорядочены. Это упорядочивание нельзя нарушить, не из- менив при этом полностью всю задачу. При установленном порядке элементов сущест- вует сравнительно небольшое количество возможных точек остановки, что позволяет получить эффективные алгоритмы. В случае же отсутствия жесткой упорядоченности объектов количество возможных частичных решений растет экспоненциально. Допустим, что состоянием нашего час- тичного решения является весь путь Р от начальной до конечной вершины. Таким об- разом. LP[i,j, Р] представляет самый длинный простой путь от вершины i к вершине j. где Р обозначает точную последовательность промежуточных вершин в этом пути. Это можно вычислить с помощью следующего рекуррентного соотношения, в котором Р + х обозначает присоединение вершины х в конец пути Р: LP[i, j, Р + х] = max LP[i, х, Р] + с(х, /) Эта формулировка корректна, но насколько она эффективна? Путь Р является упорядо- ченной последовательностью из не более и —3 вершин. Количество таких путей дохо- дит до (и — 3)!, а это очень много! Фактически, этот алгоритм использует комбинатор- ный поиск (наподобие поиска с возвратом) для создания всех возможных промежуточ- ных путей. В действительности, функция max отчасти вводит в заблуждение, г. к. для создания состояния LP[i, j, Р + х] возможно только одно значение х и одно значение Р. Но этой идее можно найти лучшее применение. Пусть LP'\i,j, Р] представляет самый длинный простой путь от вершины i к вершине j, промежуточными вершинами которо- го являются как раз те. что образуют подмножество S. Таким образом, если S= {а, Ь, с}, то существует шесть путей, совместимых с Р: iabcj, iacbj, ibacj. ibcaj. icabj и icbaj. Размер этого пространства состояний может быть максимум 2". что меньше.
Гпава 8. Динамическое программирование 325 чем перечисление всех путей. Кроме этого, эту функцию можно вычислить с помощью следующего рекуррентного соотношения: LP'[i, j,S u {х}] = max LP[i,x,S] +c{x,j), где 5u {x} обозначает объединение S и x. После этого самый длинный простой путь от вершины i к вершине / можно найти, мак- симизировав по всем возможным промежуточным подмножествам вершин: LP\i, j] = max LP'[i, у, .S’] Существует только 2" подмножеств множества п вершин, так что это большое улучше- ние по сравнению с перечислением всех /?! маршрутов. На практике этот метод можно было бы с уверенностью использовать для решения задачи коммивояжера с количест- вом вершин, близким к 30, в то время как значение п = 20 будет неприемлемым для алгоритма с временной сложностью О(«!). Тем не менее, динамическое программиро- вание наиболее эффективно на множестве хорошо упорядоченных объектов. Подведение итогов При отсутствии естественного упорядочивания слева направо алгоритм, использующий динамическое программирование, обычно обречен на экспоненциальную сложность, как по времени, так и по памяти. 8.8. История из жизни. Динамическое программирование и язык Prolog — Наш эвристический алгоритм очень хорошо проявляет себя на практике.— Мой коллега одновременно хвастался и просил о помощи. Унификация является основным вычислительным механизмом в языках логического программирования, таких как Prolog. Программа на языке Prolog состоит из набора правил, где каждое правило имеет голову и действие, которое выполняется, когда го- лова правила совпадает или объединяется с текущим вычислением. Выполнение программы на языке Prolog начинается с указания цели, например, р(а.Х, У), где а является константой, а X и Y— переменными. После этого система систематически сравнивает голову цели с головой каждого правила, которое можно унифицировать с целью. Под унификацией имеется в виду привязка переменных к кон- стантам, если их можно соотнести. В представленной в листинге 8.19 бессмысленной программе на языке Prolog цельр(Х, У, а) унифицируется с любым из первых двух пра- вил. т. к. для X и У можно выполнить привязку, чтобы сопоставить дополнительные символы. Цель р(Х, X, а) сопоставится только с первым правилом, т. к. привязываемые к первой и второй позиции переменные должны быть одинаковыми. Листинг 8.19. Пример программы на языке Prolog р(а,а,а) : = h(а) ; p(b,a,a) := h(а) h(b) ; p(c,b,b) :=h(b) + h(c); p(d,b,b) :=h(d) + h(b);
326 Часть I. Практическая разработка алгоритмов — Чтобы ускорить операции унификации, мы хотим выполнить предварительную об- работку набора голов правил, чтобы можно было бы быстро определить, какие правила соответствуют данной цели. Для этого нам нужно организовать правила в нагруженном дереве (trie). Нагруженные деревья (см. раздел 12.3) являются полезными структурами данных для работы со строками. Каждый лист нагруженного дерева представляет одну строку. Каждый узел на пути от корня к листу маркируется одним символом строки, при этом /-Й узел пути соответствует /-му символу строки. — Согласен. Нагруженное дерево является естественным способом для представления ваших голов правил. Создание нагруженного дерева для набора строк символов явля- ется прямолинейной задачей: мы просто вставляем строки, начиная с корня. Так в чем заключается ваша проблема? — спросил я. — Эффективность нашего алгоритма унификации очень зависит от минимизирования количества ребер в нагруженном дереве. Так как мы знаем все правила наперед, то у нас имеется свобода действий, чтобы переупорядочить позиции символов в правилах. Например, вместо того, чтобы корневой узел всегда представлял первый аргумент пра- вила. мы можем избрать, чтобы он представлял третий аргумент. Мы бы хотели вос- пользоваться этой свободой действий для того, чтобы создать нагруженное дерево ми- нимального размера для набора правил. Мой собеседник показал мне пример (рис. 8.9). Рис. 8.9. Два разных нагруженных дерева для одного и того же набора правил В нагруженном дереве, созданном согласно первоначальному порядку позиций строк (1, 2, 3), всего используется 12 ребер. Но переставив позиции символов в (2, 3, 1), мы можем получить нагруженное дерево лишь с 8 ребрами. — Интересно... — начал было я отвечать, но он снова прервал меня. — Есть еще одно ограничение. Листья нагруженных деревьев необходимо содержать упорядоченными, чтобы листья основного дерева располагались слева направо в том же самом порядке, в каком правила выводятся на страницу.
Гпава 8. Динамическое программирование 327 — Но почему листья нагруженных деревьев должны содержаться в заданном поряд- ке? — спросил я. — Порядок правил в программах Prolog имеет очень, очень большую важность. Если изменить порядок правил, то программа возвратит другой результат. Потом он перешел к описанию того, что им требовалось от меня. — У нас есть "жадный"” эвристический алгоритм для создания хороших, но не опти- мальных, нагруженных деревьев. Этот алгоритм основан на выборе в качестве корня дерева символа в такой позиции, которая минимизирует степень корня. Иными слова- ми, алгоритм выбирает такую позицию символа, в которой имеется наименьшее число разных символов. Этот эвристический алгоритм работает исключительно хорошо на практике. Но нам нужно доказать, что задача построения наилучшего нагруженного дерева является NP-полной, чтобы наша статья была полностью завершена. Вот в этом нам и нужна ваша помощь. Я пообещал доказать, что задача имеет такой уровень трудности. Казалось, что для по- строения минимального дерева, в самом деле, требовалось использовать какую-то не- тривиальную комбинаторную оптимизацию, но я не видел, каким образом можно было бы включить слева направо упорядочивание правил в доказательство сложности. Более того, я не мог припомнить ни одной NP-полной задачи, содержащей такое ограничение в виде упорядочивания слева направо. В конце концов, если бы данный набор правил содержал позицию символа, общую для всех правил, то она должна бы быть исследо- ванной первой в любом дереве минимального размера. Так как правила были упорядо- ченными, то каждый узел в поддереве должен представлять корень серии последова- тельных правил. Таким образом, существовало только (2) узлов, которые было воз- можно выбрать из этого дерева... Есть! Это и было ключевым аспектом решения. На следующий день я снова встретился с профессором. — Я не могу доказать, что ва- ша задача является NP-полной. Но что Вы скажете по поводу эффективного алгоритма динамического программирования для построения наилучшего нагруженного дере- ва?— Я с удовольствием наблюдал, как недовольное выражение его лица сменилось улыбкой, когда он осознал, о чем идет речь. Эффективный алгоритм для получения требуемого решения было значительно лучше, чем доказательство невозможности та- кого решения. Мое рекуррентное соотношение работало таким образом. Допустим, что у нас имеется и упорядоченных голов правил, каждая из которых имеет т аргументов. Выборка в /?-й позиции, 1 <р< т, разделяет головы правил на серии R\, ..., Rr, где каждое правило в отдельной серии Rx = s(, .... s, имеет одинаковое значение символа х,|/>]. Правила в каж- дой серии должны быть последовательными, поэтому существует только (^возмож- ных серий, о которых нужно заботиться. Стоимостью выборки в позиции р является стоимость завершения обработки деревьев, формируемых каждой созданной серией, плюс одно ребро для каждого дерева, чтобы связать его с зондированием р\ т г С[г, j] = min £(C[ik ,jk] +1) *=i
328 Часть I. Практическая разработка алгоритмов Один из аспирантов немедленно приступил к реализации этого алгоритма, чтобы мож- но было сравнить его работу с работой эвристического алгоритма заказчика. На многих входных экземплярах как оптимальный, так и "жадный" алгоритмы создавали одина- ковое нагруженное дерево. Но для некоторых экземпляров производительность алго- ритма динамического программирования была на 20% лучше, чем производительность "жадного" алгоритма, т. е. на 20% лучше, чем "очень хорошая работа" на практике. Время компилирования алгоритма динамического программирования было несколько большим, чем "жадного" алгоритма, но при оптимизации компилирования всегда луч- ше обменять немного большее время компиляции на лучшее время исполнения про- граммы. Стоит ли 20% улучшения производительности этих усилий? Это зависит от конкретной ситуации. Насколько полезной была бы для вас 20-процентная надбавка в вашей зарплате? То обстоятельство, что правила должны были оставаться упорядоченными, было ре- шающим фактором, который мы использовали в решении с применением динамиче- ского программирования. Фактически при отсутствии этого обстоятельства я смог до- казать, что задача в самом деле была NP-полной. Подведение итогов Глобальное оптимальное решение (полученное, например, с помощью динамического программирования) часто заметно лучше, чем решение, полученное посредством типич- ного эвристического алгоритма. Насколько важным является такое улучшение, зависит от конкретного приложения, но оно никогда не будет лишним. 8.9. История из жизни. Сжатие текста для штрих-кодов Инджиун (Ynjiun) провел лазерной указкой по рваным и смятым кускам этикетки со штрих-кодом. Через несколько секунд система выдала ответ Он победно усмехнул- ся. — Практически безотказно. Мне показывали свои достижения работники научно-исследовательского центра ком- пании Symbol Technologies, ведущего мирового производителя оборудования для ска- нирования штрих-кодов. В следующий раз, когда вы будете стоять в очереди к кассиру в продуктовом магазине, обратите внимание на тип используемого ими сканирующего оборудования. Скорее всего, на корпусе будет маркировка Symbol. Хотя мы принимаем штрих-коды как должное, работа с ними требует на удивление сложной технологии. Надобность в штрих-кодах существует потому, что обычные сис- темы оптического распознавания символов не обеспечивают достаточной надежности для операций с товарно-материальными запасами. Технология штрих-кодов, знакомая каждому из нас по их присутствию на каждой пачке овсянки и упаковке жевательной резинки, позволяет закодировать десятизначный номер с уровнем коррекции, делаю- щим практически невозможными ошибки при сканировании, даже если консервная банка со штрих-кодом перевернута вверх ногами или деформирована. Иногда кассиру не удается отсканировать штрих-код. но если вы слышите характерный звук аппарата, то вы знаете, что код был считан правильно.
Гпава 8 Динамическое программирование 329 Десять цифр кода обычной этикетки со штрих-кодом предоставляют возможность за- писать на ней только идентификационный номер товара. Вследствие этого любое при- ложение, занимающееся обработкой штрих-кодов, должно использовать базу данных, соотносящую штрих-код с соответствующим товаром. В течение долгого времени за- ветной целью штрих-кодовой индустрии была разработка более емкой штрих-кодовой символики, позволяющей кодировать целые документы и обеспечивающей их надеж- ное воспроизведение. -PDF-417 является нашей новой, двумерной штрих-кодовой технологией,— объяс- нил Инджиун. (Пример этикетки со штрих-кодом этого типа показан на рис. 8.10.) — Какой объем информации можно уместить с помощью этой технологии на одном квадратном дюйме? — спросил я у него. — Это зависит от заданного уровня коррекции ошибок, но в общем около 1 000 бай- тов, что достаточно для небольшого текстового файла или изображения, — ответил он. — Вам, наверное, приходится использовать какую-либо технологию сжатия данных, чтобы максимизировать объем сохраняемого на этикетке текста. (Стандартные алго- ритмы сжатия данных рассматриваются в разделе 18.5.) — Да. мы действительно используем определенный способ сжатия данных.— согла- сился Инджиун.— Мы знаем, какие типы текста наши клиенты будут помешать на этикетки. Некоторые тексты будут состоять полностью из прописных букв, а другие — из букв обоих регистров и цифр. Наш код предоставляет три разных текстовых режи- ма, каждый из которых поддерживает отдельное подмножество алфавитно-цифровых знаков. Мы можем описать каждый знак, используя только пять битов, при условии, что режим не меняется. Для описания знака из другого режима мы сначала выдаем команду переключения режимов (длиной в пять битов), а потом код символа. — Понятно. Значит, вы определили для каждого режима группы символов таким обра- зом. чтобы свести к минимуму операции переключения режимов при работе с типич- ными текстовыми файлами. (Схема переключения режимов для разных групп симво- лов показана на рис. 8.11.) Рис. 8.10. Этикетка с Геггисбсргским посланием, закодированным двумерным штрих-кодом по технологии PDF-417 Рис. 8.11. Схема переключения режимов в PDF-4I7
330 Часть I. Практическая разработка алгоритмов — Совершенно верно. Мы поместили все цифры в один режим, а все символы знаков препинания в другой. Мы также реализовали команды переключения (switch) и фикси- рования (latch) режимов. Операция переключения режимов применяется для переклю- чения в новый режим только для следующего символа, скажем, знака препинания. Таким образом мы можем избежать затрат на возвращение в текстовый режим после вывода знака препинания. А операция фиксирования режимов используется для посто- янного переключения в новый режим, если в этом режиме нужно вывести несколько символов подряд. — Интересно! — сказал я. — При таком переключении режимов должно существовать множество разных способов закодировать текст в виде штрих-кода. Как вы получаете закодированный текст наименьшего объема? — Мы используем "жадный" алгоритм, который выполняет просмотр на несколько символов вперед и решает, какой режим лучше всего использовать. Этот способ рабо- тает довольно хорошо. Я продолжал выпытывать у него подробности по этому вопросу. — Откуда вы знаете, что этот способ работает хорошо? Ведь может существовать значительно лучшая ко- дировка, которую вы просто не находите. — Вообще говоря, мы этого не знаем. Но поиск оптимальной кодировки, вероятно, является NP-полной задачей. Не так ли? — спросил он упавшим голосом. Я задумался. Каждая кодировка начинается в определенном режиме и состоит из по- следовательности кодов символов и операций переключения и фиксирования режимов. В любой позиции в тексте мы можем помещать код следующего символа (если он име- ется в текущем режиме) или выполнять операцию переключения или фиксирования режима. При продвижении в тексте слева направо текущее состояние полностью отра- жается текущей позицией символа и текущим состоянием режима. Для данной пары "позиция/режим" нас интересует самая дешевая из всех возможных кодировок дости- жения этой точки.. Я почувствовал сильное волнение. — Оптимальную кодировку для любого текста в PDF-417 можно определить с по- мощью динамического программирования. Для каждого возможного режима 1 < т < 4 и для каждой возможной позиции символа 1 < i <п мы будем сопровождать информа- цию о самой дешевой кодировке, найденной для первых i символов и заканчивающей- ся в режиме т. Нашим следующим ходом из каждой пары состояний "позиция/режим" является или сопоставление, или переключение или фиксирование режима, так что рассмотрению подлежит очень небольшое число возможных операций. По сути, мы имеем следующее соотношение: M[i, /] = min(M[/- l,/«] + c(S,,m,/)), где c(S„ m,J) представляет стоимость кодирования символа S, и переключения из ре- жима т в режим j. Самая дешевая возможная кодировка получается в результате об- ратной трассировки от узла Л/[и, т], где т представляет значение /. которое минимизи- рует min, ,<4ЛТ[«, /]. Каждую из 4и ячеек можно заполнить за постоянное время, так
Глава 8. Динамическое программирование 331 что оптимальную кодировку можно найти за линейное по отношению к длине строки время. Инджиун отнесся к этому несколько скептически, но попросил, чтобы мы реализовали для него оптимальный метод кодирования. У нас возникли определенные затруднения в связи с некоторыми странностями переключения режимов, но мой студент Ио-Линг Лин (Yaw-Ling Lin) оказался на высоте и решил эту задачу. В Symbol сравнили наш кодировщик со своим, обработав для этого 13 000 этикеток, и пришли к выводу, что динамическое программирование, в среднем, позволяло поместить на этикетку на 8% больше информации. Это было важным улучшением, т. к. никому не хочется терять 8% емкости хранения, особенно в ситуации, когда максимальная емкость хранения состав- ляет всего лишь несколько сот байтов. Для определенных приложений эти лишние 8% позволили использовать только одну этикетку, когда раньше требовалось две. Конечно же, среднее улучшение на 8% означало, что для определенных типов информации улучшение будет намного большим. В то время как наш кодировщик работал чуть медленнее, чем кодировщик на базе "жадного" алгоритма, это не представляло важно- сти, т. к. в любом случае распечатка этикеток была бы узким местом. Наблюдаемый нами эффект замены эвристического алгоритма алгоритмом поиска гло- бального оптимального решения, пожалуй, является типичным для большинства при- ложений. В общем, если вы не допустите серьезных ошибок в реализации эвристиче- ского алгоритма, то, скорее всего, он будет выдавать приемлемое решение. Но исполь- зование вместо него алгоритма поиска оптимального результата обычно дает хоть и небольшое, но немаловажное улучшение, которое может иметь положительный эффект для вашего приложения. Замечания к главе Методика динамического программирования впервые представлена в [Ве!58]. Алго- ритм определения расстояния редактирования первоначально был представлен Вагне- ром и Фишером (Wagner, Fischer) в книге [WF74], Более быстрый алгоритм для задачи разделения книг рассматривается в книге [KMS97]. Вычислительная сложность триангуляции с минимальным весом несвязного набора точек (в отличие от многоугольников) долго оставалась нерешенной задачей, пока, на- конец, не была решена Мулцером и Роте (Mulzer. Rote) (см. книгу [MR06]). Такие методики, как динамическое программирование и поиск с возвратом, можно ис- пользовать для создания эффективных (хотя все же не с полиномиальным временем исполнения) алгоритмов для решения многих NP-полных задач. Обзор этих методик см. в книге [Woe03]. Система морфинга, которая была описана в истории из жизни в разделе 8.4, более под- робно обсуждается в книге [HWK94], Дополнительную информацию по задаче мини- мизации нагруженных деревьев в Prolog, которая рассматривалась в разделе 8.8, можно найти в нашем докладе [DRR + 95]. Двумерные штрих-коды, рассматриваемые в разде- ле 8.9, были разработаны в значительной степени усилиями Тео Павлидиса (Theo Pavlidis) и Инджиуна Ванга (Ynjiun Wang) в университете Stoney Brook. Подробности см. в [PSW92],
332 Часть I. Практическая разработка алгоритмов Алгоритм динамического программирования, рассматриваемый для решения задачи синтаксического анализа, называется CKYalgonthm, по именам его трех независимых разработчиков: Коки, Касами и Янгера (Cocke, Kasarni, Younger). Подробности см. в [You67], Обобщение синтаксического анализа для задачи вычисления расстояния ре- дактирования рассматривается Ахо и Петерсоном (Aho, Peterson) в [АР72]. 8.10. Упражнения Расстояние редактирования I. [3] При вводе текста часто допускаются ошибки перестановки соседних символов, на- пример, когда вместо "Steve" вводится "Setve". Согласно традиционному определению расстояния редактирования для исправления таких ошибок требуется две замены. Включите в нашу функцию расстояния редактирования операцию обмена, чтобы такие ошибки перестановки можно было бы исправить за одну операцию. 2. [4] Дано три строки символов: X, У и Z, где |Л] = п, |У| = т и |Z| = л + т. Строка Z назы- вается перетасовкой (shuffle) строк А’ и У тогда и только тогда, когда ее можно создать, перемешивая символы строк X и У таким образом, чтобы в получившейся строке сохра- нялся первоначальный (слева направо) порядок исходных строк. а) Докажите, что строка "cchocohilaptes" является перетасовкой строк "chocolate" и "chips", но строка "chocochilatspe" — нет. б) Предоставьте эффективный алгоритм динамического программирования для опреде- ления, является ли строка Z перетасовкой строк X и К (Подсказка: значения создаваемой вами матрицы динамического программирования должны быть булевыми, а не числовы- ми ) 3. [4] Самой длинной общей подстрокой (не подпоследовательностью) двух строк X и У является самая длинная строка, которая входит в виде серии последовательных символов в обе строки Например, самой длинной обшей подстрокой строк photograph и tomography является строка ograph. а) Пусть и = |Л] и т- |У]. Предоставьте алгоритм динамического программирования с временной сложностью &(пт) для поиска самой длинной общей подстроки, основанный на алгоритме поиска самой длинной общей подпоследовательности или вычисления рас- стояния редактирования. б) Предоставьте более простой алгоритм с временной сложностью &(пт). в котором не используется динамическое программирование. 4. [6] Самой длинной общей подпоследовательностью (longest common subsequence, LCS) двух последовательностей Г и Г называется такая самая длинная последовательность L, которая является подпоследовательностью как последовательности Г, так и после- довательности Р. Кратчайшей общей надпоследовательностью (shortest common super- sequence, SCS) последовательностей T и Р называется такая кратчайшая последователь- ность L, подпоследовательностями которой являются как последовательность Т, так и последовательность Р. а) Предоставьте эффективный алгоритм поиска LCS и SCS для двух последовательно- стей. б) Пусть d(T, Р) будет минимальным расстоянием редактирования между строками Т и Р при условии, что замены запрещены, т. е. разрешены только вставка и удаление симво-
Глава 8 Динамическое программирование 333 лов. Докажите, что d(T, Р) = |SGS’(7'. /J)| - |Z.SC(Т, Р)|, где \SCS(T, Р)\ и |/.\С(7', ^)1 обозна- чают соответственно размер кратчайшей обшей надпоследовательности и самой длин- ной общей подпоследовательности Т и Р. "Жадные" алгоритмы 5. [4] Пусть Р}, Р2, ... Р„ — это п программ, которые нужно сохранить на диске емкостью D мегабайт. Для хранения программы Р, требуется s, Мбайт дискового пространства. Со- хранить все программы на диске нельзя, т. к. D < ,5, • а) Можно ли максимизировать количество сохраняемых на диске программ с помощью "жадного" алгоритма, который выбирает программы в порядке неубывающего дискового пространства Л Предоставьте доказательство или контрпример. б) Можно ли сказать, что "жадный" алгоритм, выбирающий программы в порядке не- убывающего дискового пространства s„ использует наибольшее возможное дисковое пространство? Предоставьте доказательство или контрпример. 6. [5] В Соединенных Штатах используются монеты достоинством в 1, 5, 10, 25 и 50 цен- тов. Теперь рассмотрим страну, в которой используются монеты достоинством в {d},..., dk} единиц. Нам нужен алгоритм, который дает сдачу размером в п единиц, ис- пользуя наименьшее количество монет этой страны. а) При решении этой задачи "жадный" алгоритм многократно выбирает монету самого большого достоинства, не превышающую текущий размер сдачи, до тех пор. пока размер сдачи не станет равен нулю. Докажите, что "жадный" алгоритм не всегда выбирает наи- меньшее количество монет для страны с достоинством монет в {1, 6, 10} единиц. б) Предоставьте эффективный алгоритм, который правильно определяет наименьшее ко- личество монет, которое требуется, чтобы дать сдачу стоимостью в п единиц, используя монеты достоинством в {б/ь ..., dk} единиц. Выполните анализ времени исполнения дан- ного алгоритма. 7. [5] В Соединенных Штатах используются монеты достоинством в 1, 5, 10, 25 и 50 цен- тов. Теперь рассмотрим страну, в которой используются монеты достоинством в {<7,....dk} единиц. Нам нужно определить количество С(л) разных способов дать сдачу стоимостью п единиц. Например, для страны с монетами достоинством в {1,6, 10} еди- ниц имеем С(5) = 1, С(6) = 2,..., С(9) = 2, С( 10) = 3 и С(12) = 4. а) Сколько существует способов дать сдачу стоимостью в 20 единиц монетами достоин- ством в {1,6, 10} единиц? б) Предоставьте эффективный алгоритм для вычисления С(п) и выполните анализ его сложности. (Подсказка: подумайте о вычислении C(n, d), т. е. о вычислении количества способов дать сдачу стоимостью п единиц, используя монеты наивысшего достоинства d. Будьте внимательны, чтобы не дать лишнюю сдачу.) 8. [6] Рассмотрим задачу планирования загрузки однопроцессорной системы количеством п заданий Для каждого задания i установлено время обработки t, и крайнее время завер- шения d,. Допустимым расписанием исполнения заданий является такая перестановка за- даний, что при исполнении заданий в данном порядке каждое задание завершается до крайнего времени его завершения. "Жадный" алгоритм для решения задачи планировки загрузки однопроцессорной системы выбирает первой задание с самым ранним крайним временем завершения.
334 Часть I. Практическая разработка алгоритмов Докажите, что если допустимое расписание существует, то создаваемое таким "жадным" алгоритмом расписание является допустимым. Числовые задачи 9. [6] Задача о рюкзаке задается таким образом: имея множество целых чисел Х = = {5], 52, —- s„} и целевое число Т, найти такое подмножество множества X, сумма кото- рого в точности равна Т. Например, множество X = {1, 2, 5, 9, 10} содержит подмноже- ство, сумма членов которого составляет Т = 22, но не Т = 23. Предоставьте правильный алгоритм для решения задачи о рюкзаке с временем исполнения О(пТ) 10. [6] В задаче разделения множества целых чисел требуется выяснить, содержит ли мно- жество целых чисел X = {sb ..., .$„} такое подмножество /, для которого /е/ /г/ Пусть £ s, = М . Предоставьте алгоритм динамического программирования с време- нем исполнения О(«Л7) для решения задачи разделения множества целых чисел. 11. [5] Допустим, что имеется п чисел (некоторые из которых, возможно, отрицательные), расположенных по кругу. Разработайте эффективный алгоритм поиска наибольшей суммы смежных чисел в дуге. 12. [5] Некий язык для обработки строк позволяет разбивать строку на две части. Стои- мость этого разбиения равна п единицам времени, т. к. для этого требуется выполнить копирование старой строки. Программист хочет разбить строку на несколько частей, при этом общее время выполнения разбиения может зависеть от порядка, в котором осуществляется разбиение. Например, допустим, мы хотим разбить 20-символьную строку в позициях после символов 3, 8 и 10. Если разбиения осуществляются в порядке слева направо, то первое разбиение стоит 20 единиц времени, второе— 17, а третье— 12, что дает общую стоимость в 49 единиц времени. Если же разбиения осуществляются в порядке слева направо, то первое разбиение стоит 20 единиц времени, второе — 10, а третье— 8, что дает общую стоимость в 38 единиц времени. Предоставьте алгоритм динамического программирования, который берет в качестве входа список позиций символов и определяет самое дешевое разбиение за время 13. [5] Рассмотрим следующий способ сжатия данных. Имеется таблица, содержащая m текстовых строк, каждая длиной, самое большее, к символов. Требуется закодировать строку данных D длиной п символов, используя для этого наименьшее возможное коли- чество текстовых строк из таблицы. Например, если таблица содержит строки (a. ba, ahab, b), а строка данных имеет вид bababbaababa, то наилучшим способом ко- дирования будет (b, abab, ba, abab, а), в котором используются всего пять строк из таб- лицы. Предоставьте алгоритм с временем исполнения О(птк) для определения длины наилучшей кодировки. Можно полагать, что каждую кодируемую строку можно выра- зить, по крайней мере, одной комбинацией строк в таблице. 14. [5] Традиционный матч мирового чемпионата по шахматам состоит из 24 игр. Если матч заканчивается вничью, то текущий чемпион сохраняет свой титул. Каждая игра может быть выиграна, проиграна или сыграна вничью, где выигрыш равен I, проиг- рыш — 0, а ничья — 01/2. Цвет фигур меняется каждую игру. Белые имеют преимуще- ство, т. к. они ходят первыми. В первой игре чемпион играет белыми. Его шансы выиг-
Гпава 8. Динамическое программирование 335 рыта, ничьей и проигрыша равны vr„., wd и w/ при игре белыми и b„, bd и bt при игре черными соответственно. а) Напишите рекуррентное соотношение вероятности чемпиона удержания титула. По- лагается. что в матче осталось сыграть g игр и что чемпиону нужно выиграть i игр (ре- зультат которых может быть 1/2). б) На основе данного рекуррентного соотношения, разработайте алгоритм динамиче- ского программирования для вычисления вероятности текущего чемпиона сохранения своего титула. в) Выполните анализ временной сложности вашего алгоритма для матча из п игр. 15. [8] Если бросить яйцо с достаточной высоты, оно разобьется. В частности, в любом достаточно высоком здании должен быть f-i\ этаж, при падении с которого яйцо разо- бьется, но при падении с (/ - 1)-го этажа— нет. Если яйцо всегда разбивается, то тогда f= 1. Если яйцо никогда не разбивается, то тогда f= п + 1. Нужно найти критический этаж f в «-этажном здании. Для этого можно выполнять только одну операцию — бросить яйцо с определенного этажа и наблюдать за результа- тами. Нужно выполнить как можно меньше таких операций, но в любом случае требу- ется потратить не больше, чем k яиц. Разбитые яйца снова использовать нельзя Пусть Е(к, п) означает минимальное количество бросков, достаточное для получения решения. а) Докажите, что £(!,«) = п. б) Докажите, что Е(к. п) = ©(/7|/А). в) Выведите рекуррентное соотношение для Е(к, п). Каким будет время исполнения ди- намической программы для вычисления Е(к, л)? Задача на графах 16. [4] Рассмотрим город, улицы в котором задаются решеткой А" х у. Нам нужно пройти из верхнего левого угла решетки в нижний правый угол. К сожалению, в городе имеются районы с плохой репутацией, и мы не хотим проходить по улицам в этих районах. Име- ется матрица BAD размером X х К, в которой BAD[i, j] = "yes" тогда и только тогда, ко- гда перекресток улиц i и j находится в районе, через который мы не хотим проходить. а) Приведите пример такого содержимого матрицы BAD, для которого не существует пути между требуемыми точками без прохождения через плохие районы. б) Предоставьте алгоритм сложностью О(ХУ) для поиска пути, позволяющего избежать нежелательных районов. в) Предоставьте алгоритм сложностью О(ХУ) для поиска кратчайшего пути, позволяю- щего избежать нежелательных районов. Полагается, что все кварталы одинаковой дли- ны. Чтобы решение было зачтено частично, алгоритм может иметь время исполнения О(А2Г2). 17. [5] Даны такие же условия, как и в предыдущей задаче, т. е. город, улицы в котором оп- ределены решеткой X х У. Нам нужно пройти из верхнего левого угла города/решетки к нижнему правому углу. Плохие районы определены матрицей BAD размером X х у, в которой BAD[i,j] = "yes" тогда и только тогда, если перекресток улиц i и j находится в районе, через который мы не хотим проходить. Если бы в городе не было неблагополучных районов, которые мы вынуждены избегать, то длина кратчайшего пути между указанными точками была бы равной (X- 1) + (У - I)
336 Часть I. Практическая разработка алгоритмов кварталам. Более того, было бы много таких путей, каждый из которых состоял бы только из движений вправо и вниз. Разработайте алгоритм, который принимает в качестве входа массив BAD и возвращает количество безопасных путей длиной X + У-2. Чтобы решение было засчитано пол- ностью, время исполнения алгоритма должно быть равным OCYP)- Задачи по разработке 18. [4] В библиотеке нужно разместить на полках п книг в установленном в каталоге поряд- ке. Поэтому мы можем представить позицию конкретной книги как Ь,. ее толщину как а высоту как где 1 < i < и. Все полки в библиотеке имеют одинаковую длину L. Допустим, что все книги имеют одинаковую высоту h (т. е., h = h,= h, для всех z, /), а расстояние между полками больше чем /?, вследствие чего любую книгу можно поста- вить на любую полку. "Жадный" алгоритм поставит на первую полку наибольшее воз- можное количество книг, пока не будет достигнуто такое наименьшее значение /, для которого книга Ь, не будет вмещаться на данную полку, после чего повторит эту про- цедуру с последующими полками. Докажите, что "жадный" алгоритм всегда находит оптимальный порядок размещения полок и укажите его временную сложность. 19. [6] Данная задача является вариантом предыдущей. В данном случае книги разной вы- соты, но высоту каждой полки можно подогнать под высоту самой высокой книги на ней. Таким образом, стоимость определенного размещения является суммой высот наи- высших книг на каждой полке. • Предоставьте пример, показывающий, что "жадный" алгоритм, заполняющий каж- дую полку насколько возможно, не всегда дает минимальную общую высоту. • Разработайте алгоритм для решения этой задачи и выполните анализ его временной сложности. Подсказка-, используйте динамическое программирование. 20. [5] Требуется найти самый легкий способ набрать определенный номер из п цифр на стандартном телефонном кнопочном номеронабирателе, используя только два пальца. Начальными позициями этих двух пальцев являются кнопки звездочки (*) и решетки (#), а стоимость перемещения пальца от одной кнопки к другой пропорциональна эвк- лидову расстоянию между кнопками. Разработайте алгоритм для набора номера с наи- меньшим общим расстоянием, пройденным пальцами, с номеронабирателя с к разными кнопками (для стандартных телефонов к = 16). Попытайтесь получить время исполне- ния алгоритма О(пк2). 21. [6] Имеется вектор из и действительных чисел, для которого нужно найти подвектор с максимальной суммой. Например, в векторе {31,-41, 59, 26, -53, 58, 97, -93, -23, 84} подвектором с максимальной суммой будут третий по седьмой элементы, а именно: 59 + 26 + (-53) + 58 + 97 = 187. Если все элементы массива положительные, то решени- ем является весь массив, а когда все элементы отрицательные, то решением является пустой подвектор с суммой элементов, равной нулю. • Разработайте простой алгоритм с временем исполнения ®(zr) для поиска подвектора последовательных элементов с максимальной суммой. • Далее разработайте для решения этой же задачи алгоритм динамического програм- мирования с временем исполнения ®(«). Чтобы решение было зачтено частично, можно разработать алгоритм типа "разделяй и властвуй" с временем исполнения O(Hlog;j).
Гпава 8. Динамическое программирование 337 22. [7] Даны алфавит из к символов, строка х = из символов этого алфавита и таб- лица умножения символов алфавита. Нужно определить, возможно ли заключить части строки в скобки таким образом, чтобы в результате получить а, где а является символом алфавита. Таблица умножения не обладает ни перестановочным, ни ассоциативным свойствами, вследствие чего порядок выполнения операций умножения имеет значение. а b с а а а с Ь с а с с с b с Для примера, рассмотрим предшествующую таблицу умножения и строку bbbba. За- ключение частей строки в скобки в виде (b(bb))(ba) дает результат о, но в виде ({{{bb)b)b)a) дает результат с. Предоставьте алгоритм с полиномиальной временной сложностью по отношению кии А для определения, можно ли заключить в скобки части данной строки, чтобы получить целевой элемент согласно данной таблице умножения. 23. [6] Даны константы аир. Допустим, что стоимость движения влево в дереве равна а, а вправо— р. Разработайте алгоритм с оптимальной стоимостью создания дерева в наихудшем случае для ключей к\, ..., к„, вероятность просмотра каждого из которых со- ставляет рь ..., р„. Задачи, предлагаемые на собеседовании 24. [5] Для определенного набора достоинств монет найти наименьшее количество монет, которым можно дать сдачу определенного размера. 25. [5] Имеется массив из п чисел, каждое из которых может быть положительным, отрица- тельным или нулем. Предоставьте эффективный алгоритм для определения индексов элементов i и j максимальной суммы от /-го до /-го числа. 26. [7] При вырезании символа из страницы журнала также вырезается символ на обратной стороне страницы. Предоставьте алгоритм для определения, возможно ли создать опре- деленную строку из вырезанных из определенного журнала символов. Можно полагать, что имеется функция, которая позволяет определить символ на обратной стороне стра- ницы и его позицию для любого данного символа на лицевой стороне страницы. Задачи по программированию Эти задачи доступны на сайтах http://www.programming-challenges.com и http:// uva.onlinejudge.org. 1. Is Bigger Smarter? 111101/10131. 2. Weights and Measures. 111103/10154. 3. Unidirectional TSP. 111104/10116. 4. Cutting Sticks. 111105/10003. 5. Ferry Loading. 111106/10261.
ГЛАВА 9 Труднорешаемые задачи и аппроксимирующие алгоритмы В этой главе обсуждаются методы доказательства того, что для решения данной задачи не существует эффективного алгоритма. Читателям, не склонным к теоретизированию, необходимость доказывать что-либо, будет, скорее всего, неприятна, а сама мысль тра- тить время на доказательство чего-то несуществующего покажется абсолютно непри- емлемой. Однако, если мы не знаем, как решить какую-то задачу, бывает небесполезно выяснить, что она не имеет решения. Теория NP-полноты является чрезвычайно полезным инструментом разработчика ал- горитмов, даже когда дает отрицательные результаты. В частности, теория NP-полноты позволяет разработчику алгоритмов тратить усилия более эффективно, показав, что поиск эффективного алгоритма для данной задачи обречен на неудачу. Зато, если вы потерпели неудачу, доказывая нерешаемость задачи, вы можете ожидать, что сущест- вует эффективный алгоритм для ее решения. Две истории из жизни, приведенные в этой книге, описывают случаи, закончившиеся благополучно, несмотря на неоправдан- ные жалобы разработчиков на сложность задачи. Теория NP-полноты также позволяет распознать свойства, которые делают определен- ную задачу сложной. Это дает нам представление о том, как следует моделировать ее разными способами или использовать ее более благоприятные характеристики. Пони- мание того, какие задачи являются сложными, исключительно важно для разработчи- ков алгоритмов, и оно может быть приобретено только в попытках доказать сложность задач. Для доказательства сложности задач мы будем использовать такой фундаментальный принцип, как сведение между парой задач, показывая, что эти задачи в действительно- сти одинаковые. Для демонстрации этой идеи мы рассмотрим последовательность све- дений, каждое из которых дает эффективный алгоритм или доказательство, что такого алгоритма не существует. Кроме этого, предоставляется краткое введение в: ♦ теоретико-сложностные аспекты NP-полноты, одного из наиболее фундаменталь- ных понятий теории вычислительных систем; ♦ теорию аппроксимирующих алгоритмов, позволяющую получить эвристический алгоритм, который, возможно, выдаст решение, близкое к оптимальному. 9.1. Сведение задач В этой книге нам встретилось несколько задач, для решения которых мы не могли най- ти никаких эффективных алгоритмов. Теория NP-полноты предоставляет инструменты, с помощью которых можно показать, что на определенном уровне все эти задачи в действительности являются одной и той же задачей.
Гпава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 339 Ключевым принципом для демонстрации сложности задачи является сведение, или преобразование одной задачи в другую. С объяснением этой идеи может помочь такая аллегория NP-полноты. Несколько парней по очереди дерутся друг с другом, чтобы выяснить, кто из них "круче". Адам победил Билла, который затем одолел Чака. Кто же из этих троих "крутой", и есть ли смысл вообще говорить об этом? Без какого-либо внешнего стандарта ответить невозможно. Если бы Чак оказался Чаком Норрисом, в "крутости" которого никто не сомневается, то Адам и Билл были бы не менее "кру- тыми", чем он. С другой стороны, допустим, что отношения выясняют мальчишки из начальных классов. Хотя никому не придет в голову называть меня "крутым", даже если я мог бы справиться с Адамом. Отсюда следует, что ни одного из этих троих нельзя считать "крутым". В этой аллегории NP-полноты каждая драка представляет сведение, а Чак Норрис играет роль задачи выполнимости (satisfiability), т. е. достовер- но трудно решаемой задачи. Я же выступаю в роли неэффективного алгоритма, с воз- можностью исправления. Сведение— это операция преобразования одной задачи в другую. Чтобы описать све- дение, нам нужно использовать довольно строгие определения. Алгоритмическая зада- ча представляет собой вопрос общего характера, для которого даются входные пара- метры и условия, формулирующие, что можно считать удовлетворительным ответом или решением. Экзе.мпчяро.м (instance) называется задача, для которой указаны пара- метры входа. Разницу между задачей и ее экземпляром можно продемонстрировать на следующем примере: Задача. Задача коммивояжера. Вход. Взвешенный граф G. Выход. Маршрут, минимизирующий У''Д[г„ v,, ,] + vj. Любой взвешенный граф определяет экземпляр задачи коммивояжера, а каждая кон- кретная задача имеет, по крайней мере, один маршрут с минимальной стоимостью. Для общего экземпляра задачи коммивояжера требуется найти алгоритм для определения оптимального маршрута для всех возможных экземпляров задачи. 9.1.1. Ключевая идея Теперь рассмотрим две алгоритмические задачи, называющиеся Bandersnatch и Bo- billy. Допустим, что у нас имеется алгоритм решения задачи Bandersnatch (листинг 9.1). Листинг 9.1. Решение задачи Bandersnatch Bandersnatch (G) Преобразуем G в экземпляр У задачи Bo-billy Вызываем процедуру Во-billy для решения экземпляра У Возвращаем результат Во-ЬШу(У) в качестве ответа для Bandersnatch(G) Этот алгоритм выдаст правильное решение задачи Bandersnatch при условии, что при ее преобразовании в задачу Bo-billy всегда сохраняется правильность ответа. Иными сло- вами, для любого экземпляра G преобразование имеет свойство: Bandersnatch(G) = Bo-billy(f)
340 Часть I. Практическая разработка алгоритмов Преобразование экземпляров одного типа задачи в экземпляры задачи другого типа с сохранением правильных ответов и называется сведением (reduction). Теперь допустим, что при таком сведении экземпляр G преобразуется в У за время О(Р(п)). Тогда возможны два варианта: ♦ если процедура Bo-billy выполняется за время £>(/”(«)), то это означает, что время решения задачи Bandersnatch равно О(Р(п) + Р'(пУ), т. е. сумме времени преобразо- вания задачи Bandersnatch в задачу Bo-billy и времени решения последней; ♦ если известно, что выражение Q(/”(Z7)) является нижним пределом при вычислении Bandersnatch, т. е., что данную задачу нельзя решить быстрее, тогда выражение Q(/>'(«) - Р(п)) должно быть нижним пределом для вычисления Bo-billy. Почему это так? Если можно было бы решить Bo-billy быстрее, то это могло бы нарушить ниж- ний предел при решении Bandersnatch посредством ранее описанного приведения. Отсюда неявно следует, что не может существовать способа решения Bo-billy быст- рее, чем заявлено. В первом случае Стив (т. е. автор этих строк) демонстрирует ловким ударом в челюсть Адама, что все остальные являются слабаками. А во втором описывается подход с уча- стием Чака Норриса, используемый для доказательства сложности задачи. По сущест- ву, данное сведение показывает, что задача Bo-billy не легче, чем задача Bandersnatch. Поэтому если задача Bandersnatch сложная, то это означает, что задача Bo-billy также должна быть сложной. Для иллюстрации мы рассмотрим в этой главе несколько примеров сведения задач. Подведение итогов Посредством сведения можно показать, что две задачи являются, по сути, одинаковыми. Наличие эффективного алгоритма для решения одной задачи (или отсутствие такого) под- разумевает наличие (или отсутствие) эффективного алгоритма для решения другой. 9.1.2. Задачи разрешимости При сведении задача одного типа преобразуется в задачу другого типа таким образом, что ответы для всех экземпляров задачи являются идентичными. Задачи отличаются друг от друга диапазоном или типом возможных ответов. Решение задачи коммивоя- жера состоит из перестановки вершин, в то время как решения других задач возвра- щаются в виде чисел, диапазон которых может быть ограничен положительными или целыми числами. Диапазон решений наиболее интересного класса задач ограничен значениями ИСТИНА или ЛОЖЬ. Эти задачи называются задачами разрешимости (decision prob- lems). Удобно сводить одну задачу разрешимости к другой, т. к. допустимыми ответа- ми как для начальной, так и для конечной задачи являются только ИСТИНА или ЛОЖЬ. К счастью, большинство представляющих интерес задач оптимизации можно сформу- лировать в виде задачи разрешимости, которая отражает суть вычислений. Например, задачу разрешимости для задачи коммивояжера можно определить таким образом:
Гпава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 341 Задача. Разрешимость задачи коммивояжера. Вход. Взвешенный граф G и целое число к. Выход. Ответ, существует ли маршрут стоимостью < к. Формулировка в виде задачи разрешимости отражает саму суть задачи коммивояжера в том смысле, что если существует эффективный алгоритм для решения задачи разре- шимости, то с его помощью можно выполнить двоичный поиск для разных значений к и быстро оптимизировать решение А приложив некоторые усилия, можно по эффек- тивному решению задачи разрешимости восстановить и саму перестановку узлов мар- шрута. В дальнейшем мы будем, как правило, говорить о задачах разрешимости, потому что такой подход, во-первых, проще, а во-вторых, обладает достаточной теоретической мощью. 9.2. Сведение для создания новых алгоритмов На кухне сидят инженер и алгорист. Алгорист просит инженера вскипятить воды для чая. Инженер встает со своего стула, берет со стола чайник, наливает в него воды, ста- вит его на плиту, зажигает горелку, ждет, пока вода закипит, после чего выключает горелку. Некоторое время спустя инженер просит алгориста вскипятить еще воды. Тот поднимается со стула, берет чайник с плиты, ставит на стол и снова садится. — Сдела- но,— говорит он. — Я свеп задачу, требующую решения, к уже решенной задаче. Пример со сведением задачи кипячения воды иллюстрирует достойный способ созда- ния новых алгоритмов из старых. Если входные данные для задачи, которую мы хо- тим решить, можно преобразовать во входные данные для задачи, которую мы знаем, как решить, то процедура преобразования и известное решение образуют алгоритм для решения нашей задачи. В этом разделе мы рассмотрим несколько примеров сведения, которые позволяют вос- пользоваться эффективными алгоритмами. Чтобы решить задачу а, мы выполняем све- дение экземпляра задачи а к экземпляру задачи Ь, после чего решаем данный экземп- ляр, используя эффективный алгоритм для решения задачи Ь. Общее время исполнения в данном случае равно времени, необходимому для выполнения сведения, плюс время для решения экземпляра задачи Ь. 9.2.1. Поиск ближайшей пары В задаче поиска ближайшей пары требуется найти в множестве чисел пару чисел с наименьшей разницей между ними. Эту задачу можно преобразовать в задачу разре- шимости, задав вопрос, является ли данная разница меньшей, чем некое пороговое значение. А именно: Вход. Множество S из н чисел и пороговое значение /. Выход. Решение, существует ли такая пара чисел s„ s,eS, для которой |s, — .vj < /.
342 Часть I. Практическая разработка алгоритмов Задача поиска ближайшей пары решается применением простой сортировки, т. к. после сортировки члены ближайшей пары должны находиться рядом друг с другом. Отсюда следует этот алгоритм: CloseEnoughPair(S, t; Sort S Is mini l Si - s1+i | t? По поводу этого простого сведения можно сделать несколько замечаний. 1. Сведенная к задаче разрешимости версия задачи сохраняет суть первоначальной за- дачи, что означает, что решить ее не легче, чем первоначальную задачу поиска бли- жайшей пары. 2. Временная сложность алгоритма решения зависит от временной сложности сорти- ровки. Например, если применить для сортировки алгоритм с временной слож- ностью <J(«log»), то время поиска ближайшей пары будет равно 62(nlog« + и). 3. Это сведение и факт существования нижнего предела времени исполнения сорти- ровки. равного Q(«log«), не доказывают, что в наихудшем случае достаточно близ- кую пару можно найти за время Q(/?log«). Возможно, существует более быстрый алгоритм, который только нужно поискать? 4. С другой стороны, если бы мы знали, что в наихудшем случае достаточно близкую пару можно найти за время Q(/dogn). то это сведение было бы достаточным доказа- тельством того, что сортировку нельзя осуществить быстрее, чем за время Q(«logn), поскольку такая более быстрая сортировка подразумевала бы более быстрый алго- ритм поиска достаточно близкой пары. 9.2.2. Максимальная возрастающая подпоследовательность В г паве 8 было продемонстрировано использование динамического программирования для решения разных задач, включая вычисление расстояния редактирования строки (см. раздел 8.2) и максимальной возрастающей подпоследовательности (см. раздел 8.3). Приведу краткий обзор этих задач. ЗАДАЧА. Вычислить расстояние редактирования. Вход. Последовательности целых чисел или символов S и Т: стоимость каждой опера- ции вставки обозначена как с,„д, удаления — см и замены — Выход. Последовательность операций с минимальной стоимостью для преобразования последовательности S в последовательность Т. ЗАДАЧА. Найти максимальную возрастающую подпоследовательность. Вход. Последовательность S целых чисел или текстовых символов. Выход. Такая максимальная последовательность позиций целых чисел {/?!, для которой р, < р, । и SPi < SPi ,. В действительности, задачу поиска максимальной возрастающей подпоследовательно- сти можно решить в виде частного случая задачи вычисления расстояния редактирова- ния, как показано в листинге 9.2.
Гпава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 343 Листинг 9.2. Поиск максимальной возрастающей подпоследовательности »....................................;.жг,................. ..................; Longest Increasing Subsequence(<S1) T = Sort(S) Clr«fc ~ ~ 1 C.r Return (|S| - EditDistance(S, T, cins, cdear cdei)/2) Почему этот алгоритм дает правильный результат? Создавая вторую последователь- ность Т из элементов последовательности 5, отсортированных в возрастающем поряд- ке, мы обеспечиваем, что любая общая подпоследовательность должна быть возрас- тающей. Если же замены не разрешены ни при каких обстоятельствах (т. к. с,,,;, = и), то в результате оптимального выравнивания двух последовательностей мы находим их максимальную общую подпоследовательность и удаляем все прочее. Таким образом, для преобразования последовательности {3,I, 2} в последовательность {1, 2, 3} требу- ются две операции, а именно удаление и вставка цифры 3, не попавшей в максималь- ную общую подпоследовательность. Длиной максимальной возрастающей подпосле- довательности будет длина последовательности 5 за вычетом половины стоимости это- го преобразования. Каковы последствия данного сведения? Исполнение самого сведения занимает время Ofalogn). Поскольку вычисление расстояния редактирования занимает время O(|S|-|7]), то временная сложность алгоритма поиска максимальной возрастающей подпоследо- вательности в S будет квадратичной, т. е. такой же, как и сложность алгоритма, рас- смотренного в разделе 8.3. В действительности, для поиска максимальной возрастаю- щей подпоследовательности существует более быстрый алгоритм с временной слож- ностью O(n\ogn), основанный на удачно подобранных структурах данных, в то время как алгоритм вычисления расстояния редактирования в худшем случае имеет квадра- тичную временную сложность. В данном случае в результате сведения задач мы полу- чим простой, но не оптимальный алгоритм с полиномиальной временной сложностью. 9.2.3. Наименьшее общее кратное При работе с целыми числами часто требуется найти наименьшее общее кратное и наибольший общий делитель. Говорят, что а кратно Ь (т. е. />|а), если существует такое целое число d. для которого а = bd. Тогда постановка этих двух задач выглядит таким образом: ЗАДАЧА. Найти наименьшее общее кратное (НОК). Вход. Два целых числа х и у. Выход. Наименьшее целое число т, которое кратно и числу х, и числу г. ЗАДАЧА. Найти наибольший общий делитель (НОД). Вход. Два целых числа х и у. Выход. Наибольшее целое число d, на которое делится как число х, так и число у. Например, НОК(24,36)= 72. а НОД(24.36)= 12. Обе задачи можно с легкостью решить, разложив числа х и у на простые множители, но до сих пор не предложен эф- фективный алгоритм для разложения целых чисел на множители (см. раздел 13.8).
344 Часть I. Практическая разработка алгоритмов К счастью, задачу поиска наибольшего общего делителя можно эффективно решить с помощью алгоритма Евклида, не прибегая при этом к разложению на множители. Этот рекурсивный алгоритм основан на двух наблюдениях. Первое: если Л|о. тогда НОД(о, Ь) = Ь. Это должно быть довольно очевидным, т. к. если а кратно А, тогда a = bk для некоторо- го целого числа к, откуда следует, что НОД(6Д Ь) = Ь. Второе: если a = bt + г для целых чисел / и г, тогда НОД(о, Ь) = НОД(й, г). Поскольку х-у является кратным как х. так и у. то НОК(х, у) < ху. Существование мень- шего общего кратного возможно лишь в том случае, если существует какое-либо не- тривиальное общее кратное для х и у. Данное свойство в совокупности с алгоритмом Евклида обеспечивает эффективный способ для вычисления наименьшего общего кратного, как показано в следующем псевдокоде: LeastCommonMultiple(х, у) Return (ху/НОД(х,у)) . Это сведение позволяет нам применить алгоритм Евклида для решения другой задачи. 9.2.4. Выпуклая оболочка (*) В нашем последнем примере сведения "легкой" задачи (т. е. задачи, которая решается за полиномиальное время) мы используем сортировку для поиска выпуклой оболочки. Многоугольник называется выпуклым, если отрезок прямой линии, проведенной между любыми двумя точками внутри многоугольника Р, находится полностью внутри мно- гоугольника. Это возможно, когда многоугольник Р не содержит зазубрин или вогну- тостей, поэтому выпуклые многоугольники имеют аккуратную форму. Выпуклая обо- лочка предоставляет удобный способ структурирования множества точек. Соответст- вующие приложения рассматриваются в разделе 17.2. ЗАДАЧА. Найти выпуклую оболочку. Вход. Множество S. состоящее из п точек, лежащих в плоскости. Выход. Наименьший выпуклый многоугольник, содержащий все точки множества S. Данная задача решается с помощью сортировки (рис. 9.1). Это означает, что каждое число преобразуется в точку, для чего значение х отображается на точку (х, д ), т. е. каждое целое число отображается на точку параболы v = х*. А так как эта парабола является выпуклой, то каждая точка должна находиться на выпуклой оболочке. Кроме этого, так как соседние точки на выпуклой оболочке имеют соседние значения х, то выпуклая оболочка возвращает точки, отсортированные по х-координате, т. е. первона- чальные числа. Временная сложность алгоритма для создания и считывания точек, приведенного в листинге 9 3, равна О(п). Листинг 9.3. Алгоритм соьдани.1 и считывания точек выпуклой оболочки .....................;...................ii.s...-....................... Sort(S' Для каждого числа ieS создаем точку (i, i2) . Вызываем процедуру выпуклой оболочки для этого набора точек. Начиная с самой левой точки оболочки, считываем точки слева направо.
Глава 9 Труднорешаемые задачи и аппроксимирующие алгоритмы 345 Рис. 9.1. Сведение задачи поиска выпуклой оболочки к задаче сортировки посредством отображения точек на параболе Что все это означает? Вспомним, что нижний асимптотический предел сортировки ра- вен Q(»lg/z). Если бы временная сложность вычисления выпуклой оболочки могла быть лучшей, чем nig/?. то это сведение подразумевает временную сложность сортировки лучшую, чем Q(nlgn). что нарушает наш нижний предел. Соответственно, временная сложность вычисления выпуклой оболочки также должна быть равной Q(olgn)! Обра- тите внимание, что любой алгоритм для вычисления выпуклой оболочки, имеющий временную сложность O(nlgn). также предоставляет нам сложный, но правильный ал- горитм сортировки с временем исполнения O(n\gn). 9.3. Простые примеры сведения сложных задач Сведения в предыдущем разделе демонстрируют преобразования между задачами, для решения которых существуют эффективные алгоритмы. Но нас, в основном, интересу- ет использование механизма сведения для доказательства сложности задачи путем де- монстрации того, что существование быстрого алгоритма решения задачи Bandersnatch влечет за собой существование алгоритма, невозможного для задачи Bo-billy. На данном этапе вам нужно просто поверить мне на слово, что задачи поиска гамиль- тонова цикла и вершинного покрытия являются сложными. Вся картина сведения (рис. 9.2) станет ясной по завершении изучения этой главы. 9.3.1. Гамильтонов цикл Задача поиска гамильтонова цикла является одной из наиболее знаменитых задач в теории графов. В ней требуется найти маршрут, который проходит через каждую вер- шину данного графа ровно один раз. Данная задача имеет долгую историю и множест- во применений, как рассматривается в разделе 16.5. Формальное определение задачи следующее:
346 Часть I. Практическая разработка алгоритмов ЗАДАЧА. Найти гамильтонов цикл. Вход. Невзвешенный граф G. Выход. Существует ли простой маршрут, который проходит через каждую вершину графа G ровно один раз? Все задачи класса NP 4 Задача выполнимости Задача 3-SAT Задачи целочисленного программирования Задача об односвязном Задача Задача о независимом множестве подграфе о гамильтоновом цикле Задача о разделении множества целых чисел Задача коммивояжера Задача о клике Общая задача календарного планирования Рис. 9.2. Часть дерева сводимости для NP-полных задач. Сплошные линии обозначают сводимости, рассматриваемые в этой главе Задача поиска гамильтонова цикла имеет очевидное сходство с задачей коммивояжера. В каждой задаче требуется найти маршрут, который проходит через каждую вершину ровно один раз. Но между этими двумя задачами также имеются различия. Задача коммивояжера решается для взвешенных графов, а гамильтонова задача— для не- взвешенных. Из показанного в листинге 9.4 сведения задачи гамильтонова цикла к за- даче коммивояжера можно видеть, что сходств между этими задачами больше, чем различий. Листинг 9.4. Алгоритм сведения задачи поиска гамильтонова цикла к задаче коммивояжера HamiltonianCycle(G = (V,Е)) Q «даем полный взвешенный граф G' = (V, Е'), где V = V. n~|V| for . 1 to п do for j - . to n do if (i, j) e E then w(i, j) =1 else w(i, j) =2 Решаем задачу Traveling-Salesman-Decision-Problem(G', n) Само сведение не несет в себе ничего сложного, причем преобразование из невзвешен- ного во взвешенный граф осуществляется за время О(п~). Кроме этого, данное преобра-
Гпава 9 Труднорешаемые задачи и аппроксимирующие алгоритмы 347 зование подобрано таким образом, чтобы была обеспечена идентичность ответов для обеих задач. Если граф G содержит гамильтонов цикл {vb .. , v„}, тогда этот же маршрут будет соот- ветствовать и ребрам в наборе ребер £', вес каждого из которых равен единице. Это дает нам маршрут для задачи коммивояжера в графе G' весом ровно в и единиц. Если граф G не содержит гамильтонов цикл, тогда граф G' не может содержать такого мар- шрута коммивояжера, т. к, единственный способ получения в графе G маршрута стои- мостью в и единиц требовал бы использования ребер, каждое весом в единицу, что подразумевает наличие гамильтонова цикла в графе G. На рис. 9.3, а представлен при- мер графа, содержащего гамильтонов цикл, а на рис. 9.3, б — графа, не содержащего гамильтонов цикл. Рис. 9.3. Примеры графов Такое сведение является эффективным и сохраняет истинность. Наличие эффективного алгоритма для решения задачи коммивояжера подразумевало бы наличие эффективно- го алгоритма для решения задачи поиска гамильтонова цикла, в то время как доказа- тельство сложности задачи поиска гамильтонова цикла также подразумевало бы слож- ность задачи коммивояжера. Поскольку в данном случае имеет место второй вариант, это сведение показывает, что задача коммивояжера является сложной, по меньшей мере, в той же степени, что и задача поиска гамильтонова цикла. 9.3.2. Независимое множество и вершинное покрытие Задача о вершинном покрытии, которая рассматривается более подробно в panic- le 16.3, заключается в поиске минимального набора вершин, в который входит хотя бы один конец каждого ребра графа. Формальное определение задачи следующее: ЗАДАЧА. Найти вершинное покрытие. Вход. Граф G = (V, Е) и целое число к < |Г]. Выход. Существует ли такое подмножество S, содержащее, самое большее, к вершин, для которого каждое ребро ееЕ содержит, по крайней мере, одну вершину в мно- жестве S? Поиск "обычного" вершинного покрытия графа, т. е. покрытия, содержащего все вер- шины, представляет собой тривиальную задачу. Более сложной является задача охвата всех ребер с использованием минимально возможного набора вершин. Для графа на рис. 9.4 вершинное покрытие образовано четырьмя из восьми вершин графа.
348 Часть I. Практическая разработка алгоритмов Рис. 9.4. Обведенные кружками вершины составляют вершинное покрытие, а остальные вершины — независимое множество Множество вершин S графа G является независимым, если не существует ребер (х, у), для которых как вершина х, так и вершина у являются элементами множества вер- шин S. Это означает, что ни одна из любой пары вершин независимого множества не соединена ребром. Как будет рассказано в разбеле 16.2, независимое множество возни- кает при решении задач размещения точек обслуживания. Формальная постановка задачи выглядит таким образом: ЗАДАЧА. Найти независимое множество. Вход. Граф G и целое число к< |К]. Выход. Существует ли в графе G независимое множество из к вершин? Как в задаче о вершинном покрытии, так и в задаче о независимом множестве нужно найти особое подмножество вершин, содержащее хотя бы один конец каждого ребра в первом случае и не содержащее концы ребер во втором. Если множество S является вершинным покрытием графа G, то остальные вершины S-Vдолжны составлять неза- висимое множество, т. к. если бы обе вершины ребра находились в подмножестве S—V. то тогда множество S не могло бы быть вершинным покрытием. Это позволяет нам выполнить сведение между этими двумя задачами (листин! 9.5). Листинг 9.5. Алгоритм для сведения задачи о независимом множестве и задачи о вершинном покрытии •*-•••*а.*ъ.-..аа..,^*а.^....;^...ж ...... — .........а.ай'лЛ.ь:;-;'.и.:.'.\1.а.вь..йг..'а — .... Ve rtexCoveг(G, k) G' k'= |V| k Решаем задачу Independentset(G', k ') Опять же, простое сведение показывает, что обе эти задачи являются идентичными. Обратите внимание на то, что это преобразование осуществляется без какой бы то ни было информации об ответе. Преобразованию подвергается вход, а не решение. Это сведение демонстрирует, что из сложности задачи о вершинном покрытии вытекает сложность задачи о независимом множестве. В данном конкретном случае преобра- зуемые задачи можно с легкостью поменять местами, и это доказывает, что они одина- ково сложны. Остановка для размышлений. Сложность общей задачи календарного планирования Докажите, что общая задача календарного планирования является NP-полной, сведя к этой задаче задачу о независимом множестве.
Гпава 9 Труднорешаемые задачи и аппроксимирующие алгоритмы 349 ЗАДАЧА. Общая задача календарного планирования. Вход. Набор /. состоящий из п наборов линейных интервалов, целое число к. Выход. Можно ли из множества I выбрать подмножество из, по крайней мере, к непе- ресекающихся наборов линейных интервалов? Решение. Вспомните задачу календарного планирования съемок в фильмах (см. раз- дел 12). Там каждому фильму соответствовал один временной интервал, во время ко- торого осуществлялись съемки фильма. Требовалось найти наибольший возможный набор неконфликтующих фильмов, т. е. фильмов, периоды съемок которых не пересе- каются. Общий вариант календарного планирования позволяет разбить выполнение одного за- дания на несколько временных интервалов. Например, задание А, с периодами испол- нения январь-март и май-июнь, не конфликтует с заданием Z3, с периодами исполне- ния апрель-август, но конфликтует с заданием С, с периодом исполнения июнь-июль. Если мы хотим доказать сложность задачи календарного планирования на основе зада- чи о независимом множестве, то какая задача должна играть роль Bandersnatch, а какая роль Bo-billy? Нам необходимо показать, как преобразовать все задачи о независимом множестве в экземпляры задачи календарного планирования, т. е. наборов непересе- кающихся линейных интервалов. Какое соответствие имеется между этими двумя задачами? В обоих случаях требуется выбрать наибольшее возможное подмножество — вершин в задаче о независимом множестве и проектов в задаче календарного планирования. Это обстоятельство наво- дит на мысль, что нужно преобразовать вершины в проекты. Более того, в обеих зада- чах требуется. чтобы выбранные элементы не конфликтовали, т. е. вершины не имели общее ребро, а временные интервалы проектов не пересекались. Соответствующее сведение показано в листинге 9.6. Листинг 9.6. Алгоритм для сведения задачи о независимом множестве к задаче календарного планирования Independentset (G, к) I = О Для i-го ребра (х, у), 1 < i < m Добавляем интервал [i, i + 0,5] для проекта х до I Добавляем интервал [i, i + 0,5] для проекта у до I Решаем задачу GeneralProjectScheduling(I, k) Мое решение строится таким образом. Для каждого из т ребер графа на линии созда- ется интервал. Соответствующий каждой вершине проект будет содержать интервалы для смежных с ним ребер, как показано на рис. 9.5. Каждая пара вершин, имеющая общее ребро (что не допускается в независимом мно- жестве), определяет пару проектов, имеющих общий временной интервал (что не до- пускается в расписании съемок актера). Таким образом, наибольшие подмножества, удовлетворяющие условиям обеих задач, являются одинаковыми, а эффективный алго- ритм для решения общей задачи календарного планирования дает нам быстрый алго-
350 Часть I. Практическая разработка алгоритмов ритм для решения задачи о независимом множестве. Соответственно, общая задача календарного планирования должна быть такой же сложной, как и задача о независи- мом множестве. Рис. 9.5. Сведение задачи о независимом множестве к задаче календарного планирования (вершины обозначены цифрами, а ребра — буквами) 9.3.3. Задача о клике Социальной кликой (clique) является группа друзей, которые вместе проводят время. В теории графов кликой называется полный подграф, в котором каждая пара вершин соединена ребром. Клики являются наиболее плотными подграфами. Формальная по- становка задачи о клике имеет такой вид: ЗАДАЧА. Найти максимальную клику. Вход. Граф G = (К Е) и целое число к < |Г]. Выход. Содержит ли граф клику из к вершин, т. е., существует ли такое подмножество ScИ. где |S| < к, для которого каждая пара вершин в S определяет ребро в G? На рис. 9.6 показан граф, содержащий клику из пяти вершин. Можно ожидать, что граф дружеских отношений будет содержать большие клики, со- ответствующие отношениям на работе, между соседями, в церковных приходах, учеб- ных заведениях и т. п. Применение клик рассматривается более подробно в раз- деле 16.1. В случае с задачей о независимом множестве нам нужно было найти такое подмноже- ство вершин S. не являющихся концами общих ребер. Задача о клике отличается от этой задачи тем, что любая пара вершин должна быть соединена ребром. Для сведения
Гпава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 351 этих задач мы выполняем операцию дополнения графа, т. е. производим обмен ролями между фактом наличия ребра и фактом его отсутствия (листинг 9.7). Листинг 9.7 Сведение задачи о независимом множестве к задаче о клике Independent Set (G, k) Создаем граф G' - (V, Е'), где V = V, и For all (i, j) not in E, add (i, j) to E' Решаем задачу Clique(G', k) Последние два сведения предоставляют цепочку, связывающую три разные задачи. Сложность задачи о клике следует из сложности задачи о независимом множестве, ко- торая. в свою очередь, следует из сложности задачи о вершинном покрытии. Выстраи- вая сведения в цепочку, мы связываем вместе пары задач в соответствии с импликаци- ей сложности. Иашу работу можно считать выполненной, если все эти цепочки начи- наются с заведомо сложной задачи. В данной цепочке первым звеном является задача выполнимости. 9.4. Задача выполнимости булевых формул Для демонстрации сложности задач с помощью сведений нам нужно начать с одной заведомо сложной задачи. Самой сложной изо всех NP-полных задач является логиче- ская задача, называющаяся задачей выполнимости (satisfiability) булевых формул. Далее приводится формальная постановка этой задачи. ЗАДАЧА. Задача выполнимости булевых формул. Вход. Набор булевых переменных V и набор дизъюнкций (clause) С над И. Выход. Существует ли выполнимый набор значений истинности для дизъюнкций из С, т. е. способ присвоить набору переменных vh .... v„ значение ИСТИНА или ЛОЖЬ та- ким образом, чтобы каждая дизъюнкция содержала, по крайней мере, один истинный литерал? Для ясности приведем два примера. Допустим, что имеется набор С’ = {{vi,v2},{ vj. v2}} над булевыми переменными V= {vb v2}. Символом v, обозначается дополнение пере- менной V,. Это позволяет нам обеспечить выполнимость дизъюнкции, содержащей г„ если v, = ИСТИНА, или дизъюнкции, содержащей v„ если v, - ЛОЖЬ. Следовательно, выполнимость некоторого набора дизъюнкций включает в себя принятие последова- тельности из п решений ИСТИНА или ЛОЖЬ в попытке найти такие булевы значения, при которых выполняются все дизъюнкции. Дизъюнкции С = {{v|, V2},{vi, v2}} можно выполнить, присвоив значения vj = v2 = ИСТИНА или V| = v2 = ЛОЖЬ. Рассмотрим теперь набор дизъюнкций С = {{v|, v2}. {к,. v2}.{ V|}}. В данном случае не существует удовлетворительного набора значений, т. к. для выполнения третьей дизъюнкции значение переменной V| должно быть ЛОЖЬ, откуда следует, что для выполнения второй дизъюнкции значение переменной v2 должно быть ЛОЖЬ, но тогда не выполняется первое выражение. И сколько бы мы ни старались, получить требуемую выполнимость нам не удастся.
352 Часть I. Практическая разработка алгоритмов По ряду общественных и технических причин существует устоявшееся мнение, что задача выполнимости булевых формул является сложной, для решения которой не су- ществует алгоритмов с полиномиальным временем исполнения в худшем случае. Бук- вально все эксперты в области разработки алгоритмов (и бесчисленное множество про- сто хороших профессионалов) пытались непосредственно или косвенно разработать эффективный алгоритм для выяснения выполнимости данного набора дизъюнкций. Все они потерпели неудачу. Кроме этого, было доказано, что если бы существовал эф- фективный алгоритм для решения задачи выполнимости, то в области вычислительной сложности были бы возможными многие странные и невероятные вещи. Задача вы- полнимости является сложной, и принятие этого факта не должно доставлять нам ни- каких неудобств. Дополнительную информацию по задаче выполнимости и ее приме- нении см. в разделе 14 10. 9.4.1. Задача выполнимости в 3-конъюнктивной нормальной форме Занимаемое задачей выполнимости первое место среди NP-полных задач обусловило тот факт, что ее трудно решить для наихудшего случая. Но некоторые экземпляры ча- стных случаев задачи не обязательно такие сложные. Допустим, что каждая дизъюнк- ция содержит ровно один литерал. Чтобы выполнить данную дизъюнкцию, этому ли- тералу необходимо присвоить правильное значение. Эту процедуру можно повторить для каждой дизъюнкции данного экземпляра задачи. Таким образом, этот набор будет невыполнимым только в том случае, если имеется две дизъюнкции, которые прямо противоречат одна другой, например, такие как С = {{vi},{vi}}- Так как наборы дизъюнкций, каждая из которых содержит только один литерал, легко поддаются выполнению, нас интересуют чуть более длинные дизъюнкции. Сколько требуется литералов в каждой дизъюнкции, чтобы превратить задачу из решаемой за полиномиальное время в труднорешаемую? Это происходит, когда каждая дизъюнкция содержит три литерала. Формальное определение выглядит таким образом: ЗАДАЧА. Задача выполнимости в 3-конъюнктивной нормальной форме (3-SAT). Вход. Коллекция дизъюнкций С, каждая из которых содержит ровно 3 литерала, над набором булевых переменных V. Выход. Существует ли набор значений истинности для переменных V, при котором выполняется каждая дизъюнкция? Поскольку это частный случай задачи выполнимости, сложность задачи 3-SAT влечет за собой сложность общей задачи выполнимости. Обратное утверждение неверно, т. к. сложность общей задачи выполнимости может зависеть от длин дизъюнкций. Слож- ность задачи 3-SAT можно показать с помощью сведения, которое преобразует каждый экземпляр задачи выполнимости в экземпляр задачи 3-SAT, не нарушая при этом его выполнимость. Данное сведение преобразует каждую дизъюнкцию отдельно, в зависимости от ее дли- ны. путем добавления новых дизъюнкциий и булевых переменных. Допустим, что дизъюнкция С, содержит к литералов: ♦ к= 1 означает, что С, = {?\}. Создадим две новые переменные vb v2 и четыре новые дизъюнкции по 3 литерала каждая: {vb v2, zi), {vb v2, z,}. {vh v2, zt} и { v,. v2, Z|}.
Гпава 9 Труднорешаемые задачи и аппроксимирующие алгоритмы 353 Обратите внимание, что все эти четыре дизъюнкции могут быть одновременно вы- полнены только в том случае, если значение z, = ИСТИНА, что также означает вы- полнимость первоначальной дизъюнкции С',; ♦ к = 2 означает, что С, = {zi, z2}. Создадим одну новую переменную v, и две новые дизъюнкции по 3 литерала: {vb zh z2) и { zb z2}. Так же, как и в предыдущем слу- чае, эти две дизъюнкции могут быть одновременно выполнены только в том случае, если, по крайней мере, значение одной из переменных zx и z2 равно ИСТИНА, что также означает выполнимость С',; ♦ к = 3 означает, что С, = {zt. z2, z2}. Дизъюнкция С, копируется в экземпляр задачи 3-SAT без изменений: {z|, z2, z2}; ♦ к> 3 означает, что С, = {zh z2, ..., z„}. Создадим последовательно п-3 новых пере- менных и п — 2 новых дизъюнкций, таких, что С,, = {v,, i,z/+|, v(/}, C,j = {zb z2, vuj и C,„_2 = {v, „ 3. z„_|, z,,} для 2 <j < n - 2. Наиболее сложным является случай с длинными дизъюнкциями. Если ни один из пер- воначальных литералов не имеет значение ИСТИНА, то тогда не будет достаточного количества новых переменных, чтобы можно было выполнить все новые дизъюнкции. Дизъюнкцию C,j можно выполнить, присвоив литералу v,j значение ЛОЖЬ, но это заставит литерал v, 2 принять значение ЛОЖЬ, и т. д. пока не окажется, что дизъюнкция С,„.2 не может быть выполнена. Но если значение некоторого литерала z, равно ИСТИНА, то у нас имеется п-3 свободных переменных ил-3 оставшихся дизъюнк- ций, так что все они могут быть выполнены. Это преобразование выполняется за время О(т + п), если экземпляр задачи выполнимости имеет п дизъюнкций и т литералов. Так как любое решение задачи SAT является решением экземпляра задачи 3-SAT, а любое решение задачи 3-SAT описыва- ет, как присвоить значения переменных, чтобы получить решение задачи SAT, то пре- образованная задача эквивалентна первоначальной. Обратите внимание, что незначительная модификация этой конструкции может быть использована для доказательста того, что задачи 4-SAT, 5-SAT и, вообще. (к> 3)-SAT также являются NP-полными. Но эта конструкция неприменима для задачи 2-SAT. т. к. у нас нет способа заполнить цепочку дизъюнкций. Но для решения задачи 2-SAT за линейное время можно использовать алгоритм обхода в глубину на соответствующем графе. Подробности см. в разделе 14 10. 9.5. Нестандартные сведения Поскольку известно, что как общая задача выполнимости, так и задача выполнимости 3-SAT являются труднорешаемыми, в сведениях можно использовать любую из них. Обычно для этого лучше подходит задача 3-SAT, т. к. с ней проще работать. С целью предоставления дополнительных примеров и расширения нашего списка известных труднорешаемых задач мы рассмотрим еще два сложных сведения. Многие сведения довольно сложны, т. к. мы, по сути, программируем одну задачу на языке другой зада- чи, значительно отличающейся от первой. Больше всего путаницы возникает при выборе направления сведения. Запомните, что требуется преобразовать каждый экземпляр заведомо NP-полной задачи в экземпляр
354 Часть I. Практическая разработка алгоритмов задачи, которая нас интересует. Если выполнить сведение в обратном направлении, то мы получим лишь медленный способ решения интересующей нас задачи в виде про- цедуры с экспоненциальным временем исполнения. Это может сбить с толку начи- нающего алгориста, т. к. указанное направление сведения кажется обратным требуе- мому. Обязательно разберитесь с правильным направлением сведения сейчас, и обра- щайтесь к этому материалу в случае затруднений по данному предмету. 9.5.1. Целочисленное программирование Рассматриваемое в разделе 13.6 целочисленное программирование представляет собой фундаментальную задачу комбинаторной оптимизации. Ее лучше всего рассматривать, как задачу линейного программирования, в которой переменные могут принимать только целочисленные значения (а не вещественные). Формальная постановка задачи имеет следующую форму: ЗАДАЧА. Целочисленное программирование. Вход. Набор целочисленных переменных Ц набор неравенств над И, функция макси- мизации^ Р) и целое число В. Выход. Существует ли набор целочисленных значений переменных К для которого выполняются все неравенства иУ(Ц)>5? Рассмотрим два примера. Допустим, что V | > 1, V?2 > О V ] + V’2 < 3 /v):2v,, В = 3 Решением для данного входного экземпляра было бы vt = 1, v2 = 2. Но не все задачи обладают реализуемыми решениями. Рассмотрим следующий входной экземпляр для этой же задачи: V ] > 1, v2 > О V ! + v2 < 3 /v):2v2, В = 5 При данных ограничениях максимальное значение функции /(v) равно 2x2 = 4, поэто- му соответствующая задача разрешимости не имеет решения. Для доказательства груднорешаемости задачи целочисленного программирования мы выполним сведение от задачи 3-SAT. В данном конкретном случае можно было вос- пользоваться общей задачей выполнимости, но, как правило, использование задачи 3-SAT облегчает сведение. В каком направлении должно выполняться сведение? Мы хотим доказать трудноре- шаемость задачи целочисленного программирования, и мы знаем, что задача 3-SAT является труднорешаемой. Если задачу 3-SAT можно было бы решить, используя це- лочисленное программирование, и задача целочисленного программирования не была бы трудной, то и задача выполнимости не была бы трудной. Отсюда следует направле- ние сведения: задачу 3-SAT нужно преобразовать в задачу целочисленного программи- рования.
Гпава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 355 Каким должно быть это преобразование? Каждый экземпляр задачи выполнимости содержит булевы переменные и дизъюнкции. Каждый экземпляр задачи целочисленно- го программирования содержит целочисленные переменные (т. е. значения, ограни- ченные множеством 0, 1,2, ...) и ограничения. Имеет смысл сопоставить целочислен- ные переменные с булевыми и использовать ограничения в той же роли, какую играют дизъюнкции в исходной задаче. Преобразованная задача целочисленного программирования будет содержать вдвое больше переменных, чем соответствующий экземпляр задачи SAT — по одной пере- менной для каждой исходной переменной и для ее дополнения. Для каждой перемен- ной v, в поставленной задаче добавляются следующие ограничения: ♦ чтобы каждая переменная V, целочисленного программирования принимала только значения 0 или 1, добавляются ограничения 1 > V, > 0 и 1 > V t> 0. С учетом це- лочисленности переменных эти значения соответствуют значениям ИСТИНА и ЛОЖЬ; ♦ для гарантии того, что одна и только одна из двух переменных в задаче целочис- ленного программирования, связанных с данной переменной в задаче выполни- мости, имеет значение ИСТИНА, добавляются ограничения 1 > V, + V,> 1. Для каждой дизъюнкции 3-SAT С, = {z,, z2, z3} создадим ограничение V\ + V2 + И3> 1. Для удовлетворения данного ограничения, по крайней мере, одному из литералов в каждой дизъюнкции нужно присвоить значение 1, чтобы он соответствовал литералу ИСТИНА. Таким образом, выполнимость данного ограничения эквивалентна выпол- нимости дизъюнкции. Функция максимизации, а также граница теряют свою актуальность, т. к. мы уже зако- дировали весь экземпляр задачи 3-SAT. Используя/(v) = V\ nB = 0, мы гарантируем, что они не будут противоречить никаким значениям переменных, удовлетворяющим всем неравенствам. Очевидно, что это сведение можно выполнить за полиномиальное время. Чтобы установить, что при данном сведении сохраняется правильность ответа, нам нужно подтвердить следующие два обстоятельства: ♦ любое решение задачи выполнимости дает решение задачи целочисленного про- граммирования. В любом решении задачи выполнимости литерал ИСТИНА соот- ветствует значению 1 в решении задачи целочисленного программирования, т. к. дизъюнкция выполняется. Следовательно, сумма в каждом неравенстве больше или равна 1; ♦ любое решение задачи целочисленного программирования дает решение первона- чальной задачи выполнимости. В любом решении данного экземпляра задачи цело- численного программирования всем переменным должно быть присвоено значение О или 1. Если V, = 1, тогда литералу z, присваивается значение ИСТИНА. Если Е = 0, тогда литералу z, присваивается значение ЛОЖЬ. Это законное присваивание значений, которое должно удовлетворять все дизъюнкции. Так как сведение выполняется в обоих направлениях, то задача целочисленного про- граммирования должна быть труднорешаемой. Обратите внимание на следующие свойства, которые остаются в силе и при доказательстве NP-полноты: I. При данном сведении сохраняется структура задачи. Задача не решается, а просто преобразуется в другой формат.
356 Часть I. Практическая разработка алгоритмов 2. Возможные экземпляры задачи целочисленного программирования, которые могут получиться в результате данного преобразования, представляют только небольшое подмножество всех возможных экземпляров этой задачи. Но так как некоторые из этих экземпляров являются труднорешаемыми, то и общая задача также должна быть труднорешаемой. 3. Это преобразование отражает суть того, почему задачи целочисленного программи- рования трудны для решения. Оно не имеет никакого отношения к большим коэф- фициентам или большим диапазонам переменных, т. к. ограничение множества зна- чений до 0 и 1 является достаточным. Оно также не имеет ничего общего с большим количеством переменных в неравенствах. Задача целочисленного программирова- ния является труднорешаемой потому, что выполнимость набора ограничений явля- ется труднорешаемой задачей. Внимательное изучение свойств задачи, необходи- мых для ее сведения, может многое рассказать нам о самой задаче. 9.5.2. Вершинное покрытие Алгоритмическая теория графов изобилует сложными задачами. Прототипичной NP- полной задачей теории графов является задача поиска вершинного покрытия, которая была определена в разделе 9.3.2 таким образом: ЗАДАЧА. Найти вершинное покрытие. Вход. Граф G = (Г, £) и целое число к < |Р]. Выход. Существует ли такое подмножество S, содержащее, самое большее, к вершин, для которого, по крайней мере, одна вершина каждого ребра е е Е является элементом S2 Доказать сложность задачи о вершинном покрытии труднее, чем сложность задач в ранее рассмотренных сведениях, из-за различия структур соответствующих задач. Све- дение задачи 3-SAT к задаче о вершинном покрытии требует создания графа G и гра- ницы к из переменных и дизъюнкций экземпляра задачи выполнимости. Сначала мы преобразуем переменные задачи 3-SAT. Для каждой булевой переменной v, мы создадим две вершины v, и v„ которые соединяются ребром. Для покрытия всех ребер потребуется, по крайней мере, п вершин, т. к. ни одна пара этих ребер не будет иметь общей вершины. Далее мы преобразуем дизъюнкции задачи 3-SAT. Для каждой из с дизъюнкций мы создадим три новые вершины, по одной для каждого литерала в каждой дизъюнкции. Эти три вершины каждой дизъюнкции будут соединены таким образом, чтобы получи- лось с треугольников. В любое вершинное покрытие этих треугольников должны быть включены, по крайней мере, две вершины каждого треугольника, при этом общее чис- ло вершин покрытия будет равным 2с. Наконец, мы соединим вместе эти два набора компонентов. Каждый литерал в вер- шинных компонентах соединяется с вершинами в компонентах дизъюнкций (треуголь- никах), разделяющих один и тот же литерал. Таким образом, из экземпляра задачи
Глава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 357 3-SAT с п переменными и с дизъюнкциями создается граф, имеющий 2п + Зс вершин. Полное сведение для задачи 3-SAT {{г,, тз, v4}. { v,. v2, v4}} показано на рис. 9.7. Рис. 9.7. Сведение экземпляра {{vb v3, v4}. { vh v2, v4}} задачи выполнимости к задаче о вершинном покрытии Этот граф создан таким образом, чтобы вершинное покрытие размером п + 2с было возможным тогда и только тогда, когда первоначальное выражение является выпол- нимым. Согласно ранее приведенному анализу, каждое вершинное покрытие должно содержать, по крайней мере, п + 2с вершин, т. к. добавление соединительных ребер в граф G не может уменьшить размер вершинного покрытия до значения, меньшего, чем размер покрытия из разъединенных элементов. Для доказательства правильности на- шего сведения нам нужно продемонстрировать, что верны следующие утверждения. ♦ Каждый выполняющий набор значений истинности дает вершинное покрытие. Для данного выполняющего набора значений истинности для дизъюнкций выберите в качестве членов вершинного покрытия п вершин из вершинных компонентов, ко- торые соответствуют истинным литералам. Так как это определяет выполняющий набор значений истинности, то истинный литерал из каждой дизъюнкции должен покрывать, по крайней мере, одно из трех поперечных ребер, соединяющих каждую вершину треугольника с компонентом вершин. Соответственно, выбирая две другие вершины каждого треугольника дизъюнкции, мы также выбираем все оставшиеся поперечные ребра. ♦ Каждое вершинное покрытие дает выполняющий набор значений истинности. В любом вершинном покрытии С размером п + 2с ровно п вершин должны принад- лежать вершинным компонентам. Пусть эти вершины первого этапа определяют набор значений истинности, в то время как остальные 2с вершины покрытия нужно распределить по две на каждый компонент дизъюнкции. В противном случае ребро компонента дизъюнкции должно остаться непокрытым. Эти вершины компонентов дизъюнкции могут покрывать только два из трех соединяющих перекрестных ребер для каждой дизъюнкции. Поэтому если С обеспечивает вершинное покрытие, то, по крайней мере, одно поперечное ребро в каждой дизъюнкции должно быть покрыто, а это означает, что соответствующий набор значений истинности является выпол- няющим для всех дизъюнкций. Данное доказательство сложности задачи о вершинном покрытии, объединенное в це- почку со сведениями задачи о клике и задачи о независимом множестве из разде- ла 9.3.2. составляет библиотеку сложных задач теории графов, которые мы можем ис- пользовать для облегчения доказательства сложности других задач.
358 Часть I. Практическая разработка алгоритмов Подведение итогов Небольшой набор NP-полных задач (3-SAT, вершинное покрытие, разделение множества целых чисел и гамильтонов цикл) является достаточным для доказательства сложности большинства других сложных задач. 9.6. Искусство доказательства сложности Доказательство сложности задач требует определенного мастерства. Однако, приобре- тя навык, вы обнаружите, что выполнение сведений может быть на удивление простым процессом. Малоизвестной особенностью доказательств NP-полноты является тот факт, что их легче создать, чем объяснить, аналогично тому, как часто бывает легче заново написать код, чем разбираться в старом. Умение оценить вероятность того, что задача трудна для решения, требует опыта. Воз- можно. что самым быстрым способом набраться такого опыта будет внимательное изучение примеров в каталоге задач. Незначительное изменение формулировки задачи может сделать полиномиальную задачу NP-полной. Задача поиска кратчайшего пути в графе является легкой, в то время как задача поиска самого длинного пути в графе яв- ляется сложной. Задача построения маршрута, который проходит по всем ребрам графа ровно один раз (цикл Эйлера), является легкой, а вот задача построения маршрута, ко- торый проходит через каждую вершину графа (гамильтонов цикл) является сложной. Если вы подозреваете, что задача является NP-полной, сначала проверьте, не содер- жится ли она в книге [GJ79], в которой перечислены несколько сотен известных NP- полных задач. Велика вероятность, что вы найдете свою задачу в этой книге. Если же нет, то вы можете воспользоваться следующими советами по доказательству сложности задач: ♦ Примите меры к тому, чтобы виши исходная задача была как можно проще Ст. е. максимально ограничена). Во-первых, никогда не пытайтесь использовать в качестве исходной задачи общую задачу коммивояжера. Вместо этого используйте задачу поиска гамильтонова цик- ла, в которой все веса равны 1 или да. Еще лучше взять задачу поиска гамильтонова пути, чтобы не нужно было беспокоиться о замыкании цикла. Но самый лучший ва- риант— это задача поиска гамильтонова пути в ориентированном планарном графе, где степень каждой вершины равна 3. Все эти задачи имеют одинаковую сложность, но чем больше ограничений имеет сводимая задача, тем проще будет процесс све- дения. Во-вторых, никогда не пытайтесь использовать полную задачу выполнимости для доказательства сложности. Начните с задачи выполнимости 3-SAT. На самом деле, вам даже не нужно использовать задачу выполнимости полной 3-SAT. Вместо этого можно использовать задачу планарной 3-SAT, для которой существует способ пред- ставления дизъюнкции в виде плоского графа, такого, что все экземпляры одного и того же литерала можно соединить вместе, избегая пересечения ребер. Это свойство полезно при доказательстве сложности геометрических задач. Все эти задачи оди- наковой сложности, и поэтому сведения NP-полноты с использованием любой из них являются одинаково убедительными.
Гпава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 359 ♦ Сделайте целевую задачу максимально сложной. Не бойтесь добавлять дополнительные ограничения или разрешения, чтобы сделать целевую задачу более общей. Возможно, задачу на неориентированном графе мож- но обобщить до задачи на ориентированном графе и, таким образом, облегчить до- казательство ее сложности. Имея доказательство сложности для общей задачи, можно возвратиться назад и попытаться упростить целевую задачу. ♦ Выбирайте исходную задачу на основании веских аргументов. Выбор правильной исходной задачи играет важную роль в доказательстве сложно- сти задачи. Здесь очень легко ошибиться, хотя теоретически одна NP-полная задача подходит так же хорошо, как и любая другая. Пытаясь доказать сложность задачи, некоторые люди просматривают списки с десятками задач в поисках наиболее под- ходящей. Это непрофессиональный подход. Натолкнувшись на подходящую задачу, они, вероятнее всего, не догадаются, что именно она им и нужна. Я использую четыре (и только четыре!) задачи в качестве кандидатов для моих ис- ходных сложных задач. Ограничение количества исходных задач четырьмя означа- ет, что я могу знать многое о каждой из них, например, какие варианты задачи яв- ляются сложными, а какие нет. Перечислю свои любимые исходные задачи: • Задача 3-SAT. Многократно проверенный вариант. Если ни одна из перечислен- ных ниже задач не кажется подходящей, я возвращаюсь к этой первоначальной исходной задаче. • Задача о разделении множества целых чисел. Это единственно возможный вы- бор в тех случаях, когда кажется, что для доказательства сложности требуется использовать большие числа. • Задача о вершинном покрытии. Это подходящий вариант для любой задачи тео- рии графов, чья сложность зависит от выбора. Решение задач поиска хроматиче- ского числа, клики и независимого множества содержит элемент выбора пра- вильного подмножества вершин или ребер. • Задача о гамильтоновом цикле. Подходит для любой задачи на графах, чья сложность зависит от упорядочивания. Если вы пытаетесь разработать маршрут или календарный план, то задача о гамильтоновом цикле, вероятнее всего, явля- ется вашим средством для решения этой задачи. ♦ Повышайте стоимость нежелательного выбора. Многие люди испытывают робость при доказательстве сложности задач. Мы пыта- емся преобразовать одну задачу в другую, как можно меньше отступив от ориги- нальной задачи. Это легче всего делать, применяя строгие штрафные санкции за любое отклонение от целевого решения. Вы должны мыслить примерно следующим образом: "Если выбирается элемент а, то придется выбрать огромное множество S, которое не позволит найти оптимальное решение". Чем дороже последствия неже- лательных действий, тем легче будет доказать эквивалентность задач. ♦ Разработайте стратегический план, а затем создавайте компоненты для такти- ческих действий.
360 Часть I. Практическая разработка алгоритмов Спрашивайте себя: • Как мне добиться того, чтобы был выбран вариант А или В, но не оба одновре- менно? • Что сделать, чтобы вариант А был выбран раньше варианта В? • Как поступить с невыбранными вариантами? После того, как вы получите общее представление о действиях, выполняемых ком- понентами, вы можете задуматься о том, как создать эти компоненты. ♦ Если вы испытываете трудности на каком-то этапе, переключайтесь с поиска алгоритма на поиск сведения и обратно. Иногда сложность задачи нельзя доказать по той причине, что для ее решения су- ществует эффективный алгоритм! Применение таких методов, как динамическое программирование или сведение задач к сложным, но полиномиальным задачам на графах, например, к задаче о сопоставлении пар или задаче потоков в сети, может привести к неожиданным результатам. Каждый раз, когда вы не можете доказать сложность задачи, имеет смысл попытаться пересмотреть свое мнение. 9.7. История из жизни. Наперегонки со временем Аудитория стремительно теряла внимание к лекции. У некоторых студентов слипались глаза, а другие уже клевали носом. До конца моей лекции по NP-полноте оставалось двадцать минут, и студентов нельзя было винить за их реакцию. Мы уже рассмотрели несколько сведений, аналогичных представленным в предыдущих разделах. Но сведения NP-полных задач легче создать, чем объяснить или понять. Было необходимо продемонстрировать студентам сведение в процессе его создания, чтобы они могли понять, как оно работает. Я потянулся за книгой "Computers and Intractability" [GJ79], выручавшей меня в труд- ные моменты. Приложение к ней содержит список из более чем четырех сотен разных известных NP-полных задач. — Итак! — сказал я достаточно громко, чтобы заставить вздрогнуть дремавших в зад- них рядах. — Доказательства NP-полноты настолько рутинны, что их можно создавать по требованию. Мне нужен помощник. Есть добровольцы? В передних рядах поднялось несколько рук. Я вызвал к доске одного из студентов. — Выбери произвольную задачу из списка в приложении в этой книге. Я могу доказать сложность любой из этих задач в течение оставшихся семнадцати минут лекции. Теперь я определенно овладел их вниманием. Но это было все равно, что жонглировать работающими бензопилами. Я должен был выдать результат, не изрезав себя в клочья. Студент выбрал задачу. — Хорошо, докажите сложность задачи о неэквивалентности программ с присваиваниями, — сказал он. — Никогда раньше не слышал об этой задаче. Прочитай мне условие, чтобы я мог на- писать его на доске.
Гпава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 361 Задача определялась таким образом: ЗАДАЧА. Неэквивалентность программ с присваиваниями (Inequivalence of Programs with Assignments). Вход. Конечное множество переменных X, множество их значений V и две программы Р\ и Ру каждая из которых состоит из последовательности присваиваний вида: хо if (xj — Xi) then хз else xj, где.г, является элементом X. Выход. Возможно ли первоначально присвоить каждой переменной из множества А' значение из множества V таким образом, чтобы две данные программы выдавали раз- ные конечные значения для некой переменной из множества А? Я посмотрел на часы. Пятнадцать минут до конца лекции. Мне предстояло решить языковую задачу. Входом задачи были две программы с переменными, и нужно было проверить, всегда ли они выдают одинаковый результат. — В первую очередь, самое важное. Нам нужно выбрать исходную задачу для нашего сведения. Какая задача подойдет для этого? Задача разделения множества целых чи- сел? Задача 3-SAT? Задача о вершинном покрытии или о гамильтоновом пути? Овладев вниманием аудитории, я рассуждал вслух. — Так как наша целевая задача не является задачей на графах или численной задачей, давайте рассмотрим старую доб- рую задачу 3-SAT. По-видимому, эти две задачи похожи в некоторых аспектах. Задача 3-SAT имеет переменные и данная задача тоже. Чтобы сделать нашу задачу еще более похожей на задачу 3-SAT, можно попробовать ограничить ее переменные двумя значе- ния, т.е. V= {ИСТИНА. ЛОЖЬ}. Да, это удобно. До конца лекции оставалось 14 минут. — В каком направлении должно идти наше све- дение? От задачи 3-SAT к языковой задаче или от языковой задачи к задаче 3-SAT? Кто-то из первого ряда пробормотал, что от задачи 3-SAT к языковой задаче. — Совершенно верно. Поэтому нам нужно преобразовать наш набор дизъюнкций в две программы. Как мы можем это сделать? Можно попробовать разделить дизъюнкции на два набора и написать отдельные программы для каждого из них. Но как их разделить? Я не вижу, как это можно сделать каким-либо естественным способом, т. к. удаление любой дизъюнкции из программы может неожиданно сделать невыполнимую формулу выполнимой, таким образом полностью меняя ответ. Давайте попробуем что-нибудь другое. Мы можем преобразовать все дизъюнкции в одну программу, а потом сделать вторую программу тривиальной Например, вторая программа может игнорировать вход и всегда выводить или только значение ИСТИНА, или только значение ЛОЖЬ. Это выглядит намного лучше. Я продолжал рассуждать вслух. В этом не было ничего особенного. Но я заставил сту- дентов слушать меня! — Как нам превратить набор дизъюнкций в программу? Мы хотим знать, может ли этот набор дизъюнкций быть выполнен, т. е. существует ли набор значений перемен- ных, делающий его значение истинным. Допустим, что мы создали программу для Проверки ВЫПОЛНИМОСТИ С\ = (Х|, Х2, Хз).
362 Часть I. Практическая разработка алгоритмов Несколько минут я водил мелом по доске, пока у меня не получилась правильная про- грамма для эмулирования дизъюнкции. Я предположил, что у нас есть доступ к кон- стантам для значений ИСТИНА и ЛОЖЬ: с-| = if О, = ИСТИНА) then ИСТИНА else ЛОЖЬ ci = if (х2 = ЛОЖЬ) then ИСТИНА else с, с, = if(x2 = ИСТИНА) then ИСТИНА else с, — Теперь у меня есть метод для оценки истинности каждой дизъюнкции. Я могу сде- лать то же самое для оценки выполнимости всех дизъюнкций. sat = if (с, = ИСТИНА) then ИСТИНА else ЛОЖЬ sat = if (c2 = ИСТИНА) then sat else ЛОЖЬ sat = if (c„ = ИСТИНА) then sat else ЛОЖЬ В задних рядах возникло оживление. Они увидели луч надежды покинуть аудиторию вовремя. До звонка оставалось две минуты. — Итак, у нас имеется программа, которая возвращает истинное значение тогда и только тогда, когда переменным можно присвоить значения так, чтобы выполнялись все дизъюнкции. Нам нужна вторая программа, чтобы закончить доказательство. А если мы напишем sat = ЛОЖЬ. Да, это все что нам нужно. В нашей языковой задаче спрашивается, выводят ли две программы всегда одинаковый результат, независимо от значений, присвоенных переменным. Если дизъюнкции являются выполнимыми, то это означает, что можно присвоить переменным значения таким образом, что длинная программа будет выводить истинное значение. Проверка эквивалентности программ это то же самое, что и проверка дизъюнкций на выполнимость. Я победно вскинул руки. — Итак, задача является NP-полной. — Как только я сказал последнее слово, прозвучал звонок. 9.8. История из жизни. Полный провал Этот педагогический прием с выбором произвольной NP-полной задачи из 400 с лиш- ним задач в книге [GJ79] и доказательством ее сложности на лету мне так понравился, что я начал пользоваться им постоянно. Прием удавался мне восемь раз подряд. Одна- ко наступил день, когда я потерпел фиаско. На этот раз группа проголосовала за сведение из раздела по теории графов, а студент- доброволец выбрал задачу № 30. Постановка задачи GT30 выглядит таким образом- ЗАДАЧА. Найти односвязный подграф. Вход. Ориентированный граф G = (И, А), положительное целое число к < |Л|. Выход. Существует ли такое подмножество дуг Я' е А, где > к, для которого в гра- фе G'- (Г, А') существует, самое большее, один ориентированный путь между любой парой вершин? — Это задача о выборе, — определил я, как только задача была озвучена. Ведь нам нужно было выбрать такое максимально возможное подмножество дуг, которое не со-
Гпава 9 Труднорешаемые задачи и аппроксимирующие алгоритмы 363 держало бы пары вершин, соединенных множественными путями. Л это означало, что предпочтительной задачей для сведения является задача вершинного покрытия. Я немного поразмыслил о том, каким образом эти две задачи были похожими друг на друга. В обеих задачах нужно было найти некоторые подмножества, хотя в задаче о вершинном покрытии требовалось найти подмножества вершин, а в задаче об одно- связном подграфе нужно было найти подмножества ребер. Кроме этого, в задаче о вершинном покрытии требовалось найти минимально возможное подмножество, в то время как в задаче об односвязном подграфе нужно было определить максимально возможное подмножество. В исходной задаче были неориентированные ребра, а в це- левой— ориентированные дуги, так что мне нужно было добавить в сведение ориен- тацию ребер. Я должен был каким-то образом придать ориентацию ребрам графа вершинного по- крытия. Можно было попробовать заменить каждое неориентированное ребро (х, у) дугой, идущей, скажем, от у к х. Но в зависимости от выбранного направления дуги получились бы совсем разные ориентированные графы. Поиск "правильной" ориента- ции ребер мог оказаться трудной задачей, слишком трудной для использования на эта- пе преобразования. Я понимал, что ребра можно ориентировать так, чтобы получившийся граф стал бес- контурным орграфом. Но что это дает? В бесконтурных орграфах пары вершин могут быть связаны большим количеством ориентированных путей. В качестве варианта можно было попробовать заменить каждое неориентированное ребро (х, у) двумя дугами — от у к х и от х к у. Теперь не нужно было выбирать пра- вильные дуги для моего сведения, но граф стал очень сложным. Я никак не мог найти способ предотвратить множественные нежелательные пути между парами вершин. Время лекции подходило к концу. В последние десять минут меня охватила паника, т к. я понял, что на этот раз я не смогу предоставить доказательство. Нет худшего чувства, чем чувство профессора, завалившего лекцию. Ты стоишь у дос- ки, что-то бормоча, и ясно видишь, что студенты не понимают, о чем идет речь, но до- гадываются, что ты сам не понимаешь, о чем говоришь. Прозвучал звонок и студенты стали покидать аудиторию, одни с сочувствующим выражением на лице, другие с ехидными ухмылками. Я пообещал им предоставить решение на следующем занятии, но каждый раз, когда я думал об этом, я застревал в одном и том же месте. Я даже попробовал поступить не- честно и найти доказательство в научном журнале, указанном в ссылке в книге [GJ79]. Но ссылка была на неопубликованный доклад 30-летней давности, который нельзя бы- ло найти ни в Интернете, ни в библиотеке. Мысль о следующем занятии, последней лекции семестра, приводила меня в ужас. Но в ночь перед лекцией решение явилось ко мне во сне. — Раздели каждое ребро попо- лам,— сказал мне голос. Я вздрогнул, проснулся и посмотрел на часы. Было три часа ночи. Вскочил с кровати и набросал доказательство. Допустим, я заменю каждое неориенти- рованное ребро (х, у) конструкцией, состоящей из новой центральной вершины vri. с исходящими из нее дугами к вершинам х и у соответственно. Неплохо. Между каки-
364 Часть I. Практическая разработка алгоритмов ми вершинами могут существовать множественные пути? Новые вершины имели толь- ко исходящие ребра, так что только они могли служить источником множественных путей. Старые вершины имели только входящие ребра. Существует максимум один способ попасть из новой вершины-истока в любую из оригинальных вершин графа вершинного покрытия, так что старые вершины не могли дать множественных путей. Теперь добавим узел стока 5 и направим в него ребра из всех оригинальных вершин. От каждой новой вершины к этому стоку будет идти ровно два пути — по одному че- рез каждую из первоначальных вершин, смежных с ней. Один из этих путей нужно ра- зорвать, чтобы создать односвязный подграф. Как это сделать? Для разъединения можно выбрать одну из двух вершин, удалив дугу (х, л) или (у, л) новой вершины vT1, Чтобы получить подграф максимального размера, мы ищем наименьшее количество дуг, подлежащих удалению. Удалим исходящие дуги хотя бы у одной из двух вершин, определяющих первоначальное ребро. Но ведь это же самое, что и поиск вершинного покрытия в этом графе! Полученное сведение показано на рис. 9.8. Рис. 9.8. Сведение задачи о вершинном покрытии к задаче односвязного подграфа посредством разделения ребер и добавления узла стока Представление этого доказательства на следующем занятии несколько подняло мою самооценку, но гораздо важнее тот факт, что оно подтвердило правильность провоз- глашенных мною принципов доказательства сложности задач. Обратите внимание, что в конечном счете сведение не было таким уж сложным: достаточно разорвать ребра и добавить узел стока. Выполнение сведения NP-полных задач нередко удивляет своей простотой, нужно лишь подойти к нему с правильной стороны. 9.9. Сравнение классов сложности Р и NP Теория NP-полноты основана на строгих определениях из теории автоматов и фор- мальных языков. Новички, не обладающие базовыми знаниями, обычно находят при- меняемую терминологию трудной для понимания или употребляют ее неправильно. Знание терминологии не является чем-то действительно необходимым при решении практических вопросов. Но, несмотря на все сказанное, вопрос "Верно ли, что Р - NP?" представляет собой важнейшую задачу в теории вычислительных систем, и каждый образованный алгорист должен понимать, о чем идет речь.
Гпава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 365 9.9.1. Верификация решения и поиск решения В сравнении классов сложности Р и NP главным вопросом является, действительно ли верификация решения представляет более легкую задачу, чем первоначальный поиск решения. Допустим, что при сдаче экзамена вы "случайно" заметили ответ у студента, сидящего рядом с вами. Принесет ли это вам какую-либо пользу? Вы бы не рискнули сдать этот ответ, не проверив его, т. к. вы способный студент и могли бы решить дан- ною задачу самостоятельно, если бы уделили ей достаточно времени. Но при этом важно, можете ли вы проверить правильность подсмотренного ответа на задачу быст- рее, чем решить саму задачу самостоятельно. В случае NP-полных задач разрешимости, которые обсуждались в этой главе, ситуация кажется очевидной Рассмотрим примеры. ♦ Можно ли проверить, что граф содержит маршрут коммивояжера с наибольшим весом к, если дан порядок вершин маршрута? Да. Просто сложим вместе веса ребер маршрута и покажем, что общий вес равен, самое большее, к. Это ведь легче, чем найти сам путь, не так ли? ♦ Можно ли проверить, что данный набор значений истинности представляет реше- ние для данной задачи выполнимости? Да. Просто проверим каждую дизъюнкцию и убедимся, что она содержит, по крайней мере, один истинный литерал из данного набора значений истинности. Это ведь легче, чем найти выполняющий набор значе- ний истинности, не так ли? ♦ Можно ли проверить, что граф G содержит вершинное покрытие, состоящее из, са- мое большее, к вершин, если известно подмножество S, состоящее из, самое боль- шее, к вершин, составляющих данное покрытие? Да. Просто обходим каждое ребро (и, v) графа G и проверяем, что или и или v является элементом S. Это ведь легче, чем найти само вершинное покрытие, не так ли? На первый взгляд, все просто. Данные решения можно проверить за линейное время для всех трех задач, однако для решения любой из них неизвестно никакого алгоритма, кроме полного перебора. Но проблема в том, что у нас нет строгого доказательства нижней границы, препятствующей существованию эффективного алгоритма для реше- ния этих задач. Возможно, в действительности существуют полиномиальные алгорит- мы (скажем, с временем исполнения О(п)), но мы просто недостаточно хорошо их ис- кали. 9.9.2. Классы сложности Р и NP Каждая четко определенная алгоритмическая задача должна иметь максимально быст- рый алгоритм решения, точнее говоря, максимально быстрый в терминах "О-большое" для наихудшего случая. Класс Р можно рассматривать как закрытый клуб алгоритмических задач, членом ко- торого задача может стать, только продемонстрировав, что для ее решения существует алгоритм с полиномиальным временем исполнения. Полноправными членами этого клуба Р являются, например, задача поиска кратчайшего пути, задача поиска мини- мального остовного дерева и задача календарного планирования. Сокращение Р озна- чает полиномиальное (polynomial) время исполнения.
366 Часть I Практическая разработка алгоритмов Менее престижный клуб открывает свои двери любой алгоритмической задаче, реше- ние которой можно проверить за полиномиальное время. Как было показано ранее, членами этого клуба являются задача коммивояжера, задача выполнимости и задача о вершинном покрытии, ни одна из которых в настоящее время не квалифицирована должным образом, чтобы стать членом клуба Р. Но все члены закрытого клуба Р авто- матически являются членами этого второго, не столь престижного, клуба. Если задачу можно решить с самого начала за полиномиальное время, то ее решение определенно можно проверить столь же быстро: просто решить ее сначала и посмотреть, совпадает ли полученное вами решение с тем, которое было заявлено. Этот менее престижный клуб называется NP. Это сокращение можно расшифровать как "not-necessarily polynomial-time" (не обязательно полиномиальное время исполне- ния)1. Важнейший вопрос состоит в том, действительно ли класс NP содержит задачи, кото- рые не могут быть членами класса Р. Если такой задачи не существует, то классы должны быть одинаковыми и Р = NP. Но если существует хотя бы одна такая задача, то классы разные и Р* NP. Большинство алгористов и теоретиков сложности вычислений придерживаются мнения, что классы разные, т. е. что Р * NP, но для него требуется гораздо более строгое доказательство, чем заявление: "Я не могу найти достаточно бы- стрый алгоритм". 9.9.3. Почему задача выполнимости является самой сложной из всех сложных задач? Существует громадное дерево сведений задач NP-полноты, которое всецело основано на сложности задачи выполнимости. Часть задач из этого дерева, рассмотренных в этой главе (и доказанных в других источниках), показана на рис. 9.2. Но здесь возникает одна тонкость. Каковы были бы последствия, если бы кто-то дейст- вительно нашел алгоритм с полиномиальным временем исполнения для решения зада- чи выполнимости? Существование эффективного алгоритма для любой NP-полной за- дачи (скажем, задачи коммивояжера) подразумевает существование эффективных ал- горитмов для решения всех задач на отрезке пути в дереве сведений между задачей коммивояжера и задачей выполнимости (задача о гамильтоновом цикле, задача о вер- шинном покрытии и задача 3-SAT). Но существование эффективного алгоритма для решения задачи выполнимости не дает нам ничего сейчас же, т. к. путь сведений от задачи SAT к задаче SAT не содержит никаких других задач. Не будем впадать в панику. Существует замечательное сведение, называемое теоремой Кука, которое сводит все задачи класса NP к задаче выполнимости. Таким образом, если доказать, что задача выполнимости, или любая NP-полная задача, является чле- ном класса Р, то за ней последуют все другие задачи в классе NP. из чего будет следо- вать, что Р = NP. Так как, в сущности, каждая упомянутая в данной книге задача явля- 1 В действительности это сокращение означает "nondeterministic polynomial-time" (недетерминированное полиномиальное время исполнения) и является термином из области теории недетерминированных авто- матов.
Глава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 367 ется членом класса NP, то результаты такого развития событий были бы весьма значи- тельными и удивительными. Теорема Кука доказывает, что задача выполнимости является такой же сложной, как и любая другая задача из класса NP. Кроме этого, она также доказывает, что каждая NP-полная задача такая же сложная, как и любая другая. Здесь уместно вспомнить про эффект домино. Но то обстоятельство, что мы не можем найти эффективного алгорит- ма ни для одной из этих задач, дает веское основание полагать, что все они действи- тельно сложные и что, вероятно, Р * NP. 9.9.4. NP-сложность по сравнению с NP-полнотой Сейчас мы обсудим различие между NP-сложностью задачи и ее NP-полнотой. Я имею склонность к несколько вольному употреблению терминологии, а между этими двумя понятиями существует тонкая (обычно несущественная) разница. Говорят, что задача является NP-сложной, если, подобно задаче выполнимости, она, по крайней мере, такая же сложная, как любая другая задача класса NP. Говорят, что зада- ча является NP-полной, если она NP-сложная и также является членом класса NP. Так как класс NP является таким большим классом задач, большинство NP-сложных задач, с которыми вам придется столкнуться, в действительности будут NP-полными, и вы всегда можете предоставить стратегию проверки решения задачи (обычно достаточно простую). Все рассмотренные в этой книге NP-сложные задачи также являются NP- полными. Тем не менее, существуют задачи, которые кажутся NP-сложными, но при этом не яв- ляются членами класса NP. Такие задачи могут быть даже более сложными,' чем NP- полные задачи! В качестве примера сложной задачи, не являющейся членом класса NP, можно привести игру для двух участников, такую, как шахматы. Представьте себе, что вы сели играть в шахматы с самоуверенным игроком, который играет белыми. Он на- чинает игру ходом центральной пешки на два поля и объявляет вам мат. Единственным способом доказать его правоту будет создание полного дерева всех ваших возможных ходов и его лучших ответных ходов и демонстрация невозможности вашего выигрыша в текущей позиции. Количество узлов этого полного дерева будет экспоненциально зависеть от его высоты, равной количеству ходов, которые вы сможете сделать, перед тем как проиграть, применяя максимально эффективную защиту. Ясно, что это дерево нельзя создать и проанализировать за полиномиальное время, по- этому задача не является членом класса NP. 9.10. Решение NP-полных задач Человек практичный никогда не останавливается на доказательстве того, что задача является NP-полной. Очевидно, у него были какие-то причины решить ее. Эти причи- ны не исчезли, когда он узнал, что для решения задачи не существует алгоритма с по- линомиальным временем исполнения. Ему все равно нужна программа для решения этой задачи. Все, что известно, так это лишь невозможность создания программы для быстрого оптимального решения задачи в наихудшем случае.
368 Часть I. Практическая разработка алгоритмов Однако для достижения цели остаются три варианта: ♦ алгоритмы, эффективные для средних случаев задачи. В качестве примеров таких алгоритмов можно назвать алгоритмы поиска с возвратом, в которых выполняются значительные отсечения; ♦ эвристические алгоритмы. Эвристические методы, такие как имитация отжига или жадные алгоритмы, можно использовать, чтобы быстро найти решение, но без га- рантии, что это решение будет наилучшим; ♦ аппроксимирующие алгоритмы. Теория NP-полноты только оговаривает сложность получения решения задачи. Но используя специализированные, ориентированные на конкретную задачу эвристические алгоритмы, скорее всего можно получить близкое к оптимальному решение для всех возможных экземпляров задачи. Аппроксимирующие алгоритмы обеспечивают решение с гарантией того, что опти- мальное решение не будет намного лучше. Таким образом, используя аппроксими- рующий алгоритм для решения NP-полной задачи, вы никогда не получите крайне не- удовлетворительный ответ. Независимо от входного экземпляра задачи, вы неизбежно будете совершать правильные действия. Кроме того, аппроксимирующие алгоритмы с удачно подобранными границами концептуально просты, работают быстро и легко поддаются программированию. Однако неясным остается следующее обстоятельство. Насколько решение, полученное с помощью аппроксимирующего алгоритма, сравнимо с решением, которое можно бы- ло бы получить с помощью эвристического алгоритма, не дающего никаких гарантий? Вообще говоря, оно может оказаться как лучше, так и хуже. Положив деньги в банк, вы получаете гарантию невысоких процентов прибыли без всякого риска. Вложив эти же деньги в акции, вы, скорее всего, получите более высокую прибыль, но при этом у вас не будет никаких гарантий. Один из способов получения наилучшего результата от аппроксимирующего и эври- стического алгоритмов заключается в том, что вы решаете данный экземпляр задачи с помощью каждого из них и выбираете тот алгоритм, который дает лучшее решение. Таким образом, вы получаете гарантированное решение и дополнительный шанс на получение лучшего решения. При применении эвристических алгоритмов для решения сложных задач вы можете рассчитывать надвойной успех. 9.10.1. Аппроксимация вершинного покрытия Как мы уже видели, задача поиска минимального вершинного покрытия графа являет- ся NP-полной. Но с помощью очень простой процедуры (листинг 9.8) можно быстро найти покрытие, которое, самое большее, вдвое больше оптимального. Листинг 9.8. Аппроксимирующий алгоритм поиска вершинного покрытия графа VertexCover(G = (V, Е) ) While (Е * 0) do: Выбираем произвольное ребро (u, v) < Е Добавляем обе вершины и и v к вершинному покрытию Удаляем из Е все ребра, входящие в и или v
Глава 9 Труднорешаемые задачи и аппроксимирующие алгоритмы 369 Должно быть очевидным, что данная процедура всегда выдает вершинное покрытие, т. к. каждое ребро удаляется только после добавления инцидентной вершины к покры- тию. Более интересным является утверждение, что любое другое вершинное покрытие должно содержать, по крайней мере, в два раза меньше вершин данного покрытия. По- чему? Рассмотрим только те ребра, выбранные алгоритмом, которые образуют паросо- четание в графе (а их не более и/2). Ни одна пара этих ребер не может иметь общую вершину. Поэтому любое покрытие, состоящее только из этих ребер, должно включать хотя бы одну вершину на каждое ребро, что делает его, по крайней мере, вдвое мень- шим, чем вершинное покрытие, полученное с помощью этого "жадного" алгоритма. Стоит отметить несколько интересных аспектов этого алгоритма. ♦ Хотя процедура проста, она не так глупа Производительность многих кажущихся более интеллектуальными эвристических алгоритмов может оказаться намного ниже в наихудшем случае. Например, почему бы не модифицировать эту процедуру, чтобы для получения вершинного покрытия вместо обеих вершин выбирать только одну из них? В конце концов, выбранное ребро будет с тем же успехом по- крыто только одной вершиной. Но посмотрим на звездообразный граф на рис. 9.9. Если не выбрать центральную вершину, вершинное покрытие окажется крайне не- удачным. Этот эвристический алгоритм выдаст двухвершинное покрытие, в то время как эвристический алгоритм, выбирающий одну вершину, выдаст покрытие размером до /? — 1 вершин, если нам не повезет, и алгоритм будет постоянно выбирать листо- вой узел вместо центрального в качестве вершины покрытия, которую следует со- хранить. ♦ "Жадный" алгоритм — это не всегда то, что нужно. Возможно, что самый естест- венный алгоритм поиска вершинного покрытия будет постоянно выбирать и уда- лять вершину наивысшей оставшейся степени для данного вершинного покрытия. В конце концов, эта вершина покроет наибольшее количество возможных ребер. Но в случае необходимости выбора между одинаковыми или почти одинаковыми вер- шинами этот эвристический алгоритм может значительно отклониться от правиль- ного пути. В наихудшем случае он может выдать покрытие больше оптимального в 0(ig«) раз. ♦ Усложнение эвристического алгоритма не обязательно улучшает его. Эвристиче- ский алгоритм легко усложнить, вставляя в него дополнительные возможности об- работки. Например, в аппроксимирующей процедуре в листинге 9.8 не указывается, какое ребро выбирается следующим. Может показаться разумным следующим вы-
370 Часть I. Практическая разработка алгоритмов бирать ребро с вершинами наивысшей степени. Но это не сузит границы наихудше- го случая и только сделает более трудным анализ работы алгоритма. ♦ Корректное завершение алгоритма позволяет улучшить результат. Дополнитель- ным преимуществом создания простых эвристических алгоритмов является то, что их часто можно модифицировать для получения лучших практических решений, при этом не ослабляя пределы аппроксимации. Например, операция постобработки, которая удаляет все ненужные вершины из покрытия, на практике может только улучшить результат, даже если она не принесет никакой пользы теоретическому пределу наихудшего случая. Важной характеристикой аппроксимирующих алгоритмов является отношение размера полученного решения к нижней границе оптимального решения. Не следует размыш- лять о том, насколько хорошим могло бы быть наше решение; мы должны думать о наихудшем случае, т. е. о том, насколько плохим оно могло бы быть. 9.10.2. Задача коммивояжера в евклидовом пространстве В большинстве естественных приложений задачи коммивояжера прямые маршруты по своей природе короче, чем обходные. Например, если в качестве веса ребер графа вы- брать расстояние между городами по прямой, то кратчайший путь отх к у всегда будет проходить по прямой. Веса ребер, назначаемые в соответствии с евклидовой геометрией, удовлетворяют ак- сиоме треугольника, т. е. для любой тройки вершин и. v и w выполняется неравенство с/(и, и ) < t/(u, v) + d(y, и’). Интуитивная очевидность этого условия демонстрируется на рис. 9.10. Аксиома треугольника d(u, vv) < d(u, v) + d(v, vv) обычно справедлива в гео- метрических задачах и задачах на взвешенных графах. Рис. 9.10. Аксиома треугольника Стоимость билета на авиарейс является примером функции расстояний, которая нару- шает аксиому треугольника, т. к. иногда транзитный рейс обходится дешевле, чем пря- мой полет до города назначения. Задача коммивояжера остается сложной, когда рас- стояния являются евклидовыми расстояниями на плоскости. Оптимальный маршрут коммивояжера можно аппроксимировать, используя мини- мальные остовные деревья или графы, подчиняющиеся аксиоме треугольника. Прежде всего, обратите внимание на то, что весом минимального остовного дерева является нижняя граница стоимости оптимального маршрута. Почему? Удаление любого ребра
Глава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 371 из маршрута оставляет путь, общий вес которого должен быть не большим, чем вес пеовоначального маршрута. Этот путь не содержит циклов, вследствие чего он являет- ся деревом, что означает, что его вес равен, по крайней мере, весу минимального ос- товного дерева. Таким образом, вес минимального остовного дерева дает нам нижнюю границу оптимального маршрута. Теперь рассмотрим, что происходит при обходе в глубину остовного дерева. Каждое ребро мы посещаем дважды: один раз спускаясь по дереву при открытии ребра, а вто- рой— возвращаясь наверх после исследования всего поддерева. Например, при обходе в глубину, показанном на рис. 9.11, вершины посещаются в следующем порядке: 1-2-1-3-5-8-5-9-5-3-6-3-1—4-7-10-7-11—7—4—1, т. е. каждое ребро проходится ровно два раза. Рис. 9.11. Обход в глубину остовного дерева Этот маршрут проходит по каждому ребру минимального остовного дерева дважды и, следовательно, стоит, самое большее, вдвое дороже, чем оптимальный маршрут. Но вершины на этом маршруте обхода в глубину будут повторяться. Чтобы удалить лишние вершины, на каждом шаге можно выбирать кратчайший путь к следующей непосещенной вершине. Кратчайший маршрут для дерева на рис. 9.11 проходит через вершины 1-2-3-5-8-9-6^4-7-10-11-1. Так как мы заменили последовательность ре- бер одним направленным ребром, то аксиома треугольника обеспечивает, что маршрут может стать только короче. Таким образом, этот кратчайший маршрут также находится в пределах допустимого веса и стоит вдвое дороже оптимального маршрута. Что еще лучше, для решения евклидовой версии задачи коммивояжера существуют более сложные алгоритмы, которые рассматриваются в разделе 16.4. Среди аппроксимирую- щих алгоритмов для решения задачи коммивояжера не известен ни один, который не удовлетворяет аксиому треугольника. 9.10.3. Максимальный бесконтурный подграф Бесконтурные орграфы легче поддаются обработке, чем общие орграфы. Иногда будет полезным упростить данный граф, удалив небольшой набор ребер или вершин, доста- точный для разрыва контуров (циклов). Такие задачи о разрывающем множестве дуг (feedback set) обсуждаются в разделе 16.11.
372 Часть I. Практическая разработка алгоритмов Здесь мы рассмотрим интересную задачу из этого класса, в которой при разрыве всех ориентированных контуров нужно оставить как можно больше ребер. Постановка за- дачи выглядит таким образом: ЗАДАЧА. Найти максимальный бесконтурный ориентированный подграф. Вход. Ориентированный граф G = (V, Е). Выход. Самое большое подмножество Е' е Е, для которого G'= (V. Е') не содержит контуров. Существует очень простой алгоритм, который гарантирует решение с количеством ре- бер равным, по крайней мере, половине оптимального. Я рекомендую вам попробовать разработать такой алгоритм самостоятельно, прежде чем вы прочитаете его описание. Итак, создаем любую перестановку вершин и рассматриваем ее как упорядочивание слева направо, аналогично топологической сортировке. Теперь некоторые ребра будут направлены слева направо, в то время как оставшиеся ребра направлены справа налево. Размер одного из этих подмножеств ребер должен быть, по крайней мере, таким же, как и другого. Это означает, что оно содержит хотя бы половину ребер. Кроме того, каждое из этих двух подмножеств ребер должно быть бесконтурным. т. к. только бес- контурные орграфы можно подвергать топологической сортировке— контур (цикл) нельзя создать, постоянно двигаясь в одном направлении. Таким образом, подмноже- ство ребер большего размера должно быть бесконтурным и содержать, по крайней ме- ре, половину ребер оптимального решения! Данный аппроксимирующий алгоритм чрезвычайно прост. Но обратите внимание, что на практике применение эвристики может улучшить его производительность, при этом сохраняя гарантию качества результата. Возможно, имеет смысл попробовать несколь- ко произвольных перестановок и выбрать ту, которая дает наилучший результат. Кроме того, можно попытаться обменивать местами пары вершин в перестановках и остав- лять те варианты обмена, которые добавляют большее количество ребер в подмноже- ство большего размера. 9.10.4. Задача о покрытии множества Предыдущие разделы могут создать ложное представление, что с помощью аппрокси- мирующих алгоритмов можно получить решение любой задачи с коэффициентом от- личия от оптимального равным двум. Однако решение некоторых задач из каталога, например задачи поиска максимальной клики, нельзя аппроксимировать с наперед за- данным коэффициентом. Задача о покрытии множества занимает промежуточную позицию между этими двумя крайностями, поскольку имеет алгоритм, выдающий решение с коэффициентом отли- чия от оптимального, равным 0(lgn). Задача о покрытии множества представляет собой общий случай задачи о вершинном покрытии. Формальная постановка задачи (рас- сматриваемой в разделе 18.1} выглядит таким образом: ЗАДАЧА. Найти покрытие множества. Вход. Семейство подмножеств 5 = {£,,..., S,,,} универсального множества U= {1,.... «}.
Гпава 9 Труднорешаемые задачи и аппроксимирующие алгоритмы 373 Выход. Подмножество наименьшей мощности Т семейства S, чье объединение равно универсальному множеству, т. е. = U. Будет естественным применить для решения этой задачи эвристический алгоритм "жадного" типа. В частности, постоянно выбираем подмножество, которое покрывает наибольшее подсемейство непокрытых на данном этапе элементов, пока не получим полное покрытие. Псевдокод соответствующего алгоритма показан в листинге 9.9. Листинг 8.9. Аппроксимирующий алгоритм поиска покрытия множества Set Cover (S) While (0 * 0) do: Определяем подмножество Si, имеющее наибольшее пересечение z множеством U Выбираем Si для покрытия множества U = U - Si Одним из побочных эффектов этого процесса выбора является тот факт, что по мере исполнения алгоритма количество покрытых на каждом шаге элементов образует не- возрастающую последовательность. Почему? Потому что в противном случае "жадный" алгоритм выбрал бы более мощное подмножество раньше, если бы оно дей- ствительно существовало. Таким образом, данный эвристический алгоритм можно рассматривать как уменьше- ние количества непокрытых элементов от п до нуля, причем невозрастающими пор- циями. Пример трассировки исполнения такого алгоритма показан в табл. 9.1. Таблица 9.1. Работа "жадного" алгоритма на экземпляре задачи о покрытии множества К иочевое событие 6 5 4 3 2 1 0 Непокрытые элементы 64 51 40 30 25 22 19 16 13 10 7 4 2 1 Размер выбранного подмножества 13 11 10 5 3 3 3 3 3 3 3 2 1 1 Важное контрольное событие этой трассировки происходит, как только количество оставшихся непокрытых элементов уменьшается в 2" раз. Очевидно, что таких событий может быть, самое большее. |~lg. Пусть та, обозначает количество подмножеств, выбранных нашим эвристическим алго- ритмом для покрытия элементов между контрольными событиями, соответствующими уменьшению в 2'+1 - 1 раз и 2' раз. Определим максимальную ширину vv, как та, где 0<i< lg?». В примере трассировки в табл. 9.1 максимальная ширина представлена пя- тью подмножествами, необходимыми для перехода от 25 - 1 к 24. Так как существует, самое большее, Ign таких контрольных событий, то выдаваемое "жадным" эвристическим алгоритмом решение должно содержать, самое большее, wlgn подмножеств. Но я утверждаю, что оптимальное решение должно содержать, по крайней мере, и подмножеств, так что выдаваемое эвристическим алгоритмом реше- ние хуже оптимального не больше, чем в Ign раз.
374 Часть I. Практическая разработка алгоритмов Почему? Рассмотрим среднее количество новых элементов, покрываемых при проходе между контрольными событиями, соответствующими уменьшению в 2'+1 - 1 раз и 2 раз. Для этих 2' элементов требуется w, подмножеств, поэтому мощность среднего по- крытия равна д, = 2'/и',. Более того, последнее (наименьшее) из этих подмножеств по- крывает. самое большее, д, подмножеств. Таким образом, множество S’ не содержит подмножества, которое может покрыть больше, чем д, из оставшихся 2' элементов. По- этому, чтобы получить решение, нам нужно, по крайней мере. 27д, = w, подмножеств. Несколько неожиданным является то обстоятельство, что на самом деле существуют экземпляры покрытия множества, для которых время исполнения этого эвристического алгоритма в Q(lg«) раз больше оптимального. Этот логарифмический множитель явля- ется свойством задачи и эвристического алгоритма для ее решения, а не следствием плохо проведенного анализа. Подведение итогов Аппроксимирующие алгоритмы гарантируют решения, которые всегда близки к опти- мальному. Они могут предоставить практический подход к решению NP-полных задач. Замечания к главе Понятие NP-полноты было впервые сформулировано Стивеном Куком (Stephen Cook) в 1971 г. (см. [Соо71]). Задача выполнимости действительно является задачей на один миллион долларов, т. к. институт Clay Mathematical Institute предлагает премию такого размера любому, кто решит вопрос Р = NP. Подробности о задаче и призе за ее реше- ние см. по адресу http://www.claymath.org/millennium/P_vs_NP/. В 1972 г. Ричард Карп (Richard Karp) продемонстрировал значимость работы Кука, предоставив сведение от задачи выполнимости к каждой из более чем 20 важных алго- ритмических задач. Я рекомендую вам ознакомиться с докладом Карпа ([Каг72]) в силу его исключительного изящества и краткости— он излагает каждое сведение в трех строчках описания, показывающего эквивалентность задач. Работы Кука и Карпа, взя- тые вместе, предоставляют инструменты для разрешения вопроса сложности буквально сотен важных задач, для решения которых не было известно эффективных алгоритмов. Наилучшим введением в теорию NP-полноты остается книга "Computers and Intractability" ([GJ79]). В ней дается введение в общую теорию, включая доступное до- казательство теоремы Кука ([Соо71]) о том, что сложность задачи выполнимости такая же. как и любой другой задачи класса NP. Книга также содержит важный справочный каталог свыше 300 NP-полных задач, что является хорошим материалом для изучения известных фактов о наиболее интересных задачах. Задачи сведения, упомянутые, но не рассмотренные в этой главе, можно найти в книгах [GJ79] и [CLRS01]. Несколько задач из каталога находятся в состоянии неопределенности, т. е. неизвестно, существует ли эффективный алгоритм для решения конкретной задачи или же она яв- ляется NP-полной. Наиболее значительными из этих задач являются задачи изомор- физма графов (см. раздел 16.9) и разложения целых чисел на множители (см. раз- дел 13.8). Краткость этого списка задач с неопределенным статусом в большой степени является следствием современного уровня развития дисциплины разработки алгорит-
Глава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 375 мов. Почти для каждой важной задачи у нас имеется либо эффективный алгоритм, ли- бо веская причина его отсутствия. Задача об односвязном подграфе, которая рассматривалась в разделе 9.8. была впервые доказана в техническом докладе [МаЬ76]. 9.11. Упражнения Преобразования и выполнимость 1. [2] Предоставьте формулу 3-SAT, которая получается в результате применения сведения задачи SAT к задаче 3-SAT для следующей формулы: (х + у + z + w + и + v )•( х + у + : + и' + и + v)-(x + у + ~ + w + и + v )-(х + у ). 2, [3] Нарисуйте граф, который получается в результате сведения задачи 3-SAT к задаче о вершинном покрытии для выражения (х + у + ;)•( х +у+ z )-(х +y + z)'(x+ у + х). 3. [4] Допустим, что у нас имеется процедура, которая может определить разрешимость задачи коммивояжера из раздела 9.1.2 за линейное время. Предоставьте эффективный алгоритм поиска самого маршрута, вызывая эту процедуру полиномиальное число раз. 4. [7] Реализуйте процедуру преобразования экземпляров задач выполнимости в эквива- лентные экземпляры задачи 3-SAT. 5. [7] Разработайте и реализуйте алгоритм поиска с возвратом для проверки набора формул на выполнимость. Какие критерии можно использовать для отсечения пространства по- иска? 6. [8] Реализуйте процедуру сведения задачи о вершинном покрытии к задаче выполнимо- сти и проверьте получившиеся дизъюнкции программой проверки на выполнимость. На- сколько такой способ применим на практике для выполнения подобных вычислений? Базовые сведения 7. [4] Дано: множество X из п элементов, семейство F подмножеств множества X и целое число к. Существует ли к подмножеств семейства F, объединение которых равно Л? Например, если X = {1,2,3,4} и F= {{1,2},{2,3},{4},{2,4}}, для к = 2 решения нет, но для к = 3 есть (например, {1,2}, {2,3}, {4}). Докажите, что задача о покрытии множества является NP-полной, выполнив сведение с задачи вершинного покрытия. 8. [4] Задача коллекционера бейсбольных карточек выглядит таким образом. Имеются па- кеты бейсбольных карточек Рь ..., Р„„ каждый из которых содержит подмножество кар- точек. выпушенных в указанном году. Возможно ли собрать все карточки за этот год. купив не более чем к таких пакетов? Например, для игроков {Aaron, Mays. Ruth. Skiena} и пакетов {{Aaron, Mays}, {Mays, Ruth}, {Skiena}, {Mays, Skiena}}, для к = 2 решения нет, но для к = 3 есть, такое как {Aaron, Mays}, {Mays, Ruth}, {Skiena}
376 Часть I. Практическая разработка алгоритмов Докажите, что задача коллекционера бейсбольных карточек является NP-полной, ис- пользуя для этого сведение к ней от задачи о вершинном покрытии. 9. [4] Приведем формулировку задачи остовного Оерева низкой степени (low-degree spanning tree problem). Дано: граф G и целое число к. Содержит ли граф G такое остов- ное дерево, степень всех вершин которого равна, самое большее, к? (Очевидно, что сте- пень определяется только по количеству ребер.) Например, граф, представленный на рис. 9.12, не содержит остовного дерева, степень всех вершин которого меньше чем три. Рис. 9.12. Пример графа а) Докажите, что задача остовного дерева низкой степени является NP-полной, исполь- зуя сведение от задачи о гамильтоновом пути. б) Теперь сформулируем задачу остовного дерева высокой степени (high-degree span- ning tree problem). Дано: граф G и целое число к. Содержит ли граф G остовное дерево, у которого максимальная степень вершины равна, как минимум, А? В предыдущем при- мере обсуждалось остовное дерево с наивысшей степенью, равной 8. Предоставьте эф- фективный алгоритм для решения задачи остовного дерева высокой степени и выпол- ните анализ его временной сложности. 10. [4] Докажите, что следующая задача является NP-полной: ЗАДАЧА. Найти плотный подграф. Вход. Граф G и два целых числа к и у. Выход. Содержит ли граф G подграф, имеющий ровно к вершин и, по крайней мере, у ребер? 11. [4] Докажите, что следующая задача является NP-полной: ЗАДАЧА. Найти клику. Вход. Неориентированный граф G = (Г, £) и целое число к. Выход. Содержит ли граф G как клику, так и независимое множество размером А? 12. [5] Эйлеровым циквоы (Eulerian cycle) называется маршрут, который проходит по каж- дому ребру графа ровно один раз. Эйлеровым подграфом (Eulerian subgraph) называется подмножество ребер и вершин графа, которое содержит эйлеров цикл Докажите, что задача определения количества ребер в наибольшем эйлеровом подграфе графа являет- ся NP-сложной. (Подсказка: задача о гамильтоновом цикле является NP-сложной, даже если каждой вершине графа инцидентны три ребра.) Нестандартные сведения 13. [5] Докажите, что следующая задача является NP-полной: ЗАДАЧА. Найти минимальное множество представителей (hitting set) Вход. Коллекция С подмножеств множества S, положительное число к.
Глава 9. Труднорешаемые задачи и аппроксимирующие алгоритмы 377 Выход. Содержит ли множество S такое подмножество S’, для которого |S| < к и каждое подмножество в коллекции С содержит, по крайней мере, один элемент из подмножест- ва S'? 14. [5] Докажите, что следующая задача является NP-полной: ЗАДАЧА. Задача о рюкзаке. Вход. Множество S, состоящее из и элементов, в котором <-й элемент имеет ценность v, и вес и’,. Два положительных целых числа: максимальный вес И и требование к ценно- сти груза К Выход. Существует ли такое подмножество S'eS, для которого „w, < И7 и v, > I7 ? (Подсказка: сначала рассмотрите задачу разделения множества целых чисел.) 15. [5] Докажите, что следующая задача является NP-полной: ЗАДАЧА. Найти гамильтонов путь. Вход. Граф G и вершины хи/. Выход. Содержит ли граф G путь с началом в вершине s и концом в вершине /, который проходит через все вершины не более одного раза? (Подсказка: начните с задачи о гамильтоновом цикле.) 16. [5] Докажите, что следующая задача является NP-полной: ЗАДАЧА. Найти самый длинный путь. Вход. Граф G и положительное целое число к. Выход. Содержит ли граф G путь, который соединяет, по крайней мере, к разных вер- шин, не проходя через каждую из них более одного раза? 17. [6] Докажите, что следующая задача является NP-полной: ЗАДАЧА. Найти доминирующее множество. Вход. Граф G = (Г, £) и положительное целое число к. Выход. Существует ли такое подмножество Р е Г, где |Г1< А, в котором для каждой вершины л- е I7 или х g Г' или существует ребро (х, у), где у g Г'. 18 [7] Докажите, что задача о вершинном покрытии (существует ли такое подмножество S. состоящее из к вершин графа G, в котором каждое ребро в графе G инцидентно, по крайней мере, одной вершине в S?) остается NP-полной, даже если степень всех вершин в графе может быть только четной? 19. [7] Докажите, что следующая задача является NP-полной: ЗАДАЧА. Задача упаковки множества (set packing problem). Вход. Коллекция С подмножеств множества S, положительное число к. Выход. Содержит ли множество S, по крайней мере, к непересекающихся подмножеств, т. е. подмножеств, не имеющих общих элементов? 20. [7] Докажите, что следующая задача является NP-полной: ЗАДАЧА. Найти разрывающее множество вершин. Вход. Ориентированный граф G = (К, А) и положительное целое число к.
378 Часть I. Практическая разработка алгоритмов Выход. Существует ли такое подмножество V' е И, где |К’| < к, удаление вершин кото- рого из графа G оставляет бесконтурный орграф? Алгоритмы для решения частных случаев задач 21. [5] Гамильтонов путь Р проходит через каждую вершину ровно один раз. Задача поиска гамильтонова пути в графе G является NP-полной. В отличие от гамильтонова цикла, гамильтонов путь не обязан содержать ребро, соединяющее начальную и конечную вершины пути Р. Предоставьте алгоритм с временем исполнения О(п + т) для проверки бесконтурного орграфа G на наличие в нем гамильтонова пути. (Подсказка: ваши рассуждения должны идти в направлении топологической сортировки и обхода в глубину.) 22. [8] В задаче 2-SAT требуется выяснить, является ли выполнимой данная булева форму- ла в 2-конъюктивной нормальной форме (КНФ). Задача 2-SAT подобна задаче 3-SAT только каждая дизъюнкция может содержать всего лишь два литерала. Например, сле- дующая формула записана в 2-КНФ: (*1 V Х2) А ( X 2 V Л’з) A (.V| v L) Предоставьте алгоритм с полиномиальным временем исполнения для решения задачи 2-SAT. Р или NP? 23. [4] Покажите, что следующие задачи принадлежат классу NP: • Содержит ли граф G простой путь (т. е. путь, в котором вершины не повторяются) длиной А9 • Является ли целое число п составным (т. е. не простым)? • Содержит ли граф G вершинное покрытие размером А? 24. [7] В течение длительного времени оставался открытым вопрос, возможно ли за поли- номиальное время найти решение такой задачи разрешимости: "Является ли целое чис- ло и составным, т. е. не простым?" Почему следующий алгоритм не доказывает, что эта задача принадлежит классу Р, хотя он и исполняется за время О(”)? PrimalityTesting(n) composite := false for i := 2 to n 1 do if (n mod i) =0 then composite := true Аппроксимирующие алгоритмы 25. [4] В задаче максимальной выполнимости требуется найти набор значений истинности, который является выполняющим для максимального количества дизъюнкций. Предос- тавьте эвристический алгоритм, выполняющий, как минимум, вдвое меньше дизъюнк- ций, чем выполняет оптимальное решение. 26. [5] Имеется следующий эвристический алгоритм поиска вершинного покрытия: созда- ется дерево обхода в глубину графа, из которого удаляются все листья; оставшиеся узлы
Гпава 9, Труднорешаемые задачи и аппроксимирующие алгоритмы 379 должны составлять вершинное покрытие графа. Докажите, что размер этого вершинно- го покрытия превышает оптимальный, самое большее, в два раза. 27. [5] В задаче максимального разреза графа G = (И, £) требуется разделить множество вершин Т на два непересекающихся подмножества А и В таким образом, чтобы макси- мизировать количество ребер (а. Ь) е £, где a е A, b е В. Рассмотрим следующий эври- стический алгоритм поиска максимального разреза. Сначала помещаем вершину v1 в подмножество А, а вершину v2 — в подмножество В. Каждую оставшуюся вершину помещаем в подмножество, которое добавляет в разрез наибольшее количество ребер. Докажите, что размер данного разреза равен, по крайней мере, половине оптимального разреза. 28. [5] В задаче об упаковке в контейнеры дается п элементов, веса которых равны ггь из, ..., м’„. Нужно найти наименьшее количество контейнеров, в которые можно упако- вать эти п элементов, при этом емкость каждого контейнера не превышает один килограмм. В эвристическом алгоритме "первый подходящий" (first-fit) объекты рассматриваются в порядке, в котором они представляются. Каждый объект укладывается в первый же контейнер, в который он помещается. Если такого контейнера нет. объект помещается в новый (пустой) контейнер. Докажите, что количество контейнеров, выдаваемое этим эвристическим алгоритмом, превышает оптимальное не более, чем в два раза. 29. [5] Для только что описанного эвристического алгоритма "первый подходящий” приве- дите пример экземпляра задачи, решение которого дает количество контейнеров, пре- вышающее оптимальное максимум в 5/3 раз. 30. [5] Целью задачи раскраски вершин графа G = (Е, £) является назначение цветов вер- шинам множества Г таким образом, чтобы вершины на концах каждого ребра были окрашены в разные цвета. Предоставьте алгоритм для раскраски графа G не более чем Д + 1 разными красками, где Д представляет максимальную степень вершин графа G. Задачи по программированию Эти задачи доступны на сайтах http://www.programming-challenges.comcom и http://uva.onlinejudge.org. Геометрия 1, The Monocycle. 111202/10047. 2. Dog and Gopher. 10310/111301. 3. Chocolate Chip Cookies. 111304/10136. 4. Birthday Cake. 111305/10167. Вычислительная геометрия 5. Closest Pair Problem. 111402/10245. 6. Chainsaw Massacre. 111403/10043. 7. Hotter Colder. 111404/10084. 8. Useless Tile Packers. 111405/10065. Примечание Эти задания не имеют непосредственного отношения к теме NP-полноты, но добавлены для полноты картины.
ГЛАВА 10 Как разрабатывать алгоритмы Разработка правильною алгоритма для определенного приложения в значительной степени является творческой деятельностью. Мы получаем задачу и придумываем ре- шение для нее. Пространство выбора при разработке алгоритмов огромно, и у вас есть много возможностей допустить ошибку. Цель этой книги состоит в том, чтобы повысить вашу квалификацию разработчика ал- горитмов. Представленные в первой части этой книги методы содержат идеи, лежащие в основе всех комбинаторных алгоритмов. А каталог задач во второй части поможет вам моделировать ваши приложения, предоставляя информацию о соответствующих задачах. Однако успешному разработчику алгоритмов требуется нечто большее, чем книжные знания. Для этого нужен определенный склад ума— правильный подход к решению задач. Этому трудно научиться по книгам, но обладание таким качеством яв- ляется необходимым условием. Ключом к разработке алгоритмов (или любой другой деятельности, связанной с ре- шением задач) является умение задавать самому себе вопросы, чтобы направлять свой процесс мышления: "Что будет, если я попробую такой способ? А если теперь попробовать другой?" Когда на каком-либо этапе вы столкнетесь с затруднениями, лучше всего перейти к следующему вопросу. Самым полезным участником любой мозговой атаки является тот, кто постоянно задает вопрос: "А почему нельзя приме- нить такой-то способ?", а не тот, кто дает ответ на этот вопрос. В конце концов, первый участник предложит подход, который второй не сможет забраковать. С этой целью мы предоставляем последовательность вопросов, которая должна на- правлять вашу деятельность в поиске правильного алгоритма для решения стоящей перед вами задачи. Но чтобы эти вопросы были эффективными, мало их задавать, нужно отвечать на них. Важно тщательно прорабатывать вопросы, записывая их в журнал. Правильным ответом на вопрос: "Можно ли сделать это таким способом?" является не просто: "Нет", а "Нет, потому что...". Четко излагая свои мысли по поводу отрицательного результата, вы сможете проверить, не упустили ли вы вариант, о кото- ром просто не подумали. Поразительно, как часто причиной отсутствия у вас правиль- ного объяснения является ошибочное умозаключение. В процессе разработки важно осознавать разницу между стратегией и тактикой и постоянно помнить о ней. Стратегия — это поиск общей картины, основы, на которой строится решение. А тактика используется для решения второстепенных вопросов на пути к глобальной цели. В процессе решения задач важно постоянно проверять, на правильном ли уровне находятся ваши рассуждения. Если у вас нет глобального пла- на (т. е. стратегии) для решения поставленной перед вами задачи, то нет никакого смысла размышлять о правильной тактике. Примером стратегического вопроса явля-
Глава 10 Как разрабатывать алгоритмы 381 ется: "Могу ли я смоделировать данное приложение в виде задачи создания алгорит- ма на графах?" А тактическим вопросом может быть: "Какую структуру данных следу- ет использовать для представления моего графа, список смежности или матрицу смеж- ности?" Конечно же, тактические решения критичны для качества конечного решения, но их можно должным образом оценить только в свете успешной стратегии. У слишком многих людей при необходимости решить задачу пропадают все мысли. Прочитав задачу, они приступают к ее решению и вдруг осознают, что не знают, что делать дальше. Чтобы не попадать в такие ситуации, следуйте списку вопросов, изложенных далее в этой главе и в большинстве разделов каталога задач. Они подска- жут вам, что делать дальше. Очевидно, что чем больше у вас имеется опыта по использованию методов разработки алгоритмов, таких как динамическое программирование, алгоритмы на графах и струк- туры данных, тем с большим успехом сможете вы следовать этим вопросам. Первая часть этой книги предназначена для развития и укрепления этих технических основ. Но независимо от уровня вашей технической подготовки будет полезно следовать этим вопросам. Самые первые и наиболее важные вопросы в этом списке помогут вам до- биться глубокого понимания решаемой задачи и не потребуют никакого специального опыта. Сам список вопросов был создан под влиянием отрывка из замечательной книги о про- грамме космических исследований США, называющейся "The Right Stuff1 ("Правиль- ный материал") ([Wol79]). В данном отрывке описывались радиопереговоры пилотов- испытателей непосредственно перед крушением самолета. Можно было бы ожидать, что в данной ситуации они впадали в панику, но вместо этого пилоты выполняли действия из списка для нештатных ситуаций: "Проверил закрылки. Проверил двига- тель. Крылья в порядке. Сбросил..." Эти пилоты обладали "Правильным материалом" и благодаря этому иногда им удавалось избежать гибели. Я надеюсь, что эта книга и список вопросов помогут вам обрести "Правильный мате- риал" разработчика алгоритмов. Я также надеюсь, что это поможет вам избежать крушений в вашей деятельности. Список вопросов разработчика алгоритмов 1. Понимаю ли я задачу? • Из чего состоит вход? • Какой именно требуется результат? • Могу ли я создать достаточно небольшой входной экземпляр, чтобы его можно было решить вручную? Что будет, если я попытаюсь решить его? • Насколько важно для моего приложения всегда получать оптимальный ре- зультат? Устроит ли меня результат, близкий к оптимальному? • Каков размер типичного экземпляра решаемой задачи? 10 элементов? 1 000 эле- ментов? 1 000 000 элементов? • Насколько важной является скорость работы приложения? Задачу нужно ре- шить за одну секунду, одну минуту, один час или за один день?
382 Часть I. Практическая разработка алгоритмов • Сколько времени и усилий могу я вложить в реализацию? Буду ли я огра- ничен простыми алгоритмами, которые можно реализовать в коде за один день, или у меня будет возможность поэкспериментировать с разными под- ходами и посмотреть, какой из них лучше? • Задачу какого типа я пытаюсь решить? Численную задачу, задачу теории графов, геометрическую задачу, задачу со строками или задачу на множестве? Какая формулировка выглядит самой легкой? 2. Могу ли я найти простой детерминированный или эвристический алгоритм для решения стоящей передо мной задачи? • Можно ли правильно решить задачу методом полного перебора всех возможных решений? D Если можно, то почему я уверен, что этот алгоритм всегда выдает правильное решение? ° Как оценить качество созданного решения? ° Каким будет время исполнения этого простого, но медленного алгорит- ма— полиномиальным или экспоненциальным? Настолько ли проста моя задача, чтобы ее можно было решить методом полного перебора? ° Уверен ли я, что моя задача достаточно четко определена, чтобы правильное решение было возможной • Можно ли решить данную задачу, постоянно применяя какое-либо простое правило, например, выбирая в первую очередь наибольший элемент, наи- меньший элемент или произвольный элемент? ° Если можно, то при каком входе этот эвристический алгоритм работает хо- рошо? Соответствует ли вход такого типа данным, актуальным для моего приложения? ° При каком входе этот эвристический алгоритм работает плохо? Если не удастся найти примеры плохих входных экземпляров, то могу ли я пока- зать. что алгоритм всегда работает хорошо? с Сколько времени требуется моему эвристическому алгоритму, чтобы вы- дать ответ? Можно ли его реализовать простым способом? 3. Есть ли моя задача в каталоге задач этой книги? • Что известно о данной задаче? Существует ли реализация решения, которую я могу использовать? • Искал ли я свою задачу в правильном месте каталога? Просмотрел ли я все рисунки? Проверил ли я все возможные ключевые слова в алфавитном указа- теле? • Существуют ли в Интернете ресурсы, имеющие отношение к моей задаче? Искал ли я в Google и Google Scholar? Посетил ли я веб-сайт этой книги, http:// www.cs.sunysb.edu/~algorith?
Глава 10. Как разрабатывать алгоритмы 383 4. Существуют ли специальные случаи стоящей передо мной задачи, которые я знаю, как решить? • Можно ли эффективно решить задачу, игнорируя некоторые входные пара- метры? • Станет ли задача проще, если присвоить некоторым входным параметрам три- виальные значения, такие как 0 или 1? • Сумею ли я упростить задачу до такой степени, что смогу решить ее эффек- тивно? • Почему алгоритм для решения этого специального случая нельзя обобщить для более широкого диапазона входных экземпляров? • Не является ли моя задача специальным случаем одной из общих задач ката- лога? 5. Какая из стандартных парадигм разработки алгоритмов наиболее соответствует мо- ей задаче? • Существует ли набор элементов, который можно отсортировать по размеру или какому-либо ключу? Способствует ли отсортированный набор данных более легкому решению задачи? • Можно ли разбить задачу на две меньших задачи, например, посредством дво- ичного поиска? Поможет ли метод разделения элементов на большие и малень- кие или правые и левые? Может быть, стоит попробовать алгоритм типа "разде- ляй и властвуй"? • Не обладают ли входные объекты или требуемое решение естественным поряд- ком слева направо, как, например, символы в строке, элементы перестановки или листья дерева? Могу ли я применить динамическое программирование, чтобы воспользоваться этим упорядочиванием? • Нет ли повторяющихся операций, таких как поиск наименьшего/наибольшего элемента? Могу ли я применить специальные структуры данных для ускорения этих операций, например, словари/таблицы хэширования или пирамиды/очереди с приоритетами? • Могу ли я применить случайную выборку следующего объекта? Могу ли я соз- дать несколько произвольных конфигураций и выбрать наилучшую из них? Мо- гу ли я применить какой-либо вид управляемой случайности, вроде метода ими- тации отжига, чтобы сосредоточить усилия на наилучшем решении? • Могу ли я сформулировать свою задачу, как линейную программу или как цело- численную программу? • Имеет ли моя задача какое-либо сходство с задачей выполнимости, задачей ком- мивояжера или какой-либо другой NP-полной задачей? Может ли задача быть NP-полной и вследствие этого не иметь эффективного алгоритма решения? Есть ли эта задача в списке из приложения к книге [GJ79]?
384 Часть I. Практическая разработка алгоритмов 6. Я все еще не знаю, что делать? • Могу и хочу ли я потратить деньги, чтобы нанять специалиста, чтобы он пояснил мне, что делать? Если можете, то ознакомьтесь со списком профессиональных консалтинговых услуг в разделе 19.4. • Не пройти ли заново по списку вопросов? Прежние ли ответы даю я при новом проходе по списку? Решение задач не является точной наукой. Отчасти это искусство, а отчасти — ре- месло. Навык решения задач — это один из полезнейших навыков, и его стоит приобрести. Моей любимой книгой по решению задач является книга "How to Solve It" ("Как решить задачу") [Р57], содержащая каталог методов решения задач. Я обожаю его просматривать, как с целью решения конкретной задачи, так и просто ради удо- вольствия.
13 Зак. 3741 ______________ЧАСТЬ 11 Каталог алгоритмических задач
ГЛАВА 11 Описание каталога Эта часть книги содержит каталог алгоритмических задач, которые часто возникают на практике. Здесь приводится общая информация по каждой задаче и даются советы, как действовать, если та или иная задача возникнет в вашем приложении. Как использовать этот каталог? Если вы знаете название вашей задачи, то с помощью алфавитного указателя или оглавления найдите ее описание. Прочитайте весь матери- ал, относящийся к задаче, поскольку он содержит ссылки на связанные с ней задачи. Пролистайте каталог, изучая рисунки и названия задач. Возможно, что-то покажется подходящим для вашего случая. Смело пользуйтесь алфавитным указателем, — каждая задача внесена в него несколько раз под разными ключевыми словами. Если вы все еще не можете найти ничего подходящего, значит, либо ваша задача не годится для решения комбинаторными алгоритмами, либо вы не полностью понимаете ее. В любом случае возвращайтесь назад к исходному этапу поиска решения. Каталог содержит обширную информацию разных типов, которая никогда раньше не собиралась в одном месте. Каждая задача сопровождается рисунком, представляющим входной экземпляр задачи и результат его решения. Эти стилизованные примеры иллюстрируют задачи лучше, чем формальные описания. Так, пример, касающийся минимального остовного дерева, демонстрирует кластеризацию точек с помощью минимальных остовных деревьев. Мы надеемся, что читатели нашей книги после беглого просмотра рисунков смогут найти то, что им нужно. Если вы нашли подходящую для вашего приложения задачу, прочитайте раздел "Об- суждение". В нем описаны ситуации, в которых может возникнуть данная задача, и особые случаи входных данных. Также рассказано, какой результат можно ожидать и, что еще более важно, как его добиться. Для каждой задачи дано краткое описание не- сложного алгоритма, которым можно ограничиться в простых случаях, и ссылки на более мощные алгоритмы, чтобы вы могли ими воспользоваться, если первое решение окажется недостаточно эффективным. Кроме этого, для каждой задачи перечислены существующие программные реализа- ции. Многие из этих процедур весьма удачны, и их код можно вставлять непосредст- венно в ваше приложение. Другие могут не отвечать требованиям промышленного применения, но я надеюсь, что они послужат хорошей моделью для вашей собственной реализации. Вообще, программные реализации приведены в порядке убывания их цен- ности, но при наличии бесспорно лучшего варианта мы отмечаем его. В главе 19 для многих из этих реализаций дается более подробная информация. Практически для всех реализаций на сайте этой книги (http://www.cs.sunysb.edu/~algorith) даются ссылки для их загрузки.
388 Часть II. Каталог алгоритмических задач Наконец, в примечаниях к задаче рассказывается ее история и приводятся результаты, представляющие, в основном, теоретический интерес. Мы постарались привести наи- лучшие известные результаты для каждой задачи и дать ссылки на работы по теорети- ческому и эмпирическому сравнению алгоритмов, если таковые имеются. Эта инфор- мация должна представлять интерес не только для студентов и исследователей, но так- же и для практиков, не удовлетворившихся рекомендуемыми решениями. Предостережения Данный материал является каталогом алгоритмических задач, а не сборником рецеп- тов, поскольку существует слишком много различных задач и вариантов их решений. Моя цель — указать вам правильное направление, чтобы вы могли решать свои задачи самостоятельно. Я попытался очертить круг вопросов, с которыми вы столкнетесь, ре- шая задачи самостоятельно. ♦ Для каждой задачи я дал рекомендацию, какой алгоритм следует использовать. Эти советы основаны на моем опыте и касаются приложений, которые я считаю типич- ными. При работе над книгой я полагал, что гораздо важнее дать конкретные реко- мендации для типичных случаев, чем пытаться охватить все возможные ситуации. Если вы не согласны с моим советом, поступайте по-своему. Однако вначале поста- райтесь понять идеи, лежащие в основе моих советов, и сформулировать причину, по которой ваше приложение не соответствует им. ♦ Рекомендуемые мною реализации не обязательно являются исчерпывающими ре- шениями вашей задачи. Я предлагаю реализацию, когда считаю, что она может быть полезной в большей степени, чем просто описание алгоритма. Некоторые про- граммы пригодны только в качестве моделей для создания вашего собственного ко- да. Другие встроены в большие системы и их может быть слишком трудно извлечь и исполнять отдельно. Исходите из предположения, что все решения содержат ошибки. Многие из этих ошибок могут оказаться довольно серьезными, так что будьте начеку. ♦ Вы обязаны соблюдать условия лицензии любой реализации, которую вы исполь- зуете в коммерческих целях. Многие из этих программ не являются свободно рас- пространяемыми и имеют лицензионные ограничения. Подробности см. в разде- ле 19.1. ♦ Мне бы хотелось узнать о результатах ваших действий согласно моим рекоменда- циям, как о положительных, так и об отрицательных. Особый интерес для меня представляет информация о других реализациях, неизвестных мне. но известных вам. Пишите мне по адресу feedback(a)algorist.com. 4
ГЛАВА 12 Структуры данных Структуры данных— это базовые конструкции для построения приложений. Чтобы максимально полно использовать возможности стандартных структур данных, необхо- димо иметь о них ясное представление. Поэтому мы сначала подробно рассмотрим разные структуры данных, а затем перейдем к обсуждению других задач. Возможно, что наиболее важным аспектом этого каталога будут предоставленные в нем ссылки на разные программные реализации и библиотеки структур данных. Хоро- шая реализация многих структур данных — дело далеко не тривиальное, поэтому про- граммы. на которые даются ссылки, будут полезными в качестве моделей, даже если они и не в точности подходя г для ваших задач. Некоторые важные структуры данных, такие как kd-деревья и суффиксные деревья, известны не настолько хорошо, насколько они того заслуживают. Будем надеяться, что этот каталог добавит им известности. Существует большое количество книг по элементарным структурам данных. На мой взгляд, лучшими из них являются следующие: ♦ [Sed98]— подробное введение в алгоритмы и структуры данных, отличающееся удачными рисунками, показывающими алгоритмы в действии. Имеются версии книги для языков С, C++ и Java; ♦ [Wei06] — хороший учебник, с упором на структуры данных, в большей степени, чем на алгоритмы. Имеются версии книги для языков Java, С, C++ и Ada; ♦ [GT05] — версия книги для Java с активным использованием авторской библиотеки структур данных JDSL (Java Data Structures Library); Книга [MS05] содержит всесторонний обзор современных исследований в области структур данных. Читатель, знакомый лишь с основными понятиями из этой области, будет удивлен объемом проводимых исследований. 1 2.1. Словарь Вход. Множество из п записей, каждая из которых идентифицируется одним или не- сколькими ключевыми полями. Задача. Создать и поддерживать структуру данных для эффективного поиска, вставки и удаления записи, связанной с ключом запроса q (рис. 12.1). Обсуждение. Абстрактный тип данных "словарь" является одной из наиболее важных структур в теории вычислительных систем. Для реализации словарей было предложено несколько десятков структур данных, включая хэш-таблицы, списки с пропусками и двоичные деревья поиска. Это означает, что выбрать наилучшую структуру данных не всегда легко. Используемая структура данных может оказать значительное влияние на
390 Часть II. Каталог алгоритмических задач производительность приложения. Однако на практике важнее избежать применения неподходящей структуры данных, чем найти самую лучшую. Вы должны тщательно изолировать словарную структуру данных от ее интерфейса. Используйте явные вызовы методов или процедур для инициализации, поиска и моди- фицирования структуры данных, а не внедряйте их в код структуры. Такой подход не только позволяет получить более аккуратную программу, но также облегчает проведение экспериментов с разными реализациями для оценки их производительно- сти. Не беспокойтесь по поводу неизбежных накладных расходов по вызову процедур. Если время работы вашего приложения является очень важным фактором, то выбор правильной реализации словаря позволит вам минимизировать его. Выбирая структуру данных для словаря, ответьте на несколько вопросов. ♦ Сколько элементов будет содержать структура данных? Известно ли их количе- ство заранее? Достаточно ли будет для решения вашей задачи простой структуры данных или же размер задачи настолько велик, что следует беспокоиться о нехватке памяти или производительности виртуальной памяти? ♦ Известна ли относительная частота операций вставки, удаления и поиска? Ста- тические структуры данных (например, упорядоченные массивы) достаточны для приложений, в которых структура данных не подвергается изменениям после ее первоначального создания. Полу динамические структуры данных, поддерживаю- щие только операцию вставки и не допускающие удаление данных, реализовать значительно проще, чем полностью динамические. ♦ Можно ли предполагать, что обращение к ключам будет единообразным и случай- ным? Во многих приложениях поисковые запросы имеют асимметричное распреде- ление обращений, т. е. некоторые элементы пользуются большей популярностью, чем другие. Кроме этого, запросы часто обладают свойством временной локально- сти. Иными словами, запросы периодически приходят группами (кластерами), а не через регулярные интервалы времени. В некоторых структурах данных (таких как
Глава 12. Структуры данных 391 косые деревья) можно воспользоваться асимметричным и ^пастеризированным пространством. ♦ Критична ли скорость отдельных операций или требуется .минимизировать толь- ко общий объем работы, выполняемый всей программой? Когда время реакции кри- тично, например, в программе управления аппаратом искусственного кровообраще- ния, время ожидания выполнения следующего шага не может быть слишком боль- шим. А в программе, выполняющей множество запросов к базе данных, например, для выяснения, кто из нарушителей закона является политическим деятелем, ско- рость поиска конкретного конгрессмена не является критичной, если удается вы- явить их всех с минимальными общими затратами времени. Представители современного поколения "объектно-ориентированных" программистов способны написать контейнерный класс не в большей степени, чем отремонтировать двигатель в своей машине. В этом нет ничего плохого — для большинства приложений вполне достаточно стандартных контейнеров. Тем не менее, иногда полезно знать, что находится "под капотом": ♦ неупорядоченные списки или массивы. Для небольших наборов данных самой про- стой в обслуживании структурой будет неупорядоченный массив. По сравнению с аккуратными и компактными массивами производительность связных структур может быть крайне неудовлетворительной. Но когда размер словаря превысит 50-100 элементов, линейное время поиска в списке или массиве сведет на нет все его преимущества. Подробности простых реализаций словарей см. в разделе 3.3. Особенно интересным и полезным вариантом является самоорганизующийся список (self-organizing list), в котором каждый вставленный или просмотренный элемент перемещается в начало списка. Таким образом, если в ближайшем будущем к дан- ному ключу осуществляется повторное обращение, то этот ключ будет находиться вблизи начала списка, и его поиск займет очень мало времени. Большинство при- ложений демонстрирует как неоднородную частоту обращений, так и пространст- венную локальность, поэтому среднее время успешного поиска в самоорганизую- щемся списке обычно намного лучше, чем в упорядоченном или неупорядоченном списке. Самоорганизующиеся структуры данных можно создавать из массивов точ- но так же. как из связных списков и деревьев; ♦ упорядоченные списки или массивы. Обслуживание упорядоченного списка обычно не стоит затрачиваемых усилий, если только вы не пытаетесь избежать создания дубликатов, т. к. эта структура данных не поддерживает двоичный поиск. Исполь- зование упорядоченного массива будет уместным только в тех случаях, когда не требуется выполнять большое количество вставок и/или удалений; ♦ хэш-таблицы. Для приложений, работающих с количеством элементов в диапазоне от умеренного до большого (примерно от 100 до 10 000 000), использование хэш- таблицы будет, скорее всего, правильным выбором. Хэш-функция устанавливает соответствие ключей (будь то строки, числа или что угодно другое) и целых чисел в диапазоне от 0 до /и-1. При создании хэш-таблицы сопровождается массив из т корзин, реализованных в виде неупорядоченного связного списка. Хэш-функция немедленно определяет корзину, содержащую данный ключ. Если используемая
392 Часть II. Каталог алгоритмических задач хэш-функция распределяет ключи достаточно равномерно и размер хэш-таблицы довольно велик, то каждая корзина будет содержать очень небольшое количество элементов, что делает линейные поиски приемлемыми. Вставка и удаление элемен- та из хэш-таблицы сводится к вставке и удалению из корзины/списка. Хэширование подробно обсуждалось в разделе 3.7 В большинстве приложений производительность хорошо отлаженной хэш-таблицы будет превосходить производительность упорядоченного массива. Но прежде чем приступить к реализации хэш-таблицы, ответьте на несколько вопросов. • Каким образом обрабатывать коллизии? Использование открытой адресации вместо корзин позволит получить небольшие таблицы с хорошей производи- тельностью, но с повышением коэффициента нагрузки (отношение заполненно- сти к емкости) производительность хэш-таблицы будет понижаться. • Каким должен быть размер таблицы? Если применяются корзины, то значение/» должно быть приблизительно равным максимальному количеству элементов, которое вы планируете поместить в таблицу. Если применяется от- крытая адресация, то значение т должно превышать ожидаемое максимальное количество элементов, по крайней мере, на 30%. Выбор в качестве т простого числа компенсирует возможные недостатки неудачной хэш-функции. • Какую хэш-функцию использовать? Для строк должна работать. например, такая ш-1 формула H(S, j) = У' g”'~('+l) х charts, ,„) mod т. где а— размер алфавита, 1=0 char(x)— функция, которая для каждого символа х возвращает его ASCII-код. Для реализации эффективного вычисления этой хэш-функции используйте пра- вило Горнера (или заранее вычисляйте значения а'), как показано в разделе 13.9. Эта хэш-функция имеет интересное свойство, заключающееся в том, что H(S,j + 1) = - a!" 'char[s^a + sl+m вследствие чего хэш-коды последовательных окон строки размером в т симво- лов можно вычислять за постоянное время, а не за время О(т) Независимо от того, какую хэш-функцию вы решите использовать, изучите стати- стические данные относительно распределения ключей по корзинам, чтобы выяс- нить, насколько она действительно является однородной. Вероятно, что самая пер- вая выбранная хэш-функция окажется не самой лучшей. Использование неудачной хэш-функции может заметно снизить производительность любого приложения. ♦ Двоичные деревья поиска Двоичные деревья являются элегантными структурами данных, которые поддерживают быстрое исполнение операций вставки, удаления и запросов. Эти структуры данных рассматриваются в разделе 3.4. Разница между ти- пами деревьев проявляется в том, выполняется ли их балансировка явным образом после операций вставки и удаления и в способе ее осуществления. В произвольных деревьях поиска узел просто вставляется в некий лист дерева и балансировка не осуществляется. Хотя такие деревья имеют хорошую производительность в случае произвольных вставок, большинство приложений не обеспечивает случайность вставок. В самом деле, несбалансированные деревья поиска, созданные вставками
Глава 12. Структуры данных 393 элементов в отсортированном порядке, никуда не годятся, т. к. их производитель- ность сравнима с производительностью связных списков. Структура сбалансированных деревьев поиска обновляется с помощью локальных операций ротации, которые перемещают удаленные узлы ближе к корню, таким образом поддерживая упорядоченную структуру дерева. В настоящее время такие сбалансированные деревья поиска, как AVL и 2-3, считаются устаревшими, и пред- почтение отдается красно-черным деревьям. Особенно интересной самоорганизую- щейся структурой данных являются косые деревья. В них ключи, к которым было осуществлено обращение, перемещаются в корень посредством операции ротации. Таким образом, часто используемые или недавно посещенные узлы находятся в верхней части дерева, что позволяет быстрее производить поиск. Итак, какой тип дерева лучше всего подойдет для вашего приложения? Наверное, тот. для которого у вас имеется наилучшая реализация. Не так важен тип исполь- зуемого сбалансированного дерева, как профессионализм программиста, реализо- вавшего его. ♦ В-деревья. Для наборов данных настолько больших, что они не помещаются в опе- ративную память (скажем, свыше I 000 000 элементов), наилучшим выбором будет какой-либо тип В-дерева. Когда структуру данных необходимо хранить вне опера- тивной памяти, время поиска увеличивается на несколько порядков. Подобное падение производительности в меньшем масштабе может наблюдаться в системах с современной архитектурой кэша, т. к. кэш работает намного быстрее, чем опера- тивная память Идея, лежащая в основе В-дерева, состоит в сворачивании нескольких уровней дво- ичного дерева поиска в один большой узел, чтобы можно было выполнить опера- цию. эквивалентную нескольким шагам поиска, прежде чем потребуется новое об- ращение к диску. С помощью В-деревьев можно получать доступ к огромному ко- личеству ключей, выполняя небольшое количество обращений к диску. Чтобы максимально эффективно использовать В-деревья, нужно понимать, как взаимодей- ствуют внешние запоминающие устройства с виртуальной памятью при учете таких аппаратных характеристик, как размер страницы и виртуальное/реальное адресное пространство Нечувствительные к кэшированию алгоритмы (cache-oblivious algo- rithms) позволяют уменьшить значение этих факторов. Даже в случае наборов данных небольшого размера результатом использования файлов подкачки может быть неожиданно низкая производительность, поэтому не- обходимо следить за интенсивностью обращений к жесткому диску, чтобы полу- чить дополнительную информацию для принятия решения, стоит ли использовать В-деревья. ♦ Списки с пропусками. Списки с пропусками представляют собой иерархическую структуру упорядоченных связных списков, в которых решение, копировать ли эле- мент в список более высокого уровня, принимается случайным образом. В структу- ре задействовано примерно Ign списков, размер каждого из которых приблизитель- но вдвое меньше размера списка, расположенного уровнем выше. Поиск начинается в самом коротком списке. Искомый ключ находится в интервале между двумя эле-
394 Часть II. Каталог алгоритмических задач ментами и потом ищется в списке большего размера. Каждый интервал поиска со- держит ожидаемое постоянное количество элементов в каждом списке, при этом общее ожидаемое время исполнения запроса составляет O(lgn). Основные достоин- ства списков с пропусками по сравнению со сбалансированными деревьями — лег- кость анализа и реализации. Реализации. Современные языки программирования содержат библиотеки подробных и эффективных реализаций контейнеров. В настоящее время библиотека STL (Standard Template Library) для языка C++ поставляется с большинством компиляторов. Кроме этого, эту библиотеку можно загрузить вместе с документацией с веб-сайта http:// www.sgi.com/tech/stl/. Более подробное руководство по использованию библиотеки STL и стандартной библиотеки C++ можно найти в книгах [Jos99], [МеуО 1 ] и [MDS01]. Библиотека LEDA (см. раздел 19.1.1) предоставляет полную коллекцию словарных структур данных, реализованных на языке C++, включая хэш-таблицы, совершенные хэш-таблицы, В-деревья, красно-черные деревья, деревья случайного поиска и списки с пропусками. В результате экспериментов, описанных в книге [MN99], было определе- но. что наилучшим вариантом для словарей являются хэш-таблицы, а списки с пропус- ками и (2,4)-деревья (частный случай В-деревьев) являются наиболее эффективными древоподобными структурами. Небольшая библиотека структур данных Java Collections (JC) входит в стандартный пакет утилит Java java.util (http://java.sun.com/javase/). Библиотека Data Structures Library in Java (JDSL) является более обширной. Ее можно загрузить для некоммерче- ского применения с веб-сайта http://www.jdsl.org/. Библиотека JDSL подробно описа- на в книгах [GTV05] и [GT05]. Примечания В книге [Knu97a] представлено наиболее подробное описание и анализ основных словар- ных структур данных. Но в ней отсутствуют некоторые современные структуры данных, такие как красно-черные и косые деревья. Чтение книг Дональда Кнута послужит хоро- шим введением в предмет для всех изучающих теорию вычислительных систем. В книге [MS05] дан всесторонний обзор современных исследований в области структур данных. Другие исследования представлены в книгах [МТ90Ь] и [GBY91], Хорошими учебниками по словарным структурам данных являются [Sed98], [Wei06] и [GT05]. Мы отсылаем читателя к этим работам, чтобы не приводить здесь ссылки на описания струк- тур данных, обсуждавшихся выше. Соревнование DIM ACS в 1996 г. было посвящено реализациям элементарных структур данных, включая словари (см. [GJM02]). Наборы данных и код можно загрузить с веб- сайта http://dimacs.rutgers.edu/Challenges. Для многих задач стоимость обмена данными между различными запоминающими устройствами (оперативной памятью, кэшем или диском) превышает стоимость собствен- но вычислений. В каждой операции по пересылке данных перемещается один блок разме- ром Ь, поэтому эффективные алгоритмы стремятся минимизировать количество переме- щений блоков. Исследование сложности фундаментальных алгоритмов и структур данных для этой модели внешней памяти представлено в журнале [VitOI]. Нечувствительные к кэшированию структуры данных дают гарантию производительности для такой модели, не требуя при этом явной информации о параметре размера блока Ь. Таким образом, хо- рошую производительность можно получить на любой машине, не прибегая к специфиче-
Глава 12. Структуры данных 395 ским для данной архитектуры настройкам. Превосходное исследование структур данных, нечувствительных к кэшированию, приведено в [ABF05]. Многие современные структуры данных, такие как косые деревья, изучались с примене- нием амортизированного анализа, при котором устанавливается верхняя граница для об- щего времени исполнения любой последовательности операций. При амортизированном анализе одна операция может быть очень дорогой, но только потому, что ее стоимость компенсируется низкими затратами на другие операции. Структура данных, имеющая амортизированную сложность является менее предпочтительной, чем структура с такой же сложностью в наихудшем случае (т. к. существует вероятность выполнения очень дорогой операции), но она все же лучше структуры с такой сложностью в среднем случае (т. к. амортизированная граница достигнет этого значения при любом входе). Родственные задачи. Сортировка {см. раздел 14.1), поиск {см. раздел 14.2). 12.2. Очереди с приоритетами Вход. Набор записей, ключи которых полностью упорядочены по номерам или каким- либо другим способом. Задача. Создать и поддерживать структуру данных для обеспечения быстрого доступа к наименьшему или наибольшему ключу (рис. 12.2). October 30 December 7 July 4 January 1 February 2 December 25 ВХОД ВЫХОД Рис. 12.2. Очередь с приоритетами Обсуждение. Очереди с приоритетами являются полезными структурами данных при моделировании поведения систем, особенно при сопровождении набора запланирован- ных событий, упорядоченных по времени. Они получили такое название потому, что элементы из них можно извлекать не в зависимости от времени вставки (как в стеке или обычной очереди) и не в результате совпадения ключа (как в словаре), а по их приоритету. Если после первоначального запроса приложение не будет вставлять новые элементы, то в явно выраженной очереди с приоритетами нет необходимости. Просто отсорти- руйте записи по приоритетам и выполняйте обработку сверху вниз, поддерживая при этом указатель на последнюю извлеченную запись. Такая ситуация возникает в алго- ритме Крускала при построении минимального остовного дерева или при выполнении полностью распланированного сценария развития событий.
396 Часть II. Каталог алгоритмических задач Но если операции вставки и удаления чередуются с запросами, понадобится настоящая очередь с приоритетами. Чтобы решить, какой тип очереди следует применить, ответь- те на несколько вопросов. ♦ Какие еще операции вам потребуются? Будет ли выполняться поиск произвольных ключей или только наименьшего ключа? Будут ли удаляться произвольные элемен- ты данных или просто будут повторно удаляться верхний или нижний элемент? ♦ Известен ли заранее максимальный размер структуры данных? Здесь вопрос за- ключается в том, можно ли будет выделить место для структуры данных заранее. ♦ Можно ли менять приоритет элементов, уже поставленных в очередь? Возмож- ность изменять приоритет элементов означает, что, кроме извлечения наибольшего элемента, элементы можно извлекать из очереди по ключу. Перечислим базовые реализации очереди с приоритетами: ♦ упорядоченный .массив ши список. Упорядоченный массив очень эффективен как для идентификации наименьшего элемента, так и для его удаления путем уменьше- ния значения максимального индекса. Но поддержание упорядоченности элементов замедляет вставку новых элементов. Упорядоченные массивы подходят для реали- зации очереди с приоритетами только тогда, когда в нее будет осуществляться очень небольшое количество вставок. Базовые реализации очередей с приоритетами рассматриваются в разделе 3.5: ♦ двоичные пирамиды. Эта простая, элегантная структура данных поддерживает как операцию вставки, так и операцию извлечения наименьшего элемента за время O(lg«). Пирамиды позволяют поддерживать явную структуру двоичного дерева в массиве, чтобы значение ключа корня любого поддерева было меньшим, чем любо- го из его потомков. Таким образом, наименьший ключ всегда находится вверху пи- рамиды. Новые элементы можно добавлять, вставляя элемент в свободный лист де- рева и перемещая его вверх до тех пор, пока он не займет правильное место при ус- ловии частичной упорядоченности. Реализация двоичной пирамиды на языке С рассматривается в разделе 4 3.2, а операция извлечения из нее наименьшего элемен- та — в разделе 4.3.3. Двоичные пирамиды являются удачным выбором в том случае, когда известна верхняя граница количества элементов в очереди с приоритетами, т. к. размер мас- сива нужно задавать при его создании. Но даже это ограничение можно обойти с помощью динамических массивов (см.раздел 3.1.1): ♦ очередь с приоритетами, имеющая ограничение по высоте Эта структура данных на основе массива позволяет выполнять операции вставки и поиска наименьшего элемента за постоянное время при ограниченном диапазоне возможных значений ключей. Допустим, мы знаем, что все значения ключей будут целыми числами в диапазоне от 1 до п. Тогда можно создать массив из п связных списков, где /-Й спи- сок играет роль корзины, содержащей все элементы с ключом /. На наименьший не- пустой список будем поддерживать указатель Юр на наименьший непустой список. Чтобы вставить в очередь с приоритетами элемент с ключом к, добавляем его к к-й корзине и присваиваем этому указателю значение Юр = гшп(ГорЛ). Чтобы извлечь наименьший элемент, определяем первый элемент в корзине top, удаляем его и. если корзина стала пустой, перемещаем указатель Юр вниз.
Глава 12. Структуры данных 397 Очереди с приоритетами, имеющие ограничение по высоте, очень полезны для хра- нения вершин графа, упорядоченных по их степеням, при том, что такое упорядочи- вание является одной из основных операций в алгоритмах на графах. Тем не менее, при этом они применяются не так широко, как того заслуживают. Обычно этот тип очереди подходит для случаев небольших, дискретных диапазонов ключей; ♦ двоичные деревья поиска. Из двоичных деревьев поиска получаются эффективные очереди с приоритетами, т. к. самый левый лист всегда содержит наименьший эле- мент. а самый правый— наибольший. Чтобы найти наименьший (наибольший) элемент, просто выполняется проход по указателям влево (вправо) вниз до тех пор. пока указатель не станет нулевым. Двоичные деревья поиска оказываются наиболее подходящими, когда требуется осуществлять другие словарные операции или в слу- чае неограниченного диапазона ключей, когда максимальный размер очереди с приоритетами неизвестен заранее: ♦ Фибоначчиевы и парные (pairing) пирамиды. Эти сложные очереди с приоритетами предназначены для ускорения выполнения операции уменьшения ключа, которая понижает приоритет поставленного в очередь элемента. Такая ситуация возникает, например, в вычислениях кратчайшего пути, при обнаружении более короткого, чем найденный ранее, маршрута к вершине v. Если их реализовать и использовать должным образом, то этот тип очереди с при- оритетами может улучшить производительность при больших объемах вычислений. Реализации. Современные языки программирования содержат библиотеки подробных и эффективных реализаций очередей с приоритетами. Методы push, top и pop шаблона priority queue библиотеки STL языка C++ воспроизводят операции с пирамидами in- sert. findmax и deletemax. Библиотеку STL можно загрузить вместе с документацией с веб-сайта http://www.sgi.com/tech/stl/. Более подробное руководство по использова- нию библиотеки STL можно найти в книгах [MeyOl] и [MDS01]. Библиотека LEDA (см. раздел 19.1.1) содержит полную коллекцию очередей с приори- тетами на языке C++, включая Фибоначчиевы пирамиды, парные пирамиды, деревья Емде Боаса и очереди с приоритетами, имеющие ограничение по высоте. В результате экспериментов, описанных в книге [MN99], было установлено, что простые двоичные пирамиды являются довольно конкурентоспособными в большинстве приложений, при этом в прямом сравнении парные пирамиды показывают лучшую производительность, чем Фибоначчиевы пирамиды. Класс PriorityQueue библиотеки Java Collections (JC) входит в стандартный пакет ути- лит Java java.util (http://java.sun.com/javase/). Библиотека Data Structures Library in Java (JDSL) предоставляет альтернативную реализацию, которую для некоммерческого применения можно за! рузить с веб-сайта http://www.jdsl.org/. Более подробное руко- водство по библиотеке JDSL можно найти в книгах [GTV05] и [GT05], В статье [SanOO] изложено описание экспериментов, демонстрирующих, что произво- дительность разработанной автором последовательной пирамиды (sequence heap), ос- нованной на k-м слиянии, приблизительно вдвое лучше, чем хорошо реализованной двоичной пирамиды. Реализацию этой пирамиды можно загрузить с веб-страницы http://www.mpi-inf.mpg.de/~sanders/programs/spq/.
398 Часть II. Каталог алгоритмических задач Примечания В книге [MS05] приводится всесторонний обзор современного состояния дел в том, что касается очередей с приоритетами. Результаты экспериментальных сравнений структур данных, реализующих очереди с приоритетами, представлены в таких работах, как [CGS99], [GBY91], [Jon86], [LL96J и [SanOOJ. Двусторонние очереди с приоритетами (double-ended priority queue) позволяют расширить набор операций с пирамидами, включив в него одновременно операции find-min и find- max. Обзор четырех разных реализаций двусторонних очередей с приоритетами см. в [Sah05]. Очереди с приоритетами, имеющие ограничение по высоте, являются подходящими структурами данных для использования на практике, но они не гарантируют высокой производительности в наихудшем случае, когда разрешены произвольные вставки и уда- ления. Однако очереди с приоритетами Емде Боаса (см. [vEBKZ77]) позволяют выполнять операции вставки, удаления, поиска, определения наибольшего и наименьшего элементов за время O(]glgn), где каждый ключ имеет значение от I до п. Фибоначчиевы пирамиды (см. [FT87]) поддерживают выполнение операций вставки и уменьшения ключа за амортизированное постоянное время, а выполнение операций из- влечения наименьшего элемента и удаления — за время O(lg/j). Постоянное время испол- нения операции уменьшения ключа позволяет получить более быструю реализацию алго- ритмов поиска кратчайшего пути, паросочетаний во взвешенных двудольных графах и минимальных остовных деревьев. Фибоначчиевы пирамиды на практике трудно реализо- вать, и время исполнения их операций содержит большие постоянные коэффициенты. Парные пирамиды поддерживают такие же пределы, но при меньших накладных расхо- дах. Эксперименты с парными пирамидами описаны в журнале [SV87]. Пирамиды поддерживают частичное упорядочивание, которое можно создать за линейное количество операций сравнения. Обычные алгоритмы слияния с линейным временем ис- полнения для создания пирамид описаны в [Flo64], В наихудшем случае достаточно 1,625л сравнений (см. [GM86]), а вообще необходимо от 1,5 л до O(lgn) сравнений. Родственные задачи. Словари (см. раздел 12.1), сортировка (см. раздел 14.1), поиск кратчайшего пути (см. раздел 15.4). 12.3. Суффиксные деревья и массивы Вход. Строка S. Задача. Создать структуру данных, позволяющую быстро определять местоположение произвольной строки запроса q в строке S (рис. 12.3). Обсуждение. Суффиксные деревья и массивы являются исключительно полезными структурами данных для элегантного и эффективного решения задач по обработке строк. Правильное применение суффиксных деревьев часто позволяет уменьшить вре- мя исполнения алгоритмов для обработки строк с О(«Э до линейного. Суффиксные деревья упоминались в истории из жизни, описанной в разделе 3.9. Простейший экземпляр суффиксного дерева представляет собой простое нагруженное дерево (trie) из п суффиксов строки S длиной в п символов. Нагруженным деревом на- зывается древовидная структура, в которой каждое ребро представляет один символ, а корень— нулевую строку. Таким образом, каждый путь из корня представляет строку,
Глава 12. Структуры данных 399 описываемую символами, которые маркируют проходимые ребра. Любой конечный набор слов определяет нагруженное дерево, а дерево для двух слов с общим префик- сом разветвляется на первом несовпадающем символе. Каждый лист дерева обозначает конец строки. На рис. 12.4 показан пример простого нагруженного дерева. Рис. 12.4. Нагруженное дерево для строк the, their, there, was и when Нагруженные деревья полезны для проверки на вхождение данной строки запроса q в набор строк. Обход нагруженного дерева выполняется с корня вдоль ветвей, опреде- ляемых последующими символами строки q. Если какая-либо ветвь отсутствует в на- груженном дереве, то тогда строка q не может быть элементом набора строк, опреде- ляемых данным нагруженным деревом. В противном случае строка q находится за |<?| сравнений символов, независимо от количества строк, содержащихся в нагруженном дереве. Нагруженные деревья очень легко создавать (последовательно вставляя новые строки), и в них быстро выполняется поиск, хотя они могут оказаться дорогими в смысле расхода памяти. Суффиксное дерево — это просто нагруженное дерево всех суффиксов строки S. Суф- фиксное дерево позволяет проверить, является ли строка запроса q подстрокой строки
400 Часть II. Каталог алгоритмических задач S, т. к. любая подстрока строки Л’ представляет собой префикс какого-либо суффикса. Время поиска — линейное по отношению к длине строки запроса q. Но проблема здесь заключается в том, что создание полного суффиксного дерева таким образом может занять время О(п~) и, что еще хуже, такой же объем памяти, т. к. сред- няя длина п суффиксов равна п/2. Но при рациональном подходе для представления полного суффиксного дерева затраты памяти будут линейными. Обратите внимание на то, что большинство узлов нагруженного суффиксного дерева находятся на простых путях между узлами на ветвях дерева. Каждый из этих простых путей соответствует подстроке первоначальной строки. Сохранив первоначальную строку в массиве и свер- нув каждый такой путь в одно ребро, всю информацию о полном суффиксном дереве можно сохранить в памяти объемом О(п). Маркировка каждого ребра состоит из ин- дексов начала и конца массива, представляющего подстроку. Такое свернутое суф- фиксное дерево показано на рис. 12.3. Для создания свернутого дерева существуют алгоритмы, использующие дополнитель- ные указатели, ускоряющие процесс построения. Время исполнения этих алгоритмов равно б)(п). Эти дополнительные указатели можно также использовать для ускорения многих приложений, основанных на суффиксных деревьях. Но для чего можно использовать суффиксные деревья? Далее приводится краткое опи- сание вариантов применения суффиксных деревьев. Подробности см. в книгах [Gus97] и [CR03], ♦ Поиск всех вхождений подстроки q в строке S. Так же, как и в случае с нагружен- ным деревом, мы можем выполнять обход, начиная с корня дерева по направлению к узлу пч, связанному с подстрокой q. Позиции всех вхождений подстроки q в стро- ку S представлены потомками узла пч, которых можно определить посредством по- иска в глубину, начиная с узла пч. В свернутых суффиксных деревьях найти к вхож- дений подстроки q в строку S можно за время + к). ♦ Поиск самой длинной подстроки, общей для набора строк. Создается одно сверну- тое суффиксное дерево, содержащее все суффиксы всех строк, в котором каждый лист дерева помечен его исходной строкой. В процессе обхода в глубину этого де- рева мы можем пометить каждый его узел информацией о длине его префикса и ко- личестве разных строк, являющихся его потомками. По этой информации наилуч- ший узел можно найти за линейное время. ♦ Поиск самого длинного палиндрома в строке S. Палиндром— это строка, которая читается одинаково в обоих направлениях, например мадам. Чтобы найти самый длинный палиндром в строке S. создаем суффиксное дерево, содержащее все суф- фиксы строки S и строки, обратной 5. Каждый лист этого дерева будет идентифици- роваться его начальной позицией. Палиндром определяется любым узлом этого дерева, у которого из одной позиции выходят потомки, одинаковые в обоих направ- лениях. Так как алгоритмы с линейным временем исполнения для создания суффиксных де- ревьев являются нетривиальными, вместо самостоятельной разработки таких алгорит- мов рекомендуется использовать существующие реализации. В качестве альтернативы можно использовать суффиксные массивы.
Глава 12. Структуры данных 401 Суффиксные массивы позволяют делать практически все то, что и суффиксные де- ревья, но при этом используют приблизительно в четыре раза меньший объем памяти. Кроме этого, их легче реализовать. В принципе, суффиксный массив представляет со- бой просто массив, содержащий все и суффиксов строки .S’ в отсортированном порядке. Таким образом, двоичного поиска строки q в этом массиве оказывается достаточно для локализации префикса суффикса, совпадающего с подстрокой q, что гарантирует эф- фективный поиск подстроки за G>(lg/?) сравнений строк. После добавления индекса, указывающего общую длину префикса всех граничных суффиксов, для любого запроса потребуется только Ign + |<у| сравнений символов, т. к. появляется возможность опреде- лять. какой символ нужно проверить следующим при двоичном поиске. Например, ес- ли нижней границей диапазона поиска является слово cowabunga. а верхней — cowslip, то все промежуточные ключи будут совпадать по первым трем символам, поэтому сравнивать с подстрокой q нужно только четвертый символ любого промежуточного ключа. На практике скорость поиска в суффиксных массивах обычно такая же. как в суффиксных деревьях или даже выше. Кроме этого, суффиксные массивы требуют меньше памяти, чем суффиксные деревья. Каждый суффикс представляется в них уникальной начальной позицией (от I до и) и считывается по мере надобности с использованием одной эталонной копии входной строки. Но для эффективного создания суффиксных массивов нужно соблюдать определенную осторожность, т. к. сортируемые строки содержат О(« ) символов. Одним из реше- ний— сначала создать суффиксное дерево, после чего выполнить симметричный об- ход этого дерева, чтобы прочитать строки в отсортированном порядке! Но последние достижения в этой области позволяют создавать эффективные по времени и памяти алгоритмы для построения суффиксных массивов напрямую. Реализации. В настоящее время существует большое количество реализаций суф- фиксных массивов. По сути, все последние алгоритмы для построения суффиксных массивов за линейное время были реализованы, а их сравнительная производитель- ность измерена (см. [PST07]). Превосходная реализация суффиксных массивов на язы- ке С описывается в книге [SS07]. Загрузить эту реализацию можно с веб-сайта http://bibiserv.techfak.uni-bielefeld.de/bpr/. А на веб-сайте Pizza&Chili Corputs (http://pizzachili.di.unipi.it) представлены восемь разных реализаций на C/C++ для сжатых текстовых индексов. Эти структуры данных позволяют значительно минимизировать расход памяти за счет чрезвычайно эффек- тивного сжатия строки входа. При этом они обеспечивают отличное время исполнения запросов. Вполне доступны и реализации суффиксных деревьев. Проект с открытым кодом BioJava (http://www.biojava.org), предоставляющий инфраструктуру Java для обра- ботки биологических данных, содержит класс SuffixTree. Реализацию алгоритма Укконена на языке С (называемую Libstree) можно загрузить с веб-сайта http:// www.icir.org/christian/libstree/. А с веб-сайта http://marknelson.us/1996/08/01/suffix-trees можно загрузить код Нель- сона на языке C++ (см. [Nel96]).
402 Часть II. Каталог алгоритмических задач Программы на языке С коллекции strmat реализуют алгоритмы для точного сравнения шаблонов, включая реализацию суффиксных деревьев (см. [Gus97])_ Эту коллекцию можно загрузить с веб-сайта http://www.cs.ucdavis.edu/~gusfield/strmat.html. Примечания Нагруженные деревья (trie) были впервые предложены Е. Фредкином (Е. Fredkin) в жур- нале [Fre62]. В книге [GBY91] представлен обзор основных структур данных нагружен- ных деревьев, сопровождаемый многочисленными ссылками. Эффективные алгоритмы для построения суффиксных деревьев были созданы Вейнером (Weiner), МакКрейтом (McCreight) и Укконеном (Ukkonen) (см. [Wei73], [МсС76] и [Ukk92] соответственно). Хорошие описания этих алгоритмов даны в журнале [CR03] и в книге [Gus97]. Суффиксные массивы были изобретены Манбером (Manber) и Майерсом (Myers) (см. [ММ93]), хотя аналогичная идея Гоннета (Gonnet) и Баеза-Йейтса (Baeza-Yates) была вы- сказана в [GBY91]. Алгоритмы построения суффиксных массивов с линейным временем исполнения были разработаны тремя независимыми командами в 2003 г. (см. [KSPP03], [КА03] и [KSB05]) и с тех пор эта область успешно развиваегся. Обзор последних разра- боток представлен в [PST07]. Результатом последних работ является создание сжатых полнотекстовых индексов, кото- рые позволяют реализовать практически все возможности суффиксных деревьев и масси- вов в структуре данных, размер которой пропорционален размеру сжатой текстовой строки. Обзор этих важных структур данных см. в [MN07]. Возможности суффиксных деревьев можно расширить, используя структуру данных для вычисления наименьшего общего предшественника любой пары узлов (х, у) за постоянное время после предварительной обработки дерева за линейное время. Первоначально эта структура данных была предложена Харелом (Harel) и Тарьяном (Tarjan) (см. [НТ84]), по- сле чего была упрощена сначала Шибером (Schieber) и Вишкиным (Vishkin) (см. [SV88]), а потом Бендером (Bender) и Фараком (Farach) (см. [BF00]). Ее описание можно найти, например, в [Gus97]. Наименьший общий предшественник двух узлов суффиксного дере- ва или нагруженного дерева определяет узел, представляющий самый длинный общий префикс двух ассоциированных строк. Удивительно, что осуществление таких запросов происходит за постоянное время, и этот факт делает их пригодными в качестве компонен- тов многих других алгоритмов. Родственные задачи. Поиск подстрок (см. раздел 18.3), сжатие текста (см. раз- дел 18.5), поиск самой длинной общей подстроки (см. раздел 18.8). 12.4. Графы Вход. Граф G. Задача. Представить граф G посредством гибкой и эффективной структуры данных (рис. 12.5). Обсуждение. Двумя основными структурами данных для представления графов явля- ются матрицы смежности и списки смежности. Подробное описание этих структур данных вместе с реализацией списков смежности на языке С дано в разделе 5.2. Для большинства задач наиболее подходящей структурой данных являются списки смеж- ности.
Глава 72. Структуры данных 403 Выбирая структуру данных, ответьте на несколько вопросов. ♦ Каким будет размер графа? Сколько вершин и ребер будет иметь граф как в ти- пичном, так и в наихудшем случае? Для графов с 1 000 вершин требуются матрицы смежности из 1 000 000 элементов, что вряд ли приемлемо. Поэтому матрицы смежности подходят только для представления небольших или очень плотных гра- фов. ♦ Какой будет плотность графа? Если граф очень плотный, т. е. ребра определены для большей части возможных пар вершин, то, пожалуй, у вас нет особой необхо- димости использовать списки смежности. В любом случае объем требуемой памяти будет О(»"), т. к. благодаря отсутствию указателей матрицы смежности для полных графов будут иметь меньший размер. ♦ Какой алгоритм будет применяться? В некоторых алгоритмах лучше использовать матрицы смежности (например, в алгоритме поиска кратчайшего пути между всеми парами вершин), а в других— списки смежности (например, в большинстве алго- ритмов, основанных на обходе в глубину). Матрицы смежности имеют явное пре- имущество в алгоритмах, в которых многократно выясняется наличие конкретного ребра в графе. Впрочем, большинство алгоритмов на графах можно разработать так, что подобные запросы в них исключены. ♦ Будет ли граф модифицироваться при выполнении приложения? Если после созда- ния графа в него не будут вставляться и/или удаляться ребра, то можно использо- вать эффективные реализации статических графов. В действительности, гораздо чаще, чем топология графа, модифицируются атрибуты его ребер или вершин — размер, вес, метка или цвет. Атрибуты лучше всего обрабатывать с помощью до- полнительных полей в записях ребер или вершин в списках смежности. Разработать универсальный программный тип графа весьма непросто. Поэтому я реко- мендую проверить наличие уже существующей реализации (в особенности в библиоте- ке LEDA), прежде чем создавать свою собственную. Обратите внимание, что преобра- зование между матрицей смежности и списком смежности занимает линейное время по отношению к размеру большей структуры данных. Маловероятно, что это преобразо- вание окажется узким местом в каком-либо приложении, поэтому не исключено ис- пользование обеих структур данных, если имеется достаточно памяти. Обычно в этом нет необходимости, но такое решение может оказаться самым простым, если не совсем понятно, какую структуру выбрать.
404 Часть II. Каталог алгоритмических задач Планарными называются такие графы, которые можно начертить на плоскости без пе- ресечения ребер. Возникающие во многих приложениях графы являются планарными по определению, например карты стран. Другие графы, например деревья, планарны сами по себе. Планарные графы всегда разреженные, т. к. любой планарный граф с п вершинами может иметь, самое большее, Зн—6 ребер. Поэтому для представления этих графов следует использовать списки смежности. Если планарное представление графа (или укладка) играет важную роль в решаемой задаче, то планарный граф лучше всего представлять геометрическим способом. Информацию об алгоритмах создания укладок графов см. в разделе 15.12. В разделе 17.15 рассматриваются алгоритмы построения графов, неявно возникающих при размещении геометрических объектов, таких как линии и многоугольники. Гиперграфами называются обобщенные графы, в которых каждое ребро может связы- вать любое количество вершин. Допустим, что мы хотим представить в виде графа членство конгрессменов в разных комитетах. Такой граф будет гиперграфом. каждая вершина которого представляет отдельного конгрессмена, а каждое гиперребро, со- единяющее несколько вершин, представляет один комитет. Такие произвольные набо- ры подмножеств некоторого множества естественно рассматривать в виде гипергра- фов. При работе с гиперграфами используются такие базовые структуры данных, как: ♦ матрицы инцидентности, которые аналогичны матрицам смежности. Для них тре- буется объем памяти размером п*т. где т— количество гиперребер. Каждая стро- ка такой матрицы соответствует вершине, а каждый столбец— ребру. Значение ячейки M[i. j] будет ненулевым только в том случае, если вершина i входит в реб- ро/ Каждый столбец матрицы инцидентности стандартных графов содержит две ненулевые ячейки. Количество ненулевых позиций в каждой строке зависит от сте- пени каждой вершины; ♦ двудольные структуры инцидентности, которые аналогичны спискам смежности и. следовательно, более подходят для использования с разреженными гиперграфа- ми. В структуре инцидентности имеется вершина, связанная с каждым ребром и вершиной гиперграфа. Для представления такой структуры инцидентности обычно используются списки смежности. Естественным способом визуального представле- ния гиперграфа является ассоциированный двудольный граф. Эффективно представить очень большой граф довольно трудно. Тем не менее графы, содержащие миллионы ребер и вершин, использовались для решения интересных за- дач. В первую очередь следует сделать структуру данных как можно более экономной, упаковав матрицу смежности в битовый вектор (см. раздел 12.5) или удалив ненужные указатели из представления списка смежности. Например, в статическом графе (не поддерживающем вставку или удаление ребер) каждый список ребер можно заменить упакованным массивом идентификаторов вершин, что позволит избавиться от указате- лей и сэкономить половину памяти. В случае чрезвычайно больших графов может понадобиться иерархическое представ- ление, в котором группы вершин собираются в подграфы, каждый из которых сжима- ется в одну вершину. Такие иерархические декомпозиции можно создавать двумя спо-
Глава 12. Структуры данных 405 собами. В первом граф разбивается на компоненты естественным или специфическим для приложения способом. Например, граф дорог и населенных пунктов имеет естест- венную декомпозицию — разделение карты на населенные пункты, районы и области. При втором подходе выполняется алгоритм разбиения графа, рассматриваемый в раз- деле 16.6. Для NP-полной задачи естественная декомпозиция, скорее всего, даст более приемлемый результат, чем наивный эвристический алгоритм. Если же граф действи- тельно очень большой, вы не сможете позволить себе его алгоритмическое разбиение. Поэтому, прежде чем предпринимать такие решительные действия, убедитесь, что стандартные структуры данных неприменимы к вашей задаче. Реализации. Самую лучшую на сегодня реализацию структур данных для представле- ния графов на языке C++ содержит библиотека LEDA (см. раздел 19.1.1). Хотя она и является коммерческим продуктом, вам следует изучить представленные в ней методы для работы с графами, чтобы увидеть, как правильный уровень абстракции облегчает задачу реализации алгоритмов. Более доступной является библиотека Boost Graph Library (см. [SLL02]), которую мож- но загрузить с веб-сайта http://vwvw.boost.org/libs/grapli/doc. Библиотека включает реализации списков смежности, матриц смежности и списков ребер, а также неплохую коллекцию базовых алгоритмов для решения задач на графах. Ее интерфейс и компо- ненты являются обобщенными в том же смысле, что и у библиотеки STL на языке C++. Библиотека графов JUNG (http://jung.sourceforge.net) особенно популярна среди раз- работчиков социальных сетей. Библиотека Data Structures Library in Java (JDSL) пре- доставляет реализацию разнообразных структур данных с неплохой библиотекой алго- ритмов, которую можно загрузить с веб-сайта http://www.jdsl.org/ для некоммерческо- го применения. Более подробное описание библиотеки JDSL см. в книгах [GTV05] и [GT05]. Библиотека JGraphT (http://jgrapht.sourceforge.net) является более поздней разработкой с аналогичной функциональностью. Программа Stanford GraphBase (см. раздел 19.1.8) предоставляет простую, но гибкую структуру данных для реализации графов, разработанную с использованием методоло- гии CWEB. Поучительно рассмотреть, что Кнут реализовал (и чего не реализовал) в этой базовой структуре данных, хотя в качестве основы для разработок я бы реко- мендовал другие реализации. Среди всех реализаций типов графов на языке С я (субъективно) предпочитаю библио- теку из моей книги "Programming Challenges" (см. [SR03]). Подробности см. в разде- ле 19.1.10. Простые структуры данных для представления графов на языке программи- рования пакета Mathematica, а также библиотека алгоритмов и процедур вывода име- ются в библиотеке Combinatorica (см. [PS03]). Подробности см. в разделе 19.1.9. Примечания Преимущества списков смежности для работы с графами становятся очевидными при ис- пользовании алгоритмов Хопкрофта (Hopcroft) и Тарьяна (Tarjan) с линейным временем исполнения (см. [НТ73Ь], [Таг72]). Базовые структуры данных списков и матриц смежно- сти рассматриваются практически во всех книгах по алгоритмам или структурам данных, включая книги [CLRSOl], [AHU83] и [Таг83]. Гиперграфы обсуждаются в книге [Вег89].
406 Часть II. Каталог алгоритмических задач Эффективность применения статических графов была показана Неером (Naher) и Злотов- ским (Zlotowski) в [NZ02], которым удалось повысить скорость исполнения некоторых алгоритмов из библиотеки LEDA в четыре раза, просто используя более компактную структуру для представления графов. Интересным вопросом является минимизация количества битов, необходимых для пред- ставления произвольных графов, состоящих из п вершин, особенно если требуется эффек- тивное исполнение определенных операций. Эта тема обсуждается в книге |vL90b). Динамические алгоритмы решения задач на графах поддерживают быстрый доступ к ин- вариантам (таким как минимальное остовное дерево) при вставке или удалении ребер. Общим подходом к созданию динамических алгоритмов является разрежение (sparsifi- cation) (см. [EGIN92]). Результаты экспериментальных исследований практически всех динамических алгоритмов решения задач на графах изложены в публикациях [ACI92] и [Zar02], Иерархически определяемые графы часто возникают в задачах разработки микросхем со сверхвысоким уровнем интеграции, т. к. их разработчики активно используют библиотеки схемных элементов (см. (Len90J). Алгоритмы, специфичные для иерархически определяе- мых графов, включают в себя проверку на планарность (см. [Len89]), проверку на связ- ность (см. [LW88]) и построение минимальных остовных деревьев (см. [Len87a]). Родственные задачи. Структуры данных множеств (см. раздел 12.5), разделение графа (см. раздел 16.6). 12.5. Множества Вход. Универсальное множество элементов U= {mi, .... и,,}, по которому определена коллекция подмножеств S = {5,. Задача. Представить каждое подмножество таким образом, чтобы можно было эффек- тивно проверять вхождение элемента и, в подмножество 5,, вычислять объединение или пересечение подмножеств S, и S,, вставлять или удалять члены коллекции S (рис. 12.6). ВХОД ВЫХОД Рис. 12.6. Множества X, Y, Z
Глава 12. Структуры данных 407 Обсуждение. В математике множеством называется неупорядоченная коллекция объ- ектов из фиксированного универсального множества. Но при практической реализации удобно представлять множества в едином канонически упорядоченном виде, обычно отсортированном, чтобы ускорить или упростить разные операции. Упорядоченное множество превращает задачу поиска объединения или пересечения двух подмножеств в операцию с линейным временем исполнения — мы просто перебираем элементы сле- ва направо и выясняем, какие отсутствуют. Кроме этого, в упорядоченном множестве поиск элемента занимает сублинейное время. Однако распечатка элементов канониче- ски упорядоченного множества парадоксальным образом свидетельствует о том, что порядок в действительности не имеет значения. Множества отличаются от словарей или строк. Коллекцию объектов, составленную не из универсального множества постоянного размера, лучше всего рассматривать как словарь (см. раздел 12.1). Строки являются структурами, в которых порядок элементов имеет значение, т. е., строка {А, В, С} не равна строке {В, С, А]. Структуры данных и алгоритмы для работы со строками рассматриваются в разделе 12.3 и главе 18. В мультимножествах допустимо вхождение одного и того же элемента больше одно- го раза. Структуры данных для множеств обычно можно расширить для мультимно- жеств, поддерживая поле счетчика или связный список одинаковых вхождений каждо- го элемента мультимножества. Если каждое подмножество содержит ровно два элемента, то эти подмножества можно рассматривать как ребра графа, вершины которого представляют универсальное мно- жество. Система подмножеств без ограничений на количество ее элементов называется гиперграфом. Вы должны задать себе вопрос, существует ли для вашей задачи анало- гия из теории графов, такая как поиск связных компонент или кратчайшего пути в графе. Перечислим основные структуры данных для представления произвольных подмно- жеств: ♦ битовые векторы. Посредством битового вектора или массива из п битов можно представить любое подмножество .S’ из универсального множества U. содержащее и элементов. Если i е S. то значение Его бита будет 1, а в противном случае — 0. Так как для представления каждого элемента требуется только один бит, то битовые векторы позволяют экономить память даже при очень больших значениях |Е7|. Для вставки или удаления элемента значение соответствующего бита просто меняется на противоположное. Пересечения и объединения вычисляются выполнением логи- ческих операций И и ИЛИ. Единственным недостатком битового вектора является его неудовлетворительная производительность на разреженных подмножествах. Например, чтобы явно идентифицировать все члены разреженного (даже пустого) подмножества S, требуется время О(п); ♦ контейнеры ши словари. Подмножество можно представить посредством связного списка, массива или словаря, содержащего элементы подмножества. Для такой структуры данных не требуется идея постоянного универсального множества. Для разреженных подмножеств словари могут обеспечить большую экономию памяти и времени, чем битовые векторы; к тому же, с ними легче работать. Для эффективно- сти выполнения операций объединения и пересечения следует поддерживать эле-
408 Часть II. Каталог алгоритмических задач менты каждого подмножества в отсортированном порядке, что позволяет иденти- фицировать все дубликаты проходом по обоим подмножествам за линейное время; ♦ фильтры Блума. При отсутствии фиксированного универсального множества бито- вый вектор можно эмулировать, хэшируя каждый элемент подмножества в целое число в диапазоне от 0 до п и устанавливая соответствующий бит. Таким образом, если е 6 5, то биту Н(е) будет присвоено значение I. Однако при использовании этой схемы могут возникнуть коллизии, поскольку разные ключи могут быть хэши- рованы в одно и то же целое число. Чтобы снизить уровень ошибок, в фильтрах Блума используется несколько (скажем, к) разных хэш-функций Н\, ..., Я*, и после вставки ключа е всем к битам Н,(е) при- сваивается значение 1. Теперь элемент е является членом 5 только в том случае, ко- гда значения всех битов к равны I. Таким образом, увеличивая количество хэш- функций и размер таблицы, можно снизить вероятность ошибки. Используя пра- вильно подобранные константы, можно представить каждый элемент подмножества посредством фиксированного количества битов, независимого от размера универ- сального множества. Эта структура, основанная на хэшировании, намного экономичнее словарей в смыс- ле расхода памяти. Она годится для приложений, работающих на статических под- множествах и допускающих небольшую вероятность ошибки. Таких приложений не так уж мало. Например, при проверке правописания не произойдет большой траге- дии. если программа иногда пропустит ошибочно написанное слово. Многие приложения работают с попарно непересекающимися коллекциями подмно- жеств. в которых каждый отдельный элемент является членом ровно одного подмно- жества. В качестве примера рассмотрим задачу представления компонент связности графа или принадлежности политика к партии. Здесь каждая вершина (политик) явля- ется членом ровно одной компоненты (партии). Такая система подмножеств называет- ся разбиением множества. Алгоритмы генерирования разбиений множества рассмат- риваются в разделе 14.6. Важным моментом при работе с такими структурами данных является сопровождение изменений на протяжении времени, когда в компоненту добавляются или удаляются ребра (в партию вступают новые члены или старые выходят из нее). Типичные вопро- сы, возникающие при модифицировании множества путем изменения одного элемента, слияния или объединения двух элементов или же разбиения множества на части, сво- дятся к двум: "В каком множестве находится определенный элемент?" и "Находятся ли оба элемента в одном и том же множестве?". Нам понадобятся следующие структуры данных: ♦ коллекция контейнеров. Представление каждого подмножества в его собственном контейнере/словаре обеспечивает быстрый доступ ко всем элементам подмножест- ва, что облегчает выполнение операций объединения и пересечения. Но операция проверки членства является дорогой, т. к. требует отдельного поиска в каждой структуре данных до тех пор, пока не будет найден искомый элемент; ♦ обобщенный битовый вектор. Пусть z-й элемент массива содержит номер/имя со- держащего его подмножества. Запросы, идентифицирующие множество, и модифи-
Глава 12. Структуры данных 409 кации отдельных элементов можно выполнять за постоянное время. Но для объеди- нения двух подмножеств может потребоваться время, пропорциональное размеру универсального множества, т. к. необходимо идентифицировать каждый элемент в этих двух подмножествах и изменить его имя (по крайней мере, в одном подмно- жестве); ♦ словарь с атрибутом подмножества. Подобным образом каждый элемент в двоич- ном дереве можно связать с полем, в котором хранится имя содержащего его под- множества. Запросы для идентификации множества и модифицирования одного элемента можно выполнять за время, требуемое для выполнения поиска в словаре. Однако операции объединения и пересечения по-прежнему выполняются медленно. Необходимость эффективно исполнять операции объединения заставляет нас ис- пользовать структуру данных "объединение-поиск"; ♦ структура данных "объединение-поиск" Подмножество представляется посредст- вом корневого дерева, в котором каждый узел указывает вместо своего потомка на своего предшественника. Именем каждого подмножества служит имя корневого элемента. Членство определяется с легкостью — мы просто перемещаемся от одно- го указателя на родительский элемент на другой, пока не дойдем до корня. Опера- ция объединения двух подмножеств также не представляет сложностей. Мы просто присваиваем корню одного из деревьев указатель на другое, после чего все элемен- ты получают общий корень и, следовательно, одно и то же имя подмножества. В данном случае детали реализации оказывают большое влияние на асимптотиче- скую оценку производительности. Если при выполнении операции слияния всегда выбирать большее (более высокое) дерево в качестве корневого, то в результате га- рантированно получатся деревья логарифмической высоты (см. раздел 6.1.3). По- вторное прохождение пути, проложенного при каждом поиске элемента, и снабже- ние всех узлов в этом пути явными указателями на корень делает высоту дерева почти постоянной. Структура данных "объединение-поиск" является простой и бы- стро работающей структурой, с которой должен быть знаком каждый программист. Она не поддерживает разбиение подмножеств, созданных путем объединения, но это обычно не создает проблем. Реализации. Современные языки программирования содержат библиотеки полных и эффективных реализаций множеств. Библиотека стандартных шаблонов STL языка C++ содержит контейнеры множеств (set) и мультимножеств (multiset). Стандартный пакет утилит java.util включает в себя библиотеку Java Collections (JC). которая содер- жит контейнеры HashSet И TreeSet. Библиотека LEDA (см. раздел 19.1 1) содержит эффективные структуры данных слова- рей. разреженные массивы и структуры данных "объединение-поиск" для работы с раз- биениями множеств, все на языке C++. Реализация структуры данных "объединение-поиск" лежит в основе любой реализации алгоритма Крускала для построения минимального остовного дерева. Поэтому предпо- лагается. что все библиотеки графов, упоминаемые в разделе 12.4, содержат эту реали- зацию. Реализации минимальных остовных деревьев рассматриваются в разделе 15.3. Система компьютерной алгебры REDUCE (littp://www.reiluce-algebra.eom) содержит пакет SETS, который поддерживает теоретико-множественные операции как над яв-
410 Часть II Каталог алгоритмических задач ными, так и над неявными (символическими) множествами. Другие системы компью- терной алгебры могут иметь аналогичную функциональность. Примечания Оптимальные алгоритмы для таких операций над множествами, как пересечение и объ- единение, были представлены в публикации [Rei72]. В книге [Ram05] прекрасно описаны структуры данных для выполнения операций над множествами. Квалифицированно вы- полненный обзор фильтров Блума представлен в журнале [ВМ05], а результаты последних экспериментов в этой области изложены в докладе [PSS07]. Некоторые структуры данных сбалансированных деревьев поддерживают операции merge, meld, link и cut, что позволяет быстро выполнять операции объединения над непе- ресекаюшимися подмножествами. Хорошее описание таких структур можно найти в кни- ге [Таг83]. Джекобсон (Jacobson) улучшил структуру данных битового вектора для эффек- тивной поддержки операции выбора (поиска (-го установленного бита) как по времени, так и по памяти. Обзор структур данных для объединения непересекаюшихся множеств представлен в ра- боте |GI9l]. Верхний предел производительности, равный О(та(т, п)), для т операций поиска и объединения в «-элементном множестве был вычислен Тарьяном (Tarjan) (см. [Таг75]). Им же был найден и соответствующий нижний предел в ограниченной мо- дели вычислений (см. [Таг79]). Так как обратная функция Аккермана а(т. п) возрастает крайне медленно, то производительность приближается к линейной. Интересная связь между наихудшим случаем исполнения операций поиска и объединения и длиной после- довательности Дэвенпорта-Шинцеля (Davenport-Schinzel) — комбинаторной структуры, рассматриваемой в вычислительной геометрии — представлена в книге [SA95]. Степенным множеством (power set) множествах называется коллекция всех 2'5: подмно- жеств множества X. Явная манипуляция степенными множествами быстро становится трудной вследствие их большого размера. Для нетривиальных вычислений степенные множества требуется представлять неявно в символической форме. Информацию об алго- ритмах работы с символическими представлениями степенных множеств и результатов вычислительных экспериментов с ними можно найти в [BCGR92], Родственные задачи. Генерирование подмножеств (см. раздел 14.5), генерирование разбиений (см. раздел 14.6), вершинное покрытие (см. раздел 18.1), минимальное остовное дерево (см. раздел 15.3). 12.6. Kd-деревья Вход. Множество X, состоящее из п точек или более сложных геометрических объектов в ^-мерном пространстве. Задача. Создать дерево, которое делит пространство полуплоскостями таким образом, что каждый объект содержится в своей собственной ^-мерной прямоугольной области (рис. 12.7). Обсуждение. Kd-деревья и связанные с ними пространственные структуры данных иерархически разбивают пространство на небольшое количество ячеек, каждая из ко- торых содержит несколько представителей входного множества точек. Это обеспечи- вает быстрый доступ к любому объекту по его позиции. Мы просто проходим вниз по
Глава 12. Структуры данных 411 иерархической структуре, пока не дойдем до наименьшей ячейки, содержащей требуе- мый элемент, после чего перебираем все элементы ячейки, чтобы найти нужный. ВХОД ВЫХОД Рис. 12.7. Создание kd-дерева Обычно алгоритмы создают kd-деревья путем разбиения множества точек. Каждый узел дерева определяется плоскостью, проходящей через одно из измерений. В идеале, эта плоскость разбивает множество точек на равные правое и левое (или верхнее и нижнее) подмножества. Эти подмножества опять разбиваются на равные половины плоскостями, проходящими через другое измерение. Процесс прекращается после 1g/? разбиений, при этом каждая точка оказывается в своей собственной листовой ячейке. Секущие плоскости вдоль любого пути от корня к другому узлу определяют уникаль- ную прямоугольную область пространства. Каждая следующая плоскость разделяет эту область на две меньших. Каждая прямоугольная область определяется 2k плоскостями, где к— количество измерений. По мере продвижения вниз по дереву мы продолжаем ограничивать интересующую нас область этими полупространствами. Тип kd-дерева определяется выбором секущей плоскости. Возможны следующие вари- анты: ♦ циклический проход по измерениям. Сначала разбивается измерение сЦ, потом <72, .... d„, после чего мы возвращаемся к г/,; ♦ сечение вдоль самого большого измерения. Выбираем такое измерение, чтобы форма получившихся в результате его сечения областей была как можно ближе к квадрат- ной. Разбиение точек пополам не обязательно означает, что секущая плоскость про- ходит точно посередине прямоугольной области, т. к. все точки могут лежать у од- ного края подлежащей сечению области; ♦ квадродеревья и октодеревья. Вместо сечения одной плоскостью используются плоскости, параллельные всем координатным осям, которые проходят через данную точку сечения. Это означает, что в двух измерениях создаются четыре дочерние ячейки (квадрадерево), а в трех измерениях— восемь (октадерево). Использование квадрадеревьев особенно популярно при работе с видеоданными, когда листья де- рева таковы. что все пикселы в этих областях имеют одинаковый цвет;
412 Часть II. Каталог алгоритмических задач ♦ BSP-деревья. При двоичном разбиении пространства используются секущие плос- кости общего вида (т. е. не обязательно параллельные координатным осям), чтобы разделить пространство на ячейки таким образом, чтобы каждая ячейка содержала только один объект (скажем, многоугольник). Для некоторых наборов объектов та- кое разделение невозможно осуществить посредством только секущих плоскостей, параллельных координатным осям. Недостатком этого подхода является то обстоя- тельство, что получившиеся многогранные ячейки труднее поддаются обработке, чем ячейки прямоугольной формы; ♦ R-деревья. Это другая пространственная структура данных, подходящая для геомет- рических объектов, которые нельзя разделить по прямоугольным областям, ориен- тированным вдоль координатных осей, не разрезая их на части. На каждом уровне сечения объекты разбиваются на меньшее количество прямоугольных областей (возможно, накладывающихся друг на друга), чтобы создать удобные для поиска иерархические структуры, не разрезая объекты на части. В идеале, сечения разделяют поровну как пространство (давая в результате большие области правильной формы), так и набор точек (обеспечивая дерево логарифмической высоты); но для определенного экземпляра входа выполнить сечение таким образом может оказаться невозможным. Преимущества ячеек большого размера становятся очевидными во многих применениях kd-деревьев, обсуждаемых далее. ♦ Определение местоположения точки. Для выявления ячейки, в которой находится целевая точка q. выполняется обход дерева, начиная с корня, в процессе которого оба полупространства с каждой стороны секущей плоскости проверяются на нали- чие в них данной точки. Повторяя этот процесс на соответствующем дочернем узле, мы спускаемся вниз по дереву, пока не найдем листовую ячейку с искомым элемен- том q. Такой поиск занимает время, пропорциональное высоте дерева. Дополни- тельную информацию по вопросу определения местоположения точки см. в разде- ле 7 7 7 ♦ Поиск ближайшей точки. Чтобы в множестве .8’ найти точку, ближайшую к точке запроса <?, выполняется процедура определения местоположения ячейки с, содер- жащей точку q. Так как на границе ячейки с находится какая-либо точкар, то можно вычислить расстояние d(p, q) от точки р до точки q. Точкар, скорее всего, располо- жена вблизи точки q, но она может быть не единственной точкой, близкой к ней. Почему? Допустим, что точка q находится на границе ячейки. Тогда точка, бли- жайшая к точке q, может быть расположена сразу же слева от этой границы в дру- гой ячейке. Значит, нам нужно обойти все ячейки, которые находятся от ячейки с на расстоянии не превышающем d(p, q), и проверить, что ни одна из них не содержит более близких точек. В дереве из крупных ячеек правильной формы проверить нужно будет очень небольшое количество ячеек. Поиск ближайшей точки подробно обсуждается в разделе 17.5. ♦ Поиск в диапазоне. Как определить, какие точки находятся в целевой области? На- чиная с корня дерева, проверяем целевую область на пересечение с ячейкой (или на вхождение в нее этой ячейки), определяющей текущий узел. При положительном результате проверки выполняем такую же проверку на потомках; в противном слу- чае никакая из листовых ячеек ниже данного узла уже не будет нас интересовать.
Глава 12. Структуры данных 413 Таким образом быстро отсекаются не имеющие отношения к делу области про- странства. Поиск точки в указанном диапазоне подробно обсуждается в раз- деле /7 б. ♦ Пинск по частичному ключу. Допустим, что нам нужно найти точку р в множестве 5, но мы не располагаем подробной информацией об этой точке. Предположим, мы ищем в 3-мерном дереве, имеющем измерения для возраста, роста и веса, узел, представляющий человека в возрасте 35 лет ростом 175 сантиметров и с неизвест- ным весом. Начиная путь от корня, мы можем найти подходящий потомок по всем измерениям, кроме веса. Чтобы иметь уверенность, что найдена нужная точка, мы должны искать оба потомка этих узлов. Чем больше полей (измерений) известно, тем лучше, но такой поиск точки по частичной информации о ней может пройти значительно быстрее, чем сравнение всех точек с искомым ключом. Kd-деревья лучше всего годятся для приложений с умеренным количеством измерений, в диапазоне от 2 до 20. С увеличением количества измерений они теряют эффектив- ность, в основном из-за того, что объем единичного шара в ^-мерном пространстве уменьшается экспоненциально, в отличие от объема единичного куба. Следовательно, в приложениях, связанных, например, с поиском ближайшей точки, экспоненциально возрастает количество ячеек, проверяемых в пределах данного радиуса с центром в точке запроса. Кроме этого, для любой ячейки количество смежных ячеек увеличива- ется до 2k и, в конце концов, становится неуправляемым. Главный вывод из сказанного состоит в том, что следует избегать работы в многомер- ных пространствах, возможно, отказываясь от рассмотрения наименее важных измере- ний. Реализации. Программа KDTREE 2 содержит реализации kd-деревьев на языках C++ и FORTRAN 95 для эффективного поиска ближайшего соседа во многих измерениях. Подробности см. по адресу http://arxiv.org/abs/physics/0408067. Апплеты Java коллекции Spatial Index Demos (http://donar.umiacs.umd.edu/quadtree) демонстрируют многие варианты kd-деревьев. Алгоритмы, реализуемые в апплетах, рассматриваются в книге [Sam06]. Библиотека с открытым кодом на языке C++, носящая имя TerraLib (http:// www.terralib.org), предназначена для разработки приложений GIS (Geographic Infor- mation System, географическая информационная система). Эта библиотека включает в себя реализацию пространственных структур данных. Соревнование DIMACS в 1999 г. фокусировалось на структурах данных для поиска ближайшего соседа (см. [GJM02]). Наборы данных и коды можно загрузить с веб-сайта http://dimacs.rutgers.edu/Challenges. Примечания Самым лучшим справочником по kd-деревьям и другим пространственным структурам данных является книга [Sam06]. В ней подробно рассмотрены все основные (и многие второстепенные) варианты этих структур. Книга [Sam05] — краткий обзор этих структур. Принято считать, что kd-деревья разработал Дж. Бентли (J. Bentley) (см. [Веп75]), но ис- тория их появления туманна, как и у большинства популярных структур данных. Большое количество измерений ведет к падению производительности пространственных структур данных. Недавно был представлен простой, но мощный метод понижения раз-
414 Часть II. Каталог алгоритмических задач мерности, заключающийся в отображении многомерных пространств на произвольную гиперплоскость меньшей размерности. Как теоретические, так и экспериментальные ре- зультаты (см. [IM04] и [ВМ01] соответственно) указывают на то, что этот метод сохраняет расстояния довольно хорошо. Новейшей разработкой в области поиска ближайшего соседа в пространственных струк- турах высокой размерности являются алгоритмы для быстрого поиска точки, которая на- ходится доказуемо близко к точке запроса. На основе набора данных создается разрежен- ный взвешенный граф, после чего поиск ближайшего соседа осуществляется с примене- нием "жадного" обхода от произвольной точки по направлению к точке запроса. Ближайшим соседом считается точка, найденная в результате нескольких таких операций поиска, начатых от произвольной точки. Подобные структуры данных являются перспек- тивными для решения других задач в пространствах с большим количеством измерений. Подробности см. в [АМ93, AMN+98]. Родственные задачи. Поиск ближайшего соседа (см. раздел 17.5), определение место- положения точки (см. раздел 17.7). поиск в диапазоне (см. раздел 17.6).
ГЛАВА 13 Численные задачи Если вам, в основном, приходится иметь дело с численными задачами, значит, вы чи- таете не ту книгу. В таком случае вам лучше подойдет книга [PFTV07], в которой дает- ся отличный обзор фундаментальных задач в области численных методов, включая линейную алгебру, численное интегрирование, статистику и дифференциальные урав- нения. Соответствующие версии этой книги содержат исходный код для всех рассмат- риваемых в ней алгоритмов на языке C++, FORTRAN и даже Pascal. Комбинаторно- численные задачи, рассматриваемые в этом разделе, обсуждаются в ней не очень под- робно. но. тем не менее, следует знать о существовании данной книги. Дополнитель- ную информацию см. на веб-сайте http://www.nr.coni. Между численными и комбинаторными алгоритмами существуют, по крайней мере, два различия: ♦ по вопросам точности. Численные алгоритмы обычно выполняют повторяющиеся вычисления с плавающей точкой, и с каждой операцией накапливается ошибка, по- ка результаты в конце концов не теряют всякий смысл. Моим любимым примером такого накопления ошибки вычислений является случай с индексом Ванкуверской фондовой биржи, в котором за 22 месяца накопилась достаточно большая ошибка округления, чтобы индекс уменьшился до значения 574,081, в то время как пра- вильное значение должно быть 1098,982. Существует простой и надежный способ проверки погрешностей округления в чис- ленных программах. Для этого программа исполняется на данных как с одинарной, так и с двойной точностью, после чего результаты сравниваются; ♦ по наличию библиотек кода. Начиная с 1960-х годов, для численных алгоритмов создано большое количество высококачественных библиотек процедур, каких еще нет для комбинаторных алгоритмов. Причин этого несколько, включая: • раннее становление языка FORTRAN в качестве стандарта для численных рас- четов; • независимость численных расчетов от приложений, в которых они применяются; • существование крупных научных сообществ, нуждающихся в библиотеках об- щего назначения для решения численных задач. Скорее всего, у вас нет никаких причин для самостоятельной реализации алгорит- мов для решения любой задачи, описанной в этом разделе. Я рекомендую начинать поиски готового кода с библиотеки NetLib (см. раздел 19.1.5). Представления многих ученых и инженеров об алгоритмах основаны на методах про- граммирования и численных методах языка FORTRAN. А программисты изучают в студенческие годы работу с указателями и применение рекурсии, поэтому они чувст-
416 Часть II. Каталог алгоритмических задач вуют себя свободно с более сложными структурами данных, используемыми в комби- наторных алгоритмах. Оба эти сообщества могут и должны учиться друг у друга, по- скольку для моделирования таких задач, как распознавание образов, можно применять как численный, так и комбинаторный подход. Существует множество книг, посвященных численным алгоритмам. В частности, в до- полнение к уже упомянутой выше книге я рекомендую для ознакомления и такие: ♦ [СС05] — современный лидер продаж среди руководств по численному анализу: ♦ [Мак02] — это хорошо написанное учебное пособие, в котором язык Java вводится в мир численных расчетов. В книгу включен исходный код примеров; ♦ [Нат87] — это старое, но не устаревшее, пособие содержит четкое и понятное рас- смотрение фундаментальных методов численных расчетов; ♦ [SK00] — интересное обсуждение базовых численных методов, в котором нет слишком подробного описания алгоритмов благодаря использованию системы вы- числительной алгебры Mathematica. На мой взгляд, это очень хорошая книга; ♦ [CK07J — традиционный учебник по численному анализу, написанный с использо- ванием языка FORTRAN и включающий в себя обсуждение методов оптимизации и метода Монте-Карло в дополнение к таким традиционным темам, как извлечение корня, численное интегрирование, системы линейных уравнений, сплайны и диф- ференциальные уравнения; ♦ [ВТ92]— всестороннее, не привязанное ни к какому конкретному языку програм- мирования. обсуждение всех стандартных тем, включая параллельные алгоритмы. Это наиболее полное изо всех упомянутых здесь руководств. 13.1. Решение системы линейных уравнений Вход. Матрица А размером т х п и вектор b размером т х I, совместно представляю- щие т линейных уравнений с п переменными (рис. 13.1). Рис. 13.1. Решение системы линейных уравнений
Глава 13. Численные задачи 417 Задача. Найти такой вектор х. для которого А-х = Ь. Обсуждение. Потребность в решении систем линейных уравнений возникает в при- близительно 75% всех научных вычислительных задач (см. [DB74]). Например, приме- нение закона Кирхгофа для анализа электрических схем порождает систему уравнений, решение которой определяет прохождение тока через каждую ветвь схемы. При анали- зе сил, действующих на натянутый трос, возникает аналогичная система уравнений. Даже задача поиска точки пересечения двух или больше линий сводится к решению небольшой системы линейных уравнений. Ноне все системы уравнений имеют решения, например: |2х + Зу = 5 [2х + Зу = 6 А некоторые системы уравнений, наоборот, имеют несколько решений, например: j 2х + Зу = 5 14 X + бу = IО Такие системы уравнений называются вырожденными, и их можно распознать, прове- рив определитель матрицы коэффициентов на равенство нулю. Задача решения систем линейных уравнений настолько важна в научных и экономиче- ских приложениях, что для ее решения существует достаточное количество превосход- ных программных реализаций. Нет никакого смысла создавать собственную реализа- цию, хотя базовый алгоритм (метод Гаусса) и изучается в старших классах средней школы. Особенно это справедливо в отношении систем с большим количеством неиз- вестных. Метод Гаусса основан на том обстоятельстве, что при умножении уравнения на кон- станту его решение не меняется (если х = у. то 2х = 2у) и при сложении уравнений сис- темы ее решение не меняется (т. е. решение системы уравнений х = у и те = z такое же. как и решение системы х = у и х + w = у + z). При решении методом Гаусса уравнения умножаются на константу и складываются так, чтобы можно было последовательно исключать переменные из уравнений и привести систему к ступенчатому виду. Временная сложность метода Гаусса для системы из п х п уравнений равна О(п). т. к. для того, чтобы исключить /-ю переменную, мы добавляем умноженную на константу копию н-го члена /-й строки к каждому из оставшихся п — 1 уравнений. В этой задаче константы играют важную роль. Алгоритмы, которые для получения решения выпол- няют только частичную обработку матрицы коэффициентов, а затем производят обрат- ную замену, используют на 50% меньше операций с плавающей точкой, чем простые алгоритмы. Прежде чем приступазь к решению системы линейных уравнений, вам нужно ответить наследующие вопросы: ♦ Влияют ли на решение ошибки округления? Реализация метода Гаусса была бы со- всем простым делом, если бы не ошибки округления. Они накапливаются с каждой операцией и вскоре могут полностью исказить решение, особенно для почти выро- жденных матриц. 14 Зак. 3741
418 Часть II. Каталог алгоритмических задач Чтобы устранить влияние ошибок округления, следует подставлять решение в каж- дое из первоначальных уравнений и проверять, насколько оно близко к искомому значению. Повысить точность решений систем линейных уравнений можно с по- мощью итерационных методов решения. Хорошие пакеты для решения систем ли- нейных уравнений должны содержать такие процедуры. Ключом к минимизации ошибок округления в решениях методом Гаусса является выбор правильных опорных уравнений и переменных и умножение уравнений на константы для исключения больших множителей. Это особое искусство, и вам сле- дует использовать тщательно разработанные библиотечные процедуры, описанные ниже. ♦ Какую библиотечную процедуру нужно использовать? Умение выбрать правиль- ный код также является своего рода искусством. Если вы следуете советам из этой книги, то начните с неспециализированных пакетов для решения систем линейных уравнений. Очень вероятно, что их будет достаточно для ваших задач. Однако вам следует поискать в руководстве пользователя эффективные процедуры для решения особых типов систем линейных уравнений. Если окажется, что ваша матрица при- надлежит к одному из специальных типов, поддерживаемых пакетом, то время ее решения может быть уменьшено с кубического до квадратичного или даже линей- ного. ♦ Является ли решаемая система разреженной? Если в матрице А окажется неболь- шое количество ненулевых элементов, то матрица является разреженной (sparse), и нам повезло. Если эти несколько ненулевых элементов сосредоточены возле диаго- нали, то матрица является ленточной (banded), и нам повезло еще больше. Алго- ритмы для уменьшения ширины ленты матрицы рассматриваются в разделе 13.2. Для решения можно воспользоваться многими другими стандартными образцами разреженных матриц, подробности о которых можно узнать в руководстве пользо- вателя выбранного вами программного пакета или в книге, целиком посвященной численному анализу. ♦ Будет ли использоваться одна и та же матрица коэффициентов для решения не- скольких систем? В таких приложениях, как аппроксимация кривой методом наименьших квадратов, уравнение А-х = b требуется решать несколько раз с разны- ми векторами Ь. Чтобы облегчить этот процесс, матрицу А можно подвергнуть предварительной обработке. LU-разложение — это представление матрицы А с по- мощью нижней (Z,) и верхней (U) треугольных матриц, для которых L-U = А. LU-разложение можно использовать для решения уравнения А-х = Ь. т. к. А-х = (L • U) • х = /,•( U-x) = b. Это эффективное решение, т. к. обратная подстановка позволяет решить треуголь- ную систему уравнений за квадратичное время. После выполнения LU-разложения за время О(п ) решение уравнения L-y = Ь, а потом уравнения U-x =у дает нам ре- шение для х за два шага с временем исполнения каждого О(п~), вместо одного шага с временем исполнения О(п). Задача решения системы линейных уравнений эквивалентна задаче обращения матри- цы, т. к. Ах = В ♦-> АлАх = А’'в, где I— единичная матрица (1 = А'1 А). Но этого подхо-
Глава 13. Численные задачи 419 да следует избегать, т. к. он в три раза медленнее, чем метод Гаусса. LU-разложение оказывается полезным как для обращения матриц, так и для вычисления определите- -лей (см. раздел 13.4). Реализации. Самой лучшей библиотекой для решения систем линейных уравнений по- видимому является библиотека LAPACK, более поздняя версия библиотеки UNPACK (см. [DMBS79]). Обе эти библиотеки, написанные на языке FORTRAN, являются частью библиотеки NetLib. Подробности см. в разделе 19.1.5. Версии библиотеки LAPACK имеются и для других языков, в частности для С — CLAPACK и для C++— LAPACK++. Для этих процедур имеется интерфейс на языке C++, Template Numerical Toolkits, который можно загрузить с веб-сайта http://math.nist.gov/tnt. Универсальная библиотека для научных расчетов JScience содержит обширный пакет инструментов для линейной алгебры (включая определители). Еще одним пакетом для работы с матрицами на языке Java является библиотека JAMA. Ссылки на эти и другие подобные библиотеки см. на сайте http://math.nist.gov/javanumerics. Еще одним источником руководства и процедур для решения систем линейных урав- нений является книга [PFTV07] (www.nr.com). Наиболее убедительной причиной для использования этих реализаций вместо бесплатных будет отсутствие у разработчика уверенности при работе с численными методами. Примечания Стандартным справочником по алгоритмам для работы с системами линейных уравнений является книга [GL96]. Хорошие описания алгоритмов для метода Гаусса и LU-раз- ложения можно найти в книге [CLRSOl], Существует множество пособий по численному анализу, таких как [ВТ92], [СК07], [SKOO]. Обзор структур данных для систем линейных уравнений представлен в книге [РТ05]. Параллельные алгоритмы для систем линейных уравнений обсуждаются в публикациях [Gal90], [KSV97] и [Ort88]. Решение систем линейных уравнений является одним из наи- более важных приложений, в которых на практике широко применяются параллельные архитектуры. Обращение матриц и, соответственно, решение систем линейных уравнений можно вы- полнить за время, требуемое для умножения матриц, с помощью алгоритма Штрассена с последующим сведением. Среди хороших описаний равноценных задач можно назвать [AHU74] и [CLRSOl], Родственные задачи. Умножение матриц (см. раздел 13.3), определители и перманен- ты (см. раздел. 13 4). 13.2. Уменьшение ширины ленты матрицы Вход. Граф G = (V, Е), представляющий матрицу Мразмером п х п. Задача. Найти такую перестановку вершин р, которая минимизирует самое длинное ребро, когда вершины выстроены в линию, т. е. минимизирует max(/>rt е E\p(i)—p(j)\ (рис. 13.2).
420 Часть II Каталог алгоритмических задач Обсуждение. Уменьшение ширины ленты представляет собой незаметную на первый взгляд, но очень важную задачу, как для графов, так и для матриц. Применительно к матрицам, уменьшение ширины ленты заключается в перестановке строк и столбцов разреженной матрицы, чтобы минимизировать расстояние b любого ненулевого эле- мента от центральной диагонали. Это важная операция в решении систем линейных уравнений, т. к. на матрицах с лентой шириной b метод Гаусса исполняется за время О(пЬ~), что является большим улучшением общего времени исполнения О(п} алгорит- ма, когда b « п. Уменьшение ширины ленты в графах проявляется не столь очевидным образом. В ка- честве одного примера задачи уменьшения ширины ленты в графах можно привести задачу упорядочивания п элементов электрической схемы в линию таким образом, чтобы минимизировать длину самого длинного проводника (и таким образом миними- зировать временную задержку). В данном случае каждая вершина графа представляет элемент схемы, а каждое ребро — проводник, соединяющий два элемента. Рассмотрим другой пример: гипертекстовое приложение, где нам нужно сохранять большие объек- ты (скажем, изображения) на магнитной ленте. Каждое изображение имеет набор ука- зателей на набор изображений, к которым можно обратиться на следующем шаге (т. е. гиперссылки). Мы хотим поместить связанные изображения на магнитной ленте как можно ближе друг к другу, чтобы минимизировать время поиска. Это как раз и являет- ся задачей уменьшения ширины ленты. Более общие задачи, такие как компоновка прямоугольных схем и магнитные диски, наследуют сложность и эвристические клас- сы линейных версий. В задаче уменьшения ширины ленты требуется упорядочить вершины в линию таким образом, чтобы минимизировать длину самого длинного ребра, но существует не- сколько вариантов задачи. В задаче линейного расположения (linear arrangement) тре- буется минимизировать сумму длин всех ребер. Эту задачу можно применить для ком- поновки схем, в которой требуется расположить микросхемы таким образом, чтобы минимизировать общую длину проводников. В задаче минимизации профиля требуется минимизировать сумму длин однонаправленных ребер (т. е. для каждой вершины v минимизировать длину самого большого ребра, вторая вершина которого находится слева от вершины v). К сожалению, задача уменьшения ширины ленты во всех ее вариантах является NP-полной. Она остается NP-полной, даже если графом входа является дерево с мак-
Глава 13. Численные задачи 421 симальной степенью вершины, равной 3, что является самым строгим ограничением, которое я когда-либо встречал. Поэтому единственными способами ее решения явля- ются метод полного перебора решений и эвристический подход. К счастью, эвристические методы специального назначения хорошо изучены и для наилучших из них имеются высококачественные реализации. В основе этих реализа- ций лежит обход в ширину, начинающийся с определенной вершины v, где эта верши- на помещается в самую левую позицию упорядочения. Все вершины на расстоянии I от вершины v помешаются непосредственно справа от нее, за ними следуют вершины на расстоянии 2 и т. д., пока не будут обработаны все вершины графа. Популярные эвристические алгоритмы различаются по количеству рассматриваемых начальных вершин и по способу упорядочивания равноудаленных вершин. В последнем случае хорошим методом, по-видимому, является выбор самой левой из равнозначных вершин низкой степени. Реализации наиболее популярных эвристических алгоритмов— Катхиллд-Макки (Cuthill-McKee) и Гиббса-Пула-Стокмейера (Gibbs-Poole-Stockmeyer)— рассматрива- ются в подразделе "Реализации". Время исполнения в наихудшем случае для алгоритма Гиббса-Пула-Стокмейера равно О(п), что свело бы на нет любую возможную эконо- мию при решении систем линейных уравнений, но его практическая производитель- ность близка к линейной. Программы полного перебора могут точно найти минимальную полосу ленты посред- ством поиска с возвратом в наборе из п\ возможных перестановок вершин. Простран- ство поиска можно значительно разредить с помощью отсечения; для этого нужно на- чать с хорошего эвристического решения задачи уменьшения ширины ленты и пооче- редно добавлять вершины в крайние слева и справа свободные позиции в частичной перестановке. Реализации. Код, написанный Дель Корсо (Del Corso) и Манзини (Manzini), для поис- ка точных решений задач минимизации ширины ленты (см. [СМ99]) можно загрузить с веб-сайта http://www.mfn.unipmn.it/~manzini/bandmin. Улучшенные способы реше- ния задачи на основе целочисленного программирования были разработаны Капрарой (Саргага) и Салазар-Гонзалесом (Salazar-Gonzalez), см. [CSG05]. Их реализацию алго- ритма метода ветвей и границ на языке С можно загрузить с сайта http://joc.pubs. informs.org/Supplements/Caprara-2. Библиотека NetLib (см. раздел 19.1.5) содержит реализации на языке FORTRAN алго- ритмов Катхилла-Макки (см. [CGPS76], [Gib76], [СМ69]) и Гиббса-Пула-Стокмейера (см. [Lew82], [GPS76]). Обзор экспериментальных оценочных тестов этих и других ал- горитмов на наборе из 30 матриц рассматривается в публикации [Eve79b]. Согласно этим тестам, бессменным лидером является алгоритм Гиббса-Пула-Стокмейера. В докладе [PetO3] предоставляются результаты всестороннего изучения эвристических алгоритмов для решения задачи минимального линейного размещения. Примечания Отличный обзор современных алгоритмов решения задачи минимизации ширины ленты и родственных задач на графах представлен в [DPS02], Обзор решений задачи минимизации ширины ленты для графов и матриц вплоть до 1981 г. можно найти в [CCDG82],
422 Часть II. Каталог алгоритмических задач Эвристические алгоритмы специального назначения явились предметом всестороннего изучения, что свидетельствует о их важности в численных расчетах В докладе [Eve79b] перечислены не менее полусотни различных алгоритмов минимизации ширины ленты. В публикации [CR.01] представлены результаты исследований нового класса спектраль- ных эвристических алгоритмов минимизации ширины ленты. Сложность задачи минимизации ширины ленты была впервые установлена в докладе [Рар76Ь], а ее сложность на деревьях с максимальной степенью 3 — в докладе [GGJK.78]. Для задачи минимизации ленты фиксированной ширины к существуют алгоритмы с поли- номиальным временем исполнения (см. [Sax80]). Для общей задачи существуют аппрок- симирующие алгоритмы с гарантированным полилогарифмическим временем исполнения (см. [BKRV00]). Родственные задачи. Решение линейных уравнений (см. раздел /3.7), топологическая сортировка (см. раздел 15.2). 13.3. Умножение матриц Вход. Матрица Л размером х х у и матрица В размером у * z. Задача. Вычислить матрицу А х В размером х х z (рис. 13.3). ВХОД 13 18 23 18 25 32 23 32 41 ВЫХОД Рис. 13.3. Умножение матриц Обсуждение. Умножение матриц является фундаментальной задачей линейной алгеб- ры. Значение этой задачи для комбинаторных алгоритмов определяется ее эквивалент- ностью многим другим задачам, включая транзитивное замыкание и сокращение, син- таксический разбор, решение системы линейных уравнений и обращение матриц. Та- ким образом, создание быстрого алгоритма умножения матриц влечет за собой появление быстрых алгоритмов для решения всех этих задач. Умножение матриц воз- никает как самостоятельная задача при вычислении результатов таких координатных преобразований, как масштабирование, вращение и перемещение в робототехнике и компьютерной графике. В листинге 13.1 представлен псевдокод алгоритма, вычисляющего произведение мат- рицы А размером х х у и матрицы В размером у х z за время O(xyz). Листинг 13.1, Умножение матриц . ......... ifc‘.. . ........ Инициализируем массив M[i, j] нулями для всех i z for i = 1 to x do
Гпава 13. Численные задачи 423 for j = 1 to Z for к = 1 to у M[i, j] = + A[i, k] - A[k, j] Реализация этого алгоритма на языке С приводится в листинге 2.4 в разделе 2.5.4. Ка- залось бы, что трудно превзойти этот простой алгоритм на практике. Но обратите вни- мание, что три его цикла можно переставить произвольным образом, не повлияв на конечный результат. После такой перестановки изменятся характеристики обращений к памяти и. вследствие этого, эффективность использования кэша. Можно ожидать, что разброс во времени исполнения между шестью возможными реализациями (переста- новками циклов) этого алгоритма достигнет 10-20%, но уверенно предсказать наилуч- шую из них (обычно ikj) нельзя, не выполнив ее на конкретном компьютере с конкрет ными матрицами. При умножении матриц с шириной ленты, равной Ь, когда все ненулевые элементы матриц А и В расположены среди b элементов главных диагоналей, возможно ускоре- ние времени исполнения до O(xbz), т. к. нулевые элементы не участвуют в вычислении произведения матриц. Использование рекуррентных методов типа "разделяй и властвуй" позволяет получить еще более быстрые алгоритмы для умножения матриц. Но эти алгоритмы трудно про- граммировать. а превзойти тривиальный алгоритм они могут только на очень больших матрицах. Кроме того, они обладают меньшей вычислительной устойчивостью. Наи- более известным из этих алгоритмов является алгоритм Штрассена с временем испол- нения О(п2,81). Экспериментальные результаты (которые рассматриваются ниже) рас- ходятся в определении критической точки, в которой производительность алгоритма Штрассена начинает превышать производительность простого кубического алгоритма, но можно считать, что это происходит при п ~ 100. Однако при умножении цепочки из более чем двух матриц есть лучший способ сэко- номить на вычислениях. Вспомните, что при умножении матрицы размером х х у на матрицу размером y*z образуется матрица размером х х z. Таким образом, при умно- жении цепочки матриц слева направо могут образовываться большие промежуточные матрицы, вычисление каждой из которых занимает много времени. Умножение матриц не обладает перестановочным свойством, но обладает ассоциативным, поэтому эле- менты цепочки можно заключить в скобки любым удобным способом, что не повлияет на конечный результат умножения. Оптимальный порядок скобок можно создать с по- мощью стандартного алгоритма динамического программирования. Но такая оптими- зация может быть оправданной только в том случае, если матрицы далеки от квадрат- ных, а их перемножение выполняется достаточно часто. Обратите внимание, что опти- мизации подвергаются размеры измерений в цепочке, а не сами матрицы. Если же все матрицы одинакового размера, то такая оптимизация невозможна. Умножение матриц имеет особенно интересную трактовку в задаче подсчета количест- ва путей между двумя вершинами графа. Пусть А представляет матрицу смежности графа G. т. е. /ф,у] = 1, если вершины i и j соединены ребром, и A[i,j] = 0 в противном случае. Теперь рассмотрим квадрат этой матрицы. А2 = А * А. Если A2[i,j]> 1, то это означает, что должна существовать такая вершина к, для которой A[i, к] = A[k,j], по- этому длина пути от вершины / к вершине к и к вершине j равна 2. В более общем
424 Часть II. Каталог алгоритмических задач смысле, Л* [Л /] подсчитывает количество путей длиной ровно к между вершинами i и j. При этом подсчете учитываются непростые пути, т. е. пути с повторяющимися верши- нами. например путь, проходящий через последовательность вершин /, к, i и j Реализации. В работе [DN07] описывается очень эффективный код для умножения матриц, в котором в оптимальной точке выполняется переключение с алгоритма Штрассена на кубический алгоритм. Эту реализацию можно загрузить с веб-сайта http://www.ics.uci.edu/~fastmm/. В предшествующих экспериментах было установлено значение точки пересечения п ~ 128, в которой производительность алгоритма Штрас- сена начинает превосходить производительность кубического алгоритма (см. [BLS91], [CR76]). Таким образом, кубический алгоритм будет, вероятнее всего, самым лучшим выбором для матриц не очень большого размера. Лучшей библиотекой процедур линейной ал- гебры является LAPACK, более поздняя версия библиотеки UNPACK (см. [DMBS79]). которая содержит ряд процедур для умножения матриц. Эти коды на языке FORTRAN являются частью библиотеки NetLib (см. раздел 19.1.5). Алгоритм 601 (см. [McN83]) из коллекции алгоритмов ассоциации АСМ (http://calgo.acm.org/) представляет собой пакет на языке FORTRAN для работы с раз- реженными матрицами и содержит процедуры для умножения любых комбинаций раз- реженных и плотных матриц. Подробности см. в разделе 19.1.5. Примечания Алгоритм Винограда (Winograd) для быстрого умножения матриц уменьшает количество операций умножения наполовину по сравнению с обычным алгоритмом. Хотя этот алго- ритм поддается реализации, необходимые дополнительные накладные расходы ставят под сомнение его превосходство. Описания алгоритма Винограда (см. [Win68]) можно найти в [CLRS01], [Мап89] и [Win80]. По моему мнению, история теоретической разработки алгоритмов берет начало с публи- кации алгоритма Штрассена (см. [Str69]) для умножения матриц с временем исполнения Впервые улучшение алгоритма в асимптотическом отношении стало целью, за- служивающей самостоятельного исследования. Последующие улучшения алгоритма Штрассена становились все менее практичными. В настоящее время наилучшим алгорит- мом для умножения матриц является алгоритм Коперсмита-Винограда (см. [CW90]) с временем исполнения О(п2'31ь), но высказываются догадки, что можно добиться произ- водительности 0(ц“). Описание альтернативного подхода, позволившего получить алго- ритм с временем исполнения О(п2‘"), см. в [CKSU05]. Создание эффективных алгоритмов для умножения матриц требует тщательного управле- ния кэшем. Исследования по этому вопросу см. в [BDN01] и [HUW02], Интерес к квадратам графов возник не только в связи с задачей подсчета путей. В работе [Fle74] приведено доказательство, что квадрат любого двусвязного графа содержит га- мильтонов цикл. Алгоритмы поиска квадратных корней графов, т. е. вычисление матрицы .1 по матрице А2, обсуждаются в [LS95], Задачу умножения булевых матриц можно свести к общей задаче умножения матриц (см. [CLRS01]). В алгоритме четырех русских (four-Russians algorithm) для умножения бу- левых матриц (см. [ADK.F70]) применяется предварительная обработка с целью создания всех подмножеств lg/г строк для ускорения доступа при выполнении собственно операции умножения, что позволяет получить время исполнения O(«7lgH). Дополнительная предва-
Глава 13. Численные задачи 425 ригельная обработка может улучшить это время до O(n7lg"n) (см. [Ryt85]). Описание ал- горитма четырех русских, включая процедуру дополнительного ускорения, см. в [Мап89]. Хорошее описание алгоритма для умножения цепочки матриц представлено в [BvG99] и [CLRS01] в виде стандартного хрестоматийного примера динамического программирова- ния. Родственные задачи. Решение линейных уравнений (см. раздел 13.1), поиск кратчай- шего пути (см. раздел 15 4) 13.4. Определители и перманенты Вход. Матрица Л/размером п * п. Задача. Найти определитель |Л7] или перманентрегт(М) матрицы т (рис. 13.4). 2* det -2 10 -3 13 det 4 -2 10 -1 * del 5 -3 13 ВХОД выход Рис. 13.4. Вычисление определителя матрицы Обсуждение. Определители матриц— это понятие из линейной алгебры, которое можно использовать для решения многих задач. В частности, таких как: ♦ проверка, является ли матрица вырожденной, т. е. проверка существования матри- цы, обратной данной. Матрица Л/является вырожденной, если |Л7] = 0. ♦ проверка, располагается ли множество из d точек на плоскости в менее, чем d изме- рениях. Если да, то определяемая ими система уравнений является вырожденной, поэтому |Л7] = 0; ♦ проверка, находится ли точка справа или слева от линии или плоскости. Эта задача сводится к проверке знака определителя (см. раздел 17.1)\ ♦ вычисление площади или объема треугольника, четырехгранника или другого мно- гогранника. Эти величины являются функцией значения определителя (см. раз- дел 171). Определитель матрицы М— это сумма всех п\ возможных перестановок л, из п столб- цов матрицы: <=i /=1
426 Часть II Каталог алгоритмических задач где signing— количество пар элементов, расположенных не по порядку (называемых инверсиями), в перестановке тт,. Прямая реализация этого определения дает алгоритм с временем исполнения О(п\) точно так же, как и метод алгебраических дополнений. Алгоритмы с лучшей произво- дительностью для вычисления определителей основаны на LU-разложении, рассматри- ваемом в разделе 13.1. При этом подходе определитель матрицы М вычисляется просто как произведение диагональных элементов LU-разложения матрицы, для чего требует- ся время О(р). В комбинаторных задачах часто встречается похожая на определитель функция, назы- вающаяся перманентом. Например, перманент матрицы смежности графа подсчитыва- ет количество совершенных паросочетаний графа. Перманент матрицы М вычисляется таким образом: регт(М) = П М J >77! ] /=i I-1 Единственное отличие перманента от определителя — то, что все произведения поло- жительные. Удивительно, задача вычисления перманента является NP-полной, хотя определитель можно с легкостью вычислить за время О(п). Фундаментальная разница между этими двумя функциями состоит в том, что det(AB) = det(A) * det(B), в то время как репп(АВ) Фрегт{А) * репп{В). Существуют алгоритмы вычисления перманента за вре- мя О(л"2"), что оказывается значительно быстрее, чем время О(п\). Таким образом, вы- числение перманента матрицы размером 20*20 является вполне реальной задачей. Реализации. Пакет инструментов для линейной алгебры UNPACK, содержит множе- ство процедур на языке FORTRAN для вычисления определителей. Этот пакет входит в библиотеку NetLib, информацию о которой см. в разделе 19.1.5. Библиотека для научных расчетов JScience содержит обширный пакет инструментов для линейной алгебры (включая определители). Еще одним пакетом для работы с мат- рицами на языке Java является библиотека JAMA. Ссылки на эти и многие другие биб- лиотеки см. на сайте http://math.nist.gov/javanumcrics Эффективная процедура для вычисления перманентов матриц приводится в книге [NW78], Подробности см. в разделе 19.1.10. В работе [Cas95] описывается процедура на языке С для вычисления перманентов на основе подсчета структур Кекуле (Kekule structures) в вычислительной химии. Две разных процедуры для аппроксимации перманента разработаны профессором математики университета Мичигана Александром Барвинком. Первый, на основе рабо- ты [BS07], предоставляет коды для аппроксимации перманента и гафниана матрицы, а также количества остовных деревьев графа. Подробности см. на веб-сайте http://www.math.lsa.umich.edu/~barvinok/manual.html. Второй, на основе работы [SBOI], может выдавать приблизительное значение перманента матрицы размером 200*200 за секунды. Подробности см. по адресу http://www.math.lsa.umich.edu/ ~barvinok/code.html.
Глава 13. Численные задачи 427 Примечания Правило Крамера сводит задачи обращения матриц и решения систем линейных уравне- ний к задаче вычисления определителей. Но алгоритмы на основе LU-разложения рабо- тают быстрее. Описание правила Крамера см. в [ВМ53]. Определители можно вычислять за время о(/), используя метод быстрого умножения матриц, как показано в книге [AHU83]. Такие алгоритмы обсуждаются в разделе 13.3. Разработчиком быстрого алгоритма для вычисления знака определителя является Кларк- сон (Clarkson); см. [С1а92]. В работе [Val79] было доказано, что задача вычисления перманента является #Р-полной, где #Р означает класс задач, решаемых на "счетной" машине за полиномиальное время. "Счетная" машина возвращает количество разных решений задачи. Подсчет количества гамильтоновых циклов в графе является #Р-полной задачей, очевидно NP-сложной (а, возможно, еще сложнее), т. к. любой ненулевой результат подсчета означает, что граф является гамильтоновым. Задачи подсчета могут быть #Р-полными, даже если соответст- вующую задачу разрешимости можно решить за полиномиальное время с использованием перманента и совершенных паросочетаний. Основным справочником по перманентам является [Min78]. В книге [NW78] представле- на разновидность алгоритма для вычисления перманента за время О(л~2"). За последнее время были разработаны вероятностные алгоритмы для приблизительного вычисления перманента, вершиной развития которых является полиномиальный рандоми- зированный алгоритм, обеспечивающий сколь угодно точную аппроксимацию за время, полиномиально зависящее от размера матрицы входа и допускаемого уровня погрешности (см. [JSV01]). Родственные задачи. Решение линейных уравнений (см. раздел 13.1), паросочетание (см.раздел 15.6), геометрические примитивы (см. раздел 17.1). 13.5. Условная и безусловная оптимизация Вход. Функция/*], —-Хп). Задача. Найти точку р = (р\, ..., рп), в которой функция/достигает минимума или мак- симума (рис. 13.5). Рис. 13.5. Поиск минимума и максимума функции
428 Часть II. Каталог алгоритмических задач Обсуждение. В большей части материала этой книги рассматриваются алгоритмы для оптимизации той или иной величины. В этом разделе мы рассмотрим общую задачу оптимизации функций, для которых по причине отсутствия необходимой структуры или знаний мы не можем применить подогнанные под специфичную задачу алгоритмы, рассматриваемые в других разделах книги. Задача оптимизации возникает, когда целевую функцию требуется настроить на опти- мальную производительность. Допустим, что мы разрабатываем программу для инве- стиций на бирже. Здесь мы имеем некоторые финансовые данные, которые можем ана- лизировать.— отношение цены акции к чистой прибыли, процентную ставку и цену акции. Все эти величины являются функцией времени /. Самым сложным здесь являет- ся определение коэффициентов следующей формулы: качество акций(/) = -с, х цена(/) + с2 х процентная ставка(/) + с3 х отношение цены к прибыли(/) Нам нужно найти такие значения с2 и с3, которые оптимизируют значение функции качества акций. Подобные вопросы возникают при настройке оценочных функций для любой задачи распознавания закономерностей. Задачи безусловной оптимизации также возникают в научных расчетах. Физические системы, от протеиновых цепочек до галактик, естественно стремятся минимизировать свою "энергию" или "потенциальную функцию". Поэтому в программах, которые пы- таются эмулировать природу, потенциальная функция часто определяется путем при- сваивания некоторого значения каждой возможной конфигурации объектов с после- дующим выбором из всех этих конфигураций той, у которой потенциал минимален. Глобальные задачи оптимизации, как правило, сложны, и для их решения существует множество подходов. Чтобы выбрать правильный путь решения таких задач, ответьте наследующие вопросы. ♦ Задача какого типа решается (условной или безусловной оптимизации)? В задачах безусловной оптимизации нет никаких ограничений на значения параметров, кроме условия, чтобы они максимизировали значение функции. Но во многих приложени- ях требуется накладывать ограничения на эти параметры, вследствие чего некото- рые значения, которые в противном случае могли бы дать оптимальное решение, становятся запрещенными. Например, предприятие не может иметь в плате мень- ше, чем нулевое количество сотрудников, невзирая на то, сколько денег они могли бы сэкономить, имея, скажем, -100 сотрудников. Задачи условной оптимизации обычно требуют подхода математического программирования наподобие линейного программирования, рассматриваемого в разделе 13.6. ♦ Можно ли оптимизируемую функцию выразить формулой? Иногда оптимизируе- мая функция представлена в виде алгебраической формулы, например, требуется найти минимальное значение /(и) = п~ -6п + 2" ' '. В таком случае нужно взять про- изводную этой функции и вычислить, для каких точек р' производная будет равна нулю, т. е./'(/>') = 0. Это будут точки локального максимума или минимума, разли- чить которые можно, взяв вторую производную или просто подставив значение р' в функцию / и оценив полученный результат. Такие системы, как Mathematica и Maple, довольно успешно вычисляют производные, хотя эффективное использова-
Глава 13. Численные задачи 429 ние компьютерных приложений вычислительной алгебры в определенной степени сродни шаманству. Тем не менее имеет смысл попробовать эти системы, поскольку с их помощью можно, как минимум, создать диаграмму оптимизируемой функции, позволяющую получить наглядное представление о поведении функции. ♦ Насколько затратно вычисление значения функции в определенной точке? Часто вместо функции приходится иметь дело с программой или процедурой, которая вы- числяет значение функции f в данной точке. Мы можем попытаться определить оп- тимальное значение, проанализировав значения функции в разных точках. Возможности нашего поиска зависят от того, насколько эффективно мы вычисляем значение функции. Допустим, что у нас имеется функция/(х|.х„) для оценки си- туации на шахматной доске, где Xi — стоимость пешки, х2 — стоимость слона и т. д. Чтобы выбрать оптимальные значения набора коэффициентов для этой функции, нам нужно сыграть с ней большое количество игр или протестировать ее на библио- теке известных игр. Очевидно, что такой подход отнимает много времени, поэтому нужно стремиться минимизировать количество вычислений функции, выполняемых при оптимизации коэффициентов. ♦ Сколько имеется измерений? Сколько измерений требуется? Трудность поиска глобального оптимального значения быстро возрастает с увеличением количества измерений (или параметров). По этой причине часто выгодно уменьшить количест- во измерений, игнорируя некоторые параметры. Это идет против интуиции, и неис- кушенный программист, скорее всего, вставит в свою оценочную функцию как можно больше переменных. Но оптимизация таких сложных функций является слишком трудной задачей. Намного лучше будет начать с трех до пяти наиболее важных переменных и получить хорошую оптимизацию для них. ♦ Насколько плавным является график функции? Основная опасность, подстерегаю- щая нас при глобальной оптимизации,— это ловушка локального оптимума. Рас- смотрим задачу поиска наивысшей точки горного массива. Если данный массив со- стоит всего лишь из одной горы правильной формы, то наивысшую точку можно найти, просто идя вверх по этой горе. Но если мы имеем дело с горным хребтом, содержащим несколько высоких гор. то найти наивысшую точку будет труднее. Плавность графика функции является качеством, позволяющим быстро определить локальный оптимум, двигаясь из данной точки. Когда мы ищем вершину горы, на- правляясь вверх, мы предполагаем наличие плавности. Если же высота в любой точке определяется полностью случайной функцией, то у нас нет иного способа по- иска оптимума, кроме проверки каждой точки. В наиболее эффективных алгоритмах безусловной глобальной оптимизации для поиска локального оптимума применяются производные и частные производные, позволяю- щие выяснить направление, в котором следует двигаться из данной точки, чтобы дос- тичь максимального или минимального значения функции. Иногда такие производные можно вычислить аналитическим способом или же значение производной может быть вычислено приблизительно, путем вычисления разности между значениями функции в соседних точках. Для поиска локального оптимума было разработано много разнооб- разных методов быстрейшего спуска (steepest descent) и сопряженных градиентов (conjugate gradient), которые во многом подобны числовым алгоритмам поиска корня.
430 Часть II. Каталог алгоритмических задач Стоит попробовать несколько разных методов для задачи оптимизации. Прежде чем пытаться реализовать свой собственный метод, поэкспериментируйте с перечисленны-> ми далее реализациями. Четкие описания таких алгоритмов можно найти во многих книгах по числовым алгоритмам, в частности, в книге [PFTV07]. В задаче условной оптимизации часто трудно найти точку, удовлетворяющую всем ограничениям. Один из возможных подходов к решению — использование метода без- условной оптимизации с добавлением штрафных санкций в зависимости от количества нарушенных ограничивающих условий. Определение правильной функции начисления штрафных санкций зависит от конкретной задачи, но часто имеет смысл менять штрафные санкции в ходе процесса оптимизации. В конечном счете штрафные санк- ции должны быть значительными, чтобы обеспечить удовлетворение всех ограничи- вающих условий. Метод имитации отжига является довольно простым и устойчивым к ошибкам подхо- дом к решению задачи условной оптимизации, особенно когда оптимизация выполня- ется по комбинаторным структурам (перестановкам, графам, подмножествам и т. п.). Технология метода имитации отжига описывается в разделе 7.5.3. Реализации. Предмет безусловной и условной оптимизации является достаточно сложным, вследствие чего было издано несколько руководств в помощь разработчи- кам. Лучшим из этих руководств является "Decision Tree for Optimization Software", доступное по адресу http://plato.asu.edu/guide.htinl. Также стоит ознакомиться с руко- водством "Guide to Available Mathematical Software" института NIST, доступным по адресу http://gams.nist.gov. Система NEOS (Network-Enabled Optimization System, оптимизирующая система с под- держкой работы в сетевой среде) предоставляет уникальный сервис— возможность решать задачи дистанционно на компьютерах и программном обеспечении Аргоннской национальной лаборатории в США. Поддерживается как линейное программирование, так и безусловная оптимизация. Вместо решения задачи средствами лаборатории мож- но загрузить программное обеспечение для самостоятельной работы. Подробности см. по адресу http://www-neos.mcs.anl.gov. Коллекция алгоритмов ассоциации АСМ содержит несколько реализаций для безус- ловной оптимизации на языке FORTRAN, из которых особенно заслуживают внимания алгоритм 566 (см. [MGH81]), алгоритм 702 (см. [SF92]) и алгоритм 734 (см. [Вис94]). Алгоритм 744 (см. [Rab95]) из этой же библиотеки реализован на языке Lisp Все эти алгоритмы являются частью библиотеки NetLib (см. раздел 19.1.5). Также доступны реализации общего назначения метода имитации отжига, которые, скорее всего, будут наилучшей отправной точкой для экспериментов с этой технологи- ей для условной оптимизации. Можете свободно пользоваться моим кодом этого мето- да (см. раздел 7.5.3). Особенно популярной реализацией метода имитации отжига явля- ется реализация Adaptive Simulated Annealing на языке С, которую можно загрузить с веб-сайта http://asa-caltech.sourceforge.net/. Как версия Java (http://jgap.sourceforge.net), так и версия С (http://gaul.sourceforge.net) пакета Genetic Algorithm Utility Library предназначена для оказания помощи в разра- ботке приложений на основе генетических и эволюционных алгоритмов. Лично я скеп-
Глава 13. Численные задачи 431 тически отношусь к генетическим алгоритмам (см. раздел 7.8), но многие восхищают- ся ими. Примечания Применение методов быстрейшего спуска для безусловной оптимизации рассматривается в большинстве книг по численным методам, включая [ВТ92] и [PFTV07]. Сама область безусловной оптимизации также является предметом многих книг, включая [Вге73] и [Fle8OJ. Метод имитации отжига был разработан в качестве современной версии алгоритма Мет- рополиса (см. [MRRT53]). В обоих алгоритмах используется метод Монте-Карло для вы- числения минимального энергетического уровня системы. Хорошее описание всех разно- видностей локального поиска, включая метод имитации отжига, представлено в [AL97]. Генетические алгоритмы были разработаны и популяризированы Холландом (Holland); см. [Но175] и [Но192]. Благожелательные описания генетических алгоритмов можно найти в работах [LP02] и [MF00]. Еще одной эвристической процедурой, имеющей преданных последователей, является алгоритм поиска с запретами (tabu search); см. [Glo90]. Родственные задачи. Линейное программирование (см. раздел 13.6), выполнимость (см. раздел 14 10). 13.6. Линейное программирование Вход. Набор 5 из п линейных неравенств с т переменными т S, = V С„Х: >Ь:. \ <1<П 1 U J 1 /=| и функция линейной оптимизации f(X) = Х"'_|С,'Х, Задача. Найти такой набор переменных X', который максимизирует целевую функ- цию/ при этом выполняя все неравенства S (рис. 13.6). ВХОД ВЫХОД Рис. 13.6. Графическое представление задачи линейного программирования Обсуждение. Задачи линейного программирования представляют важнейший тип за- дач из области математической оптимизации и исследования операций.
432 Часть II. Каталог алгоритмических задач Линейное программирование имеет следующие приложения: ♦ распределение ресурсов. В данной задаче требуется вложить определенную сумму, чтобы максимизировать полученную прибыль. Часто возможные опции капитало- вложения. выплаты и расходы можно выразить в виде системы линейных нера- венств таким образом, чтобы максимизировать потенциальный доход при указан- ных ограничивающих условиях. Крупномасштабные задачи линейного программи- рования постоянно возникают в авиакомпаниях и других корпорациях; ♦ аппроксимация решения несовместных систем уравнений. Система т линейных уравнений с п неизвестными х„ l<i<n, называется переопределенной (overdetermined), если т>п. Переопределенные системы часто являются несовме- стными (inconsistent), т. е. не существует набора значений, которые одновременно удовлетворяют всем уравнениям. Чтобы найти набор значений, наиболее подходя- щий для решения уравнений, мы можем заменить каждую переменную х, выраже- нием х', + е, и решить новую систему, минимизируя сумму элементов вектора оши- бок; ♦ алгоритмы на графах. Многие из рассматриваемых в этой книге задач на графах, включая задачу поиска кратчайшего пути, паросочетания в двудольных графах и потоков в сети, можно решить, как частные случаи задачи линейного программиро- вания. Большинство других задач, включая задачу коммивояжера, покрытие множе- ства и задачу о рюкзаке, можно решить посредством целочисленного линейного про- граммирован ия. Стандартным алгоритмом линейного программирования является симплекс-метод. При использовании этого метода каждое ограничивающее условие задачи отсекает часть пространства возможных решений. Наша цель— найти такую точку в оставшем- ся после отсечения пространстве решений, которое максимизирует (или минимизирует) функцию flX). Вращая должным образом пространство решений, оптимальную точку можно всегда сделать наивысшей точкой области. Область (симплекс), полученная в результате пересечения линейных ограничений, является выпуклой, поэтому, если только мы не находимся на самом верху, всегда существует соседняя вершина, распо- ложенная выше. Если мы не можем найти более высокую соседнюю точку, значит, мы нашли оптимальное решение. В то время как симплексный алгоритм не слишком сложен, создание его эффективной реализации, способной решать задачи линейного программирования на больших вход- ных экземплярах, требует значительного мастерства. Входные экземпляры значитель- ных размеров, как правило, обладают разреженностью (т. е. во многих неравенствах имеется малое количество переменных), что влечет за собой необходимость использо- вания сложных структур данных. Необходимо помнить о важности вопросов вычисли- тельной устойчивости и надежности, а также выбора следующей точки для посещения (так называемые правила выбора переменных — pivot rules). Кроме этого, существуют сложные методы внутренней точки (interior-point methods), в которых переход к сле- дующей точке осуществляется через внутреннюю область симплекса, а не по поверх- ности. Во многих приложениях эти алгоритмы превосходят алгоритмы симплекс- метода. Подводя итоги, можно сказать о линейном программировании следующее: будет на- много выгоднее использовать существующие реализации инструментов линейного
Глава 13 Численные задачи 433 программирования, нежели пытаться создать свою собственную программу. Кроме того, надежнее приобрести платное решение, чем искать бесплатное в Интернете За- дача линейного программирования настолько важна в экономических приложениях, что коммерческие реализации превосходят бесплатные во всех отношениях. Перечислим вопросы, возникающие при решении задач линейного программирования. ♦ Ограничены ли какие-либо переменные целочисленными значениями? Даже если со- гласно вашей модели максимальную прибыль можно получить, отправляя 6,54 рей- сов из Нью-Йорка в Вашингтон ежедневно, реализовать это на практике невозмож- но. Такие переменные, как количество авиарейсов в приведенном примере, часто обладают естественными целочисленными ограничениями. Задача линейного про- граммирования называется целочисленной, если все ее переменные обладают цело- численными ограничениями, и смешанной, если такие ограничения наложены не на все переменные, а только на некоторые из них. К сожалению, поиск оптимального решения целочисленной или смешанной задачи является NP-полной задачей. Но существуют методы целочисленного программи- рования, которые работают достаточно хорошо на практике. Например, методы се- кущей плоскости сначала решают задачу как линейную, после чего добавляют до- полнительные ограничивающие условия, чтобы наложить требование целочислен- ности в окрестности оптимального решения, и затем решают задачу повторно. После достаточного количества итераций оптимальная точка получившейся задачи линейного программирования совпадает с оптимальной точкой первоначальной це- лочисленной задачи. Как и в большинстве экспоненциальных алгоритмов, время исполнения алгоритмов целочисленного программирования зависит от сложности конкретного экземпляра задачи, поэтому предсказать его невозможно. ♦ Чего в задаче больше— переменных или ограничивающих условий? Любую задачу линейного программирования с т переменными и п неравенствами можно записать в виде эквивалентной двойственной задачи с п переменными и т неравенствами. Важно знать это обстоятельство, т. к. между временами исполнения этих двух вари- антов процедуры решения может быть значительная разница. По большому счету, задачи линейного программирования, в которых переменных намного больше, чем ограничивающих условий, следует решать прямым способом. Если же количество ограничивающих условий намного превышает количество переменных, то обычно лучше решать двойственную задачу линейного программирования или (что равно- сильно) применять двойственный симплекс-метод к первоначальной задаче. ♦ Как поступить, если функция оптимизации или ограничивающие условия не явля- ются линейными? Задача аппроксимации кривой методом наименьших квадратов заключается в поиске прямой, для которой сумма квадратов расстояний между каж- дой заданной точкой и ею минимальна. При постановке этой задачи естественная целевая функция является квадратичной, а не линейной. Хотя для аппроксимации кривой методом наименьших квадратов существуют быстрые алгоритмы, общая за- дача квадратичного программирования является NP-полной. Существуют три возможных подхода к решению задачи нелинейного программиро- вания. Самый лучший — смоделировать ее (если это возможно) каким-либо другим способом, как в задаче аппроксимации кривой методом наименьших квадратов.
434 Часть II. Каталог алгоритмических задач Также можно попытаться найти специальные реализации для квадратичного про- граммирования. Наконец, можно смоделировать задачу в виде задачи условной или безусловной оптимизации и попытаться решить ее с помощью реализаций, рассмат- риваемых в разделе 13.5. ♦ Что делать, если моя модель не совпадает с форматом входа имеющейся у меня реализации решения задачи линейного программирования? Многие реализации ре- шений задач линейного программирования принимают модели только в так назы- ваемом стандартном формате, где все переменные должны быть неотрицательными, целевая функция должна быть минимизирована, а все ограничивающие условия должны быть сформулированы в виде равенств (а не неравенств). В этом требовании нет ничего страшного. Известны преобразования для приведе- ния произвольных задач линейного программирования к стандартному формату. В частности, чтобы преобразовать задачу максимизации в задачу минимизации, нужно просто поменять знак коэффициента целевой функции на противоположный. Остальные несоответствия можно исправить, добавляя в модель фиктивные пере- менные (slack variable). Подробности можно найти в любом учебнике по линейному программированию. Кроме того, вопрос можно решить посредством добавления хорошего интерфейса к программной реализации, воспользовавшись каким-либо языком моделирования, например языком АМРС. Реализации. Очень полезным ресурсом по решению задач линейного программирова- ния является раздел часто задаваемых вопросов (FAQ) Usenet. В частности, здесь мож- но найти список имеющихся в наличии реализаций с описанием впечатлений от их применения. Дополнительную информацию см. на веб-сайте http://www-unix.mcs. anl.gov/otc/Guide/faq/. Существуют, по крайней мере, три хорошие бесплатные реализации для решения задач линейного программирования. Приложение lp_solve, написанное Майклом Беркелаа- ром (Michel Berkelaar) на языке ANSI С. работает как с целочисленными, так и со смешанными задачами. Загрузить приложение можно с веб-сайта http:// lpsolve.sourceforgc.net/5.5/. Также существует огромное сообщество пользователей этого приложения. Другая библиотека. CLP. для решения задач линейного программи- рования симплекс-методом. предоставляется проектом Computational Infrastructure for Operations Research и доступна по адресу http://www.coin-or.org/. Наконец, пакет GLPK (GNU Linear Programming Kit) предназначен для решения крупномасштабных задач линейного программирования, целочисленного программирования и других род- ственных задач. Этот пакет можно загрузить с веб-сайта http://www.gnu.org/ software/glpk/. В последних тестах открытого кода (http://plato.asu.edu/bench.html) библиотека CLP была самой быстрой при решении задач линейного программирова- ния, а библиотека Ip solve — на смешанных задачах. Система NEOS предоставляет возможность решать задачи дистанционно на компьюте- рах и программном обеспечении Аргоннской национальной лаборатории (Argonne National Laboratory). Поддерживается как линейное программирование, так и безуслов- ная оптимизация. На .эту систему стоит обратить внимание, если вам нужна не про- грамма для решения задачи, а только ответ на задачу. Дополнительную информацию см. на веб-сайте http://www.mcs.anl.gov/honie/otc/Server/.
Глава 13. Численные задачи 435 Алгоритмы 551 (см. [Abd8O]) и 552 (см. [BR80]) симплекс-метода на языке FORTRAN из коллекции алгоритмов ассоциации АСМ предназначены для решения переопреде- ленных систем линейных уравнений. Подробности см. в разделе 19.1.5. Примечания Потребность в оптимизации посредством линейного программирования возникла при ре- шении задач материально-технического снабжения во время Второй мировой войны. Симплексный алгоритм был изобретен Джорджем Данцигом (George Danzig) в 1947 г. (см. [Dan63]). А Кли (Klee) и Минти (Minty) доказали, что симплексный алгоритм имеет экспоненциальное время исполнения в наихудшем случае, но является очень эффектив- ным на практике; см. [КМ72] . При сглаживающем анализе измеряется сложность алгоритмов в предположении, что их вход содержит небольшой объем помех. Аккуратно построенные экземпляры наихудших случаев для многих задач не выдерживают такую проверку. Эффективность симплекс- метода в практическом применении была объяснена с помощью сглаживающего анализа Даниелем Шпильманом (Daniel Spielman) и Шанг-Хуа Тенгом (Shang-Hua Teng); см. [ST04]. А недавно был разработан рандомизированный симплексный алгоритм с полино- миальным временем исполнения (см. [KS05b]). Полиномиальность задач линейного программирования была впервые доказана в 1979 г. посредством эллиптического алгоритма (см. [Kha79]). Алгоритм Кармаркара (Karmarkar) на основе метола внутренней точки (см. [Каг84]) является как теоретическим, так и прак- тическим улучшением эллиптического алгоритма, а также соперником симплексного метода. Хорошие описания симплексного и эллиптического алгоритмов приведены в [Chv83], [Gas03] и [MG06]. Полуопределенное программирование применяется для решения задач оптимизации с пе- ременными симметрических положительных полуопределенных матриц и линейной функцией стоимости и линейными ограничивающими условиями. Важные частные случаи включают в себя задачи линейного программирования и задачи выпуклого квадратичного программирования с выпуклыми квадратичными ограничивающими условиями Полуоп- ределенное программирование и его использование для решения комбинаторных задач оптимизации обсуждается в публикациях [Goe97] и [VB96]. Задачи линейного программирования являются P-полными с логарифмической слож- ностью по памяти (см. [DLR79]). Вследствие этого, создать параллельный алгоритм для решения задач класса NC, скорее всего, невозможно. (Задача принадлежит классу NC тогда и только тогда, когда ее можно решить на машине PRAM за полилогарифмическое время, используя полиномиальное количество процессоров.) Любая P-полная задача по сведению с логарифмической сложностью по памяти не может быть членом класса NC, если только не выполняется условие Р = NC. Всестороннее рассмотрение теории P-полноты, вклю- чающее в себя обширный список P-полных задач, представлено в книге [GHR95], Родственные задачи. Условная и безусловная оптимизация (см. раздел 13.5), потоки в сети (см. раздел 15.9). 13.7. Генерирование случайных чисел Вход. Либо ничего, либо начальное число (seed). Задача. Сгенерировать последовательность случайных целых чисел (рис. I3.7).
436 Часть II Каталог алгоритмических задач нтнтннтннт НННТТННТТТ ннттттнтнт нтнннттннт нтгннггнтн вход выход Рис. 13.7. Последовательность случайных символов Обсуждение. Случайные числа используются в самых разных интересных и важных приложениях. Они лежат в основе метода имитации отжига и родственных эвристиче- ских методов оптимизации. Эмуляция дискретных событий выполняется на потоках случайных чисел и используется для моделирования широкого диапазона задач, от транспортных систем до игры в покер. Пароли и ключи шифрования обычно генери- руются случайным образом. Рандомизированные алгоритмы для решения задач на графах и геометрических задач коренным образом изменяют эти области и возводят рандомизацию в ранг одного из основополагающих принципов теории вычислитель- ных систем. К сожалению, задача генерирования случайных чисел выглядит гораздо более легкой, чем она в действительности является. Более того, создать истинно случайное число на любом детерминистическом устройстве, по сути, невозможно. Лучше всего это выра- зил фон Ньюман (см. [Neu63]): "Любой, кто считает возможным применение арифме- тических методов для генерирования случайных чисел, несомненно, грешит” Самое лучшее, что мы можем надеяться получить с помощью арифметических методов, это псевдослучайные числа, т. е. последовательность чисел, которые выглядят случай- ными. Отсюда вытекают серьезные последствия использования некачественного генератора случайных чисел. В одном получившем известность случае схема шифрования веб- браузера была взломана, т. к. в нем использовалось недостаточно много случайных битов (см. [GW96]). Результаты моделирования постоянно получаются неточными или даже полностью неверными вследствие использования некачественной функции гене- рирования случайных чисел. Это такая область, в которой не следует заниматься само- деятельностью. но многие обычно переоценивают свои способности. Перечислим вопросы, касающиеся работы со случайными числами. ♦ Можно ли использовать один и тот же набор "случайных" чисел при каждом ис- полнении программы? Игра в покер, в которой вам каждый раз сдаются одни и те же карты, быстро надоест. Одним из распространенных решений этой задачи явля- ется использование младших битов показаний часов компьютера в качестве началь- ного числа (seed) потока случайных чисел, чтобы при каждом исполнении програм- мы генерировалась другая последовательность. Такие методы удовлетворительны для игр, но не для моделирования серьезных си- туаций. Всякий раз, когда вызовы функции генерирования случайных чисел выпол- няются в цикле, существуез вероятность периодичности распределения случайных
Глава 13 Численные задачи 437 чисел. С другой стороны, когда программа выдает разные результаты при каждом ее исполнении, ее отладка серьезно усложняется. В случае сбоя программы вы не сможете отследить ее исполнение, чтобы выяснить причину сбоя. Возможным ком- промиссом будет использование детерминистического генератора псевдослучайных чисел с сохранением текущего начального числа в файле между исполнениями про- граммы. При отладке в этот файл можно будет записывать фиксированное началь- ное число. ♦ Насколько надежен встроенный генератор случайных чисел моего компилятора? Если вам требуются равномерно распределенные случайные числа и требования к точности моделирования не являются критическими, то я рекомендую просто ис- пользовать генератор, который встроен в компилятор. Здесь очень легко совершить ошибку, неудачно выбрав начальное число, поэтому вы должны внимательно про- читать руководство пользователя. Но если точность результатов моделирования является критической, то лучше ис- пользовать собственный генератор случайных чисел. Но следует иметь в виду, что определить на глазок, действительно ли генерируемые последовательности являют- ся случайными, очень трудно, потому что у людей имеется искаженное представле- ние о поведении генераторов случайных чисел, и они часто видят закономерности, которые в действительности не существуют. Качество генератора случайных чисел следует проверить на нескольких разных тестах и установить статистическую дос- товерность результатов. Для оценки качества генераторов случайных чисел Нацио- нальный институт стандартов и технологий США разработал специальный набор тестов, который рассматривается в подразделе "Реализации". ♦ Как реализовать собственный генератор случайных чисел? Стандартным вариан- том реализации генератора случайных чисел является линейный конгруэнтный гене- ратор (linear congruential generator). Это простой, быстрый генератор, который дает достаточно удовлетворительные псевдослучайные числа (при условии установления правильных значений констант). Случайное число R,, является функцией (п- 1)-го сгенерированного случайного числа: R„ = (а7?„_| + с) mod т В теории поведение линейного конгруэнтного генератора аналогично поведению рулетки. Длинный путь шарика вдоль окружности колеса (определяемый выраже- нием aRn । + с) заканчивается в одной из немногочисленных лунок. Этот "выбор" весьма чувствителен к длине пути (усеченной с помощью выражения mod /и). Для выбора значений констант а, с, т и /?0 была разработана специальная теория. Длина периода в значительной степени является функцией от т, значение которого обычно ограничено длиной слова конкретного компьютера. Обратите внимание на то, что последовательность чисел, выдаваемая линейным конгруэнтным генератором, начинает повторяться, как только повторяется первое число. Кроме этого, современные компьютеры обладают достаточно высокой ско- ростью, позволяющей им вызывать функцию генератора 232 раз за несколько минут. Таким образом, для любого линейного конгруэнтного генератора на компьютере с длиной слова в 32 бита существует опасность циклического повторения значений.
438•Часть II. Каталог алгоритмических задач что диктует необходимость в генераторах, имеющих значительно более длинные периоды. ♦ Что делать, если распределение генерируемых случайных чисел должно быть не- равномерным? Генерирование случайных чисел согласно данной функции неравно- мерного распределения может быть нелегкой задачей. Самым надежным будет ме- тод выборки с отклонениями (acceptance-rejection method). Геометрическая область, из которой требуется делать выборку, заключается в ограничивающее окно, а потом выбирается произвольная точка р. Это точку можно получить, независимо генери- руя произвольные значения ее координатх и у. Если полученная точка лежит в ин- тересующей нас области, то мы принимаем ее как случайно выбранную. В против- ном случае мы отказываемся от точки и процесс повторяется. Можно сказать, что мы бросаем дротики с закрытыми глазами и засчитываем только те. которые попа- дают в цель. В то время как этот метод возвращает правильные результаты, его производитель- ность может быть неудовлетворительной. Если по сравнению с ограничивающим окном площадь представляющей интерес области небольшая, то большинство на- ших дротиков не будут попадать в цель. Описание эффективных алгоритмов для других специальных распределений, включая гауссово распределение, приводится в подразделе "Реализации". Но будьте осторожны с изобретением своих собственных методов, т. к. получить правильное распределение вероятностей очень трудно. Например, генерирование полярных координат посредством случайного выбора с равномерным распределе- нием угла межд> 0 и 2тг и смещением от 0 до г будет неправильным способом выбо- ра равномерно распределенных точек из круга радиусом г. В данном случае поло- вина сгенерированных точек будет располагаться на расстоянии /72 от центра, в то время как здесь должна находиться только одна четвертая часть их количества! Эта разница достаточно большая, чтобы серьезно исказить результат, и в то же самое время довольно тонкая, что ее легко не заметить. ♦ Сколько времени нужно выполнять моделирование по методу Монте-Карло, чтобы получить наилучшие результаты? Чем больше времени выполняется программа моделирования, тем точнее будет аппроксимация предельного распределения. Но это происходит только до тех пор. пока не будет превышен период (длина цикла) генератора случайных чисел, после чего последовательность случайных чисел нач- нет повторяться, и дальнейшее исполнение не даст никакой дополнительной ин- формации. Вместо того чтобы устанавливать максимальное значение периода, стоит выполнить программу моделирования с более коротким периодом, но несколько раз (от 10 до 100) и с разными начальными числами, а потом рассмотреть полученный диапазон резуль- татов. Разброс значений даст вам хорошее представление о степени повторяемости по- лученных результатов. Таким образом вы сможете избавиться от заблуждения, будто моделирование дает "правильный" ответ. Реализации. Отличный материал по генерированию случайных чисел и стохастиче- скому моделированию можно найти на веб-сайте http://random.mat.sbg.ac.at. Там даются ссылки на научные работы и множество реализаций генераторов случайных чисел.
Глава 13 Численные задачи 439 Параллельное моделирование предъявляет особые требования к генераторам случай- ных чисел. Например, как можно гарантировать независимость случайных потоков на каждой машине? Одно из решений— воспользоваться объектно-ориентированными генераторами, описанными в работе [LSCK02], с длиной периода около 21 . Реали- зации этих генераторов на языках С, C++ и Java можно загрузить с веб-сайта http://www.iro.umontreal.ca/~lecuyer/myftp/streanisOO. Для работы на многопроцес- сорных компьютерах поддерживается генерирование независимых потоков случайных чисел. Другой подход— использовать библиотеку SPRNG (см. [MS00]), которую мож- но загрузить с веб-сайта http://sprng.cs.fsu.edu. Алгоритмы 488 (см. [Вге74]). 599 (см. [AK.D83]) и 712 (см. [Lev92]) на языке FORTRAN из коллекции алгоритмов ассоциации АСМ предназначены для генерирова- ния случайных чисел с неравномерным распределением согласно нескольким методам распределения вероятностей, включая нормальное, экспоненциальное и пуассоновское. Все эти алгоритмы являются частью библиотеки NetLib (см. раздел 19.1.5). В Национальном институте стандартов США был разработан широкий набор статисти- ческих тестов для проверки генераторов случайных чисел (см. [RSN 01]). Это про- граммное обеспечение и описывающий его доклад можно загрузить с веб-сайта http://csrc.nist.gov/rng. Генераторы действительно случайных чисел используют численные представления какого-либо физического процесса. Услуги по предоставлению случайных чисел, сге- нерированных таким образом, предоставляются на веб-сайте http://www.random.org. Эти числа генерируются на основе атмосферных шумов и проходят проверку статисти- ческими тестами Национального инстигута стандартов и технологий США. Это удач- ное решение, если вам требуется небольшое количество действительно случайных чи- сел. например, для проведения лотереи, а не сам генератор таких чисел. Примечания Генерирование случайных чисел очень подробно описано в книге Дональда Кнута [Кпв97Ь]. В его книге излагаются теоретические основы нескольких методов генерирова- ния случайных чисел, включая метод серединных квадратов и регистр сдвига, которые не были рассмотрены здесь. Кроме того, подробно обсуждаются статистические тесты для проверки генераторов случайных чисел. Информация о последних разработках в области генерирования случайных чисел собрана в книге [Gen04]. Генератор случайных чисел, называемый вихрем Мерсеина (Mersenne twister; см. [MN98]), имеет период длиной в 219937 — 1. Другие современные методы гене- рирования случайных чисел основаны на рекурсии (см. [DenO5] и [PL.M06]). Обзор мето- дов генерирования случайных чисел с неравномерным распределением представлен в книге [HLD04], а в книге [РМ88] — сравнение практического применения разных гене- раторов случайных чисел. В прежние времена, когда компьютеры не были так распространены, в большинстве ма- тематических учебников печатались таблицы случайных чисел. Наиболее известным явля- ется [RC55], в котором приводится один миллион случайных чисел. Глубинная взаимосвязь между случайностью, информацией и сжимаемостью исследуется в теории колмогоровской сложности, согласно которой о сложности строки можно судить по ее сжимаемости. Строки истинно случайных символов несжимаемы. В соответствии с этой теорией кажущиеся случайными цифры значения л не могут быть случайными, т. к.
440 Часть II. Каталог алгоритмических задач вся последовательность определяется любой программой, реализующей разложение в ряд для л. Всестороннее введение в теорию колмогоровской сложности можно найти в кни- ге [LV97]. Родственные задачи. Условная и безусловная оптимизация (см. раздел 13.5), генери- рование перестановок (см. раздел 14 4), генерирование подмножеств (см. раздел 14 5). генерирование разбиений (см. раздел 14.6). 13.8. Разложение на множители и проверка чисел на простоту Вход. Целое число п. Задача. Определить, является ли целое число п простым, а если нет, то найти его мно- жители (рис. 13.8). 8338169264555846052842102071 ВХОД 179424673 2038074743 * 22801763489 8338169264555846052842102071 ВЫХОД Рис. 13.8. Разложение на множители Обсуждение. Двойственные друг другу задачи проверки целого числа на простоту и разложения его на множители имеют неожиданно много приложений, если учесть, что долгое время они считались представляющими только математический интерес. В ча- стности, безопасность криптографической системы с открытым ключом RSA (см. раз- дел 18.6) основана на вычислительной неосуществимости решения задачи разложения на множители больших целых чисел. Более скромным приложением задачи проверки целого числа на простоту является улучшение производительности хэш-таблиц путем выбора для них размера, равного простому целому числу. Для этого процедура ини- циализации хэш-таблицы должна определить простое целое число, близкое к требуе- мому размеру хэш-таблицы. Наконец, с простыми числами просто интересно экспери- ментировать. Совсем не случайно, что на UNIX-системах программы для генерирова- ния больших простых чисел часто находятся в папке для игр. Хотя задача разложения на множители и задача проверки числа на простоту значи- тельно отличаются в алгоритмическом отношении, они являются родственными. Су- ществуют алгоритмы, которые могут показать, что целое число является составным (т. е. не простым), не предоставляя при этом самих множителей. Чтобы убедиться в наличии таких алгоритмов, обратите внимание на тот факт, что можно продемонстри- ровать составную природу любого нетривиального целого числа, заканчивающегося на 0, 2, 4, 5, 6 или 8, не выполняя для этого самой операции деления.
Глава 13. Численные задачи 441 Самым простым алгоритмом для решения обеих этих задач является проверка делени- ем. Для этого выполняется деление пН для всех I <z <yjn. Полученные при этом про- стые множители числа и будут содержать, по крайней мере, один экземпляр такого де- лителя z, для которого и/ i = |_zz/zj, если, конечно, zz не является простым числом. Но при этом необходимо обеспечить корректную обработку повторяющихся множителей, а также учесть все простые числа, большие у/п . Работу таких алгоритмов можно ускорить, используя таблицы заранее вычисленных простых чисел, чтобы не проверять все возможные значения /. Применение битовых векторов (см. раздел 12.5) позволяет представить удивительно большое количество простых чисел в неожиданно малом объеме памяти. Для хранения битового вектора, содержащего все нечетные числа до миллиона, требуется меньше, чем 64 Кбайт. Еще более плотную упаковку можно получить, удалив все числа, кратные трем, и другим небольшим простым числам. Хотя алгоритм проверки делением исполняется за время Оу/п , он не является полино- миальным алгоритмом по той причине, что для представления числа и требуется толь- ко Igy? битов, поэтому время исполнения алгоритма делением экспоненциально зави- сит от размера входа. Существуют значительно более быстрые (но с тем же экспонен- циальным временем исполнения) алгоритмы, правильность которых основана на теории чисел. Самый быстрый алгоритм, называющийся решетом числового поля (number field sieve), использует случайную последовательность для создания системы конгруэнций, решение которой обычно дает делитель целого числа. С помощью этого метода было выполнено разложение на множители целых чисел длиной в 200 цифр (663 бита), хотя для такой работы потребовалось выполнить громаднейший объем вы- числений. Проверку целых чисел на простоту значительно легче выполнять с помощью рандоми- зированных алгоритмов. Малая теорема Ферма утверждает, что a -l=1(mod zz) для всех zz, не делящихся на zz, при условии, что п является простым числом. Возьмем слу- чайное значение 1 <а<п и вычислим остаток a" '(mod zz). Если этот остаток не равен 1. то это доказывает, что zz не может быть простым числом. Такие рандомизированные проверки чисел на простоту очень эффективны. С помощью таких проверок программа шифрования PGP (см. раздел 18.6) за несколько минут находит простые числа длиной в триста с лишним цифр для использования в качестве ключей шифрования. Хотя кажется, что простые числа разбросаны среди целых чисел в случайном порядке, их распределение имеет определенную регулярность. Теорема простых чисел утвер- ждает, что количество простых чисел, меньших, чем zz (обычно обозначающееся как л(п)), приблизительно равно zz/lnzz. Кроме этого, между простыми числами никогда нет больших промежутков, поэтому, вообще говоря, можно ожидать, что для того, чтобы найти первое простое число, большее, чем zz, нужно будет проверить Inzz целых чисел. Такое распределение и наличие быстрого рандомизированного теста на простоту объ- ясняет, почему PGP находит большие простые числа с такой скоростью. Реализации. Существует несколько систем общего назначения для решения задач из вычислительной теории чисел. Система компьютерной алгебры PARI может решать сложные задачи теории чисел с целыми числами произвольной точности (чтобы быть
442 Часть II. Каталог алгоритмических задач точным, ограниченными длиной в 80 807 123 цифр на 32-разрядных машинах), а также с действительными, рациональными, сложными и алгебраическими числами и матри- цами. Система написана, в основном, на языке С, а внутренние циклы для основных компью- терных архитектур— на ассемблере, и содержит свыше 200 специализированных ма- тематических функций. Систему PARI можно использовать в виде библиотеки, но она также имеет режим калькулятора, предоставляющий немедленный доступ ко всем ти- пам и функциям. Загрузить систему можно с веб-сайта http://pari.math.u-bordeaux.fr/. Библиотека LiDlA на языке C++ (http://www.cdc.informatik.tu-darmstadt.de/ TI/LiDIA/) реализует несколько современных методов разложения целых чисел на множители. Высокопроизводительная переносимая библиотека NTL на языке C++ предоставляет структуры данных и алгоритмы для манипулирования целыми числами произвольной длины со знаком, а также для векторов, матриц и многочленов над полем целых чисел и над конечными полями. Библиотеку NTL можно загрузить с веб-сайта http:// www.shoup.net/ntl/. Наконец, библиотека MIRACL, написанная на C/C++, реализует шесть разных алго- ритмов для разложения целых чисел на множители, включая алгоритм квадратичного решета. Библиотеку MIRACL можно загрузить с веб-сайта http://www.shamus.ie/. Примечания Книги [СР05] и [Yan03] содержат описание современных алгоритмов проверки целых чи- сел на простоту и разложения на множители. Более общие сведения по вычислительной теории чисел можно найти в книгах [BS96] и [Sho05]. В 2002 г. Агравал (Agrawal). Каял (КауаГ) и Саксена (Saxena) предоставили первый детер- министический алгоритм с полиномиальным временем исполнения, проверяющий, явля- ется ли данное целое число составным (см. [AKS04]). При разработке этого незамыслова- того алгоритма они выполнили тщательный анализ известных рандомизированных алго- ритмов. Существование этого алгоритма является укором исследователям (таким как я), которые побаиваются заниматься классическими нерешенными задачами. Независимая трактовка этого результата приведена в книге [Die04], Рандомизированный алгоритм Миллера-Рабина (Miller-Rabin algorithm) для проверки (см. [Mil76], [Rab80]) целых чисел на простоту решает проблему чисел Кармайкла (Carmichael numbers), которые являются составными целыми числами, удовлетворяющи- ми условиям теоремы Ферма. Самые лучшие алгоритмы для разложения целых чисел на множители используют метод квадратичного решета (см. [Poni84]) и метод эллиптиче- ской кривой (см. [Len87b]). Механические вычислительные устройства предоставляли самый быстрый способ разло- жения целых чисел на множители задолго до наступления компьютерной эры. Занима- тельная история одного из таких устройств, созданного во время Первой мировой войны, изложена в работе [SWM95]. Приводимое в действие вручную вращением рукоятки уст- ройство доказывало простоту числа 231 - 1 за 15 минут работы. Важной задачей теории вычислительной сложности является определение истинности выражения Р = NPn co-NP. Задача разрешимости "является ли п составным числом?" долгое время была наилучшим контрпримером. Предоставляя разложение числа п на множители, она очевидным образом является членом класса NP. Можно доказать, что за-
Гпава 13. Численные задачи 443 дача является членом класса co-NP, т. к. для каждого простого числа имеется короткое доказательство его простоты (см. [Рга75]). Но недавнее доказательство, что задача про- верки составных чисел является членом класса Р, делает эту цепочку рассуждений недей- ствительной. Дополнительную информацию о классах сложности можно найти в [GJ79] и [Joh90]. Число из задачи RSA-129 было разложено на множители в апреле 1994 г. после восьми месяцев вычислений на 1 600 компьютерах. Это событие было особенно примечательным ввиду того обстоятельства, что в исходной работе RSA (см. [RSA78]) предполагалось, что решение этой задачи займет 40 квадрильонов лет (используя технологию 1970-х годов). Текущий рекорд по разложению на множители принадлежит Ф. Бару, М. Боэму, Дж. Франку и Т. Клайнджунгу (F. Bahr, М. Boehm, J. Franke, Т. Kleinjung), которые в мае 2005 г. разложили на множители число из задачи RSA-200. Для этого им потребовалось время, эквивалентное 55 годам вычислений на компьютере с одним процессором Opteron 2,2 ГГц. Родственные задачи. Криптография (см. раздел 18.6), арифметические операции вы- сокой точности (см. раздел 13.9). 13.9. Арифметика произвольной точности Вход. Два очень больших целых числах и у. Задача. Вычислить х + у. х-у, х х у и х/у (рис. 13.9). 49578291287491495151508905425869578 74367436931237242727263358138804367 ВХОД ВЫХОД Рис. 13.9. Деление очень больших целых чисел Обсуждение. Любой язык программирования, имеющий уровень выше ассемблера, поддерживает арифметические операции сложения, вычитания, умножения и деления с целыми и действительными числами с одинарной и, возможно, двойной точностью. А если мы захотим выразить государственный долг США в центах? Для выражения количества центов стоимостью в один триллион долларов потребуется число из 15 де- сятичных цифр, что намного больше, чем может вместиться в 32-битовое компьютер- ное слово. В других приложениях могут возникнуть намного большие целые числа. Например, для обеспечения достаточной безопасности в алгоритме RSA для криптографии с от- крытым ключом рекомендуется использовать целочисленные ключи длиной не менее 1 000 цифр. Исследовательские эксперименты в области теории чисел требуют выпол- нения операций с большими числами. Однажды я решил незначительную нерешенную задачу (см. [GKP89]), выполнив точные вычисления, результат которых представлен целым числом (2953) ~ 9,93285х 101775.
444 Часть II. Каталог алгоритмических задач Прежде чем приступить к работе с очень большими целыми числами, ответьте на пере- численные ниже вопросы. ♦ Я решаю экземпляр задачи, требующей больших целых чисел, шт имею дело со встроенным приложением? Если вам просто нужен ответ для конкретной задачи с большими целыми числами, как в моем случае с нерешенной задачей, описанном выше, то следует рассмотреть использование системы компьютерной алгебры, та- кой как Maple или Mathematica. Эти системы по умолчанию предоставляют воз- можности арифметических вычислений произвольной точности и используют для интерфейса удобные языки программирования наподобие языка Lisp. В таком слу- чае решение вашей задачи будет заключаться в программе из 5-10 строчек. Если же вы работаете со встроенным приложением, требующим выполнения ариф- метических операций с высокой точностью, то следует использовать какую-либо математическую библиотеку, поддерживающую вычисления с произвольной точ- ностью. Эти библиотеки, в дополнение к четырем основным операциям, часто пре- доставляют дополнительные функции для таких вычислений, как наибольший об- щий делитель. Подробности см. в подразделе "Реализации". ♦ Какой уровень точности требуется? Иными словами, имеется ли верхняя граница для чисел, с которыми вы работаете, или же требуется произвольная точность пред- ставления. От этого зависит, сможете ли вы использовать для представления целых чисел массив фиксированной длины или для этого потребуется применение связных списков. Массив проще для реализации и работы, и в большинстве приложений он не создает никаких ограничений. ♦ Какое основание следует использовать? Возможно, что легче всего будет реализо- вать свой собственный пакет для выполнения арифметических операций с высокой точностью в десятичной системе, таким образом, представляя каждое целое число в виде строки цифр с основанием 10. Но гораздо эффективнее использовать в качест- ве основания большое число, в идеале равное квадратному корню из наибольшего целого числа, поддерживаемого аппаратно. Почему? Потому, что чем больше основание, тем меньше требуется цифр для пред- ставления чисел. Например, сравните количество цифр в одном и том же значении, но представленном в десятичной и двоичной системах счисления — 64 и 1 000 000 соответственно. Так как аппаратная операция сложения обычно выполняется за один тактовый цикл независимо от фактических значений чисел, то наилучшая про- изводительность достигается при использовании максимально возможного основа- ния. Верхняя граница основания определяется значением b = Vmaxint , что позво- ляет избежать арифметического переполнения при умножении двух таких чисел. Основной сложностью использования большого основания является необходимость преобразовывать целые числа в представление с основанием 10 для входа и выхода. Но это преобразование легко выполняется, когда поддерживаются все четыре арифметические операции с высокой точностью. ♦ Какой точности будет достаточно? Сложение аппаратными средствами выпол- няется намного быстрее, чем программными, поэтому использование арифметиче- ских операций с высокой точностью в ситуациях, когда такая точность не требуется.
Гпава 13. Численные задачи 445 значительно снижает производительность. Арифметические вычисления с высокой точностью относятся к тем немногочисленным задачам, для которых реализация внутренних циклов на ассемблере оказывается хорошим способом ускорения рабо- ты программы. Подобным образом, наложение маски на уровне битов и использо- вание операций сдвига вместо арифметических операций может ускорить выполне- ние программы; но для этого вы должны хорошо разбираться в особенностях ма- шинного представления целых чисел. Далее приводятся оптимальные алгоритмы для каждой из пяти основных арифметиче- ских операций: ♦ сложение. Простой школьный способ сложения, при котором числа выравниваются по десятичному разделителю, после чего их цифры складываются с переносом справа налево, имеет время исполнения, линейно зависящее от количества цифр. Существуют более сложные параллельные алгоритмы с предсказанием переноса (carry-look-ahead) для реализации низкоуровневого машинного сложения. Скорее всего, они используются в вашем микропроцессоре для выполнения операций сло- жения с низкой точностью; ♦ вычитание. Изменяя знаковые биты, мы можем превратить операцию вычитания в специальный случай сложения: (А -(В)) = (А + В). Сложной частью операции вы- читания является "заем" из вышестоящего разряда. Для упрощения можно всегда выполнять вычитание из числа с большим абсолютным значением и корректировать знаки позже; ♦ умножение. На больших целых числах выполнение операции умножения повто- ряющимися операциями сложения занимает экспоненциальное время, поэтому из- бегайте использования этого метода. Поцифровое умножение школьным методом легко поддается программированию и дает гораздо лучшую производительность, которая, скорее всего, будет достаточной для вашего приложения. Для очень боль- ших целых чисел рекомендуется алгоритм Карацубы (Karatsuba's algorithm) типа "разделяй и властвуй" с временем исполнения О(и!’5 ). Дэн Грейсон (Dan Grayson), разработчик арифметических операций произвольной точности системы компью- терной алгебры Mathematica, обнаружил, что точка перехода к этому алгоритму на- ходится среди чисел, имеющих гораздо менее ста цифр. Еще более быстрым является алгоритм на основе преобразований Фурье. Такие ал- горитмы рассматриваются в разделе 13.1 Г, ♦ деление. Как и в случае с умножением, выполнение деления посредством много- кратного вычитания занимает экспоненциальное время; поэтому наиболее прием- лемым алгоритмом будет "деление в столбик", которому нас учат в школе. Это до- вольно сложный алгоритм, требующий операции умножения с произвольной точ- ностью и операции вычитания в качестве вызываемых процедур, а также определе- ния правильной цифры в каждой позиции частного методом проб и ошибок. Операцию деления целых чисел можно свести к операции умножения, хотя такое сведение далеко не тривиально. Так что, если вы разрабатываете асимптотически быстрое умножение, то результат можно будет использовать и для реализации опе- рации деления;
446 Часть II. Каталог алгоритмических задач ♦ возведение в степень. Значение а'1 можно вычислить, выполнив и—1 умножений числа а на само себя, т. е. а х а х ... х а. Но существует намного лучший алгоритм типа "разделяй и властвуй", основанный на том обстоятельстве, что п = п/2J + п/2\. Если число п четное, то тогда а" = (апГ1^. А если п нечетное, то тогда и2 = «(бД"/2-1)2. В любом случае, размер показателя степени был уменьшен наполовину, что обош- лось нам. самое большее, в две операции умножения. Таким образом, для вычисле- ния конечного значения будет достаточно O(lgw) операций умножения. Псевдокод соответствующего алгоритма показан в листинге 13.2. Листинг 13.2. Алгоритм возведения в степень function power(а, п; if (п - 0) return(1) = power ( a, 1 п/2 | ) if (n is even) then return(n2) eise return (a Арифметические операции высокой, но не произвольной точности, удобно выполнять, используя китайскую теорему об остатках и модульную арифметику. Китайская тео- рема об остатках утверждает, что целое число в диапазоне от 1 до Р = р, одно- значно определяется его набором остатков от деления /?„ где каждая пара /?„ р, состоит из взаимно простых целых чисел. С помощью таких систем остаточных классов можно выполнять операции сложения, вычитания и умножения (но не деления), причем для манипуляций с большими целыми числами не потребуются сложные структуры дан- ных. Многие из этих алгоритмов для вычислений с большими целыми числами можно пря- мо использовать для вычислений с многочленами. Особенно полезным алгоритмом для быстрой оценки многочленов является правило Горнера. Если выражение ос(-х' вычислять почленно, то нужно будет выполнить О(п2) операций умножения. Намного лучше будет воспользоваться тем обстоятельством, что Р(л) = Со + Х(С| + х(с2 + -г(с3 + ...))). Вычисление этого выражения требует только линейного количества операций. Реализации. Все основные коммерческие системы компьютерной алгебры, включая Maple. Mathematica. Axiom и Macsyma, поддерживают арифметические операции с вы- сокой точностью. Для быстрых результатов в независимом приложении лучшим выбо- ром будет воспользоваться одной из таких систем, если у вас имеется такая возмож- ность. В остальном материале этого подраздела рассматриваются реализации для встроенных приложений. Ведущей библиотекой на языке C/C++ для быстрого выполнения арифметических опе- раций с высокой точностью является библиотека GMP (GNU Multiple Precision Arithmetic Library, библиотека GNU для арифметических операций с многократно уве- личенной точностью), которая поддерживает операции с целыми числами со знаком.
Глава 13. Численные задачи 447 рациональными числами и числами с плавающей запятой. Загрузить ее можно с веб- сайта http://gmplib.org. Класс Biginteer пакета java.math предоставляет аналоги операторов с произвольной точностью для всех элементарных операторов Java. Этот класс также предоставляет дополнительные операции для модульной арифметики, вычисления наибольшего об- щего делителя, проверки на простоту, генерирования простых чисел, манипулирования битами и других операций. Менее производительная и не так хорошо протестированная моя собственная реализа- ция арифметических операций с высокой точностью содержится в библиотеке из моей книги (см. [SR03]). Подробности см. в разделе 19.1.10. Существует несколько общих систем для вычислительной теории чисел, каждая из ко- торых поддерживает операции с целыми числами с произвольной точностью. Инфор- мацию о библиотеках PARI, LiDIA. NTL и MIRACL для работы с задачами теории чи- сел можно найти в разделе 13.8. Пакет ARPEC предоставляет библиотеку на языке C++ и FORTRAN для арифме- тических операций с произвольной точностью и сопутствующий интерактивный калькулятор. А пакет MPFUN90 предоставляет подобную функциональность, но исключительно на языке FORTRAN-90. Оба пакета можно загрузить с веб-сайта http://crd.lbl.gov/~dhbailey/mpdist/. Алгоритм 693 (см. [Smi91]) из коллекции алгорит- мов АСМ предоставляет реализацию на языке FORTRAN возможностей арифметиче- ских операций с плавающей точкой с многократно увеличенной точностью. Подробно- сти см. в разделе 19.1.5. Примечания Основным справочником по алгоритмам для всех основных арифметических операций, включая их реализацию на языке ассемблера MIX, является книга Дональда Кнута [Knu97b], Более современное описание вычислительной теории чисел представлено в книгах [BS96] и [Sho05]. В число работ с описанием алгоритма типа "разделяй и властвуй" для умножения с време- нем исполнения О(н],5У) входят книги [AHU74] и [Мап89]. Алгоритм на основе быстрого преобразования Фурье умножает два числа длиной в п бит за время Op/lg/zlglgn) (см. [SS71 ]). Его описание можно найти, например, в [AHU74] и [Knu97b], Здесь же име- ется описание сведения задачи деления целых чисел к задаче их умножения. Использова- ние быстрого умножения для выполнения других арифметических операций обсуждается в работе [ВегО4Ь] Хорошие описания алгоритмов модульной арифметики и китайской теоремы об остатках содержатся в [AHIJ74] и [CLRS01]. А в книге [CLRS01] дано описание алгоритмов для вы- полнения элементарных арифметических операций. Самым старым представляющим интерес алгоритмом, вероятно, является алгоритм Евклида для вычисления наибольшего общего делителя двух чисел. Его описание можно найти, например, в [Ман89] и [CLRS01]. Родственные задачи. Разложение целых чисел на множители (см. раздел 13.8). крип- тография (см. раздел 18.6).
448 Часть II. Каталог алгоритмических задач 13.10. Задача о рюкзаке Вход. Множество предметов S = {1,п}, где размер /-го предмета равен s„ а значе- ние — V,. Емкость рюкзака равна С Задача. Найти подмножество S'c:S, которое максимизирует значение . учиты- вая. что л <С. т. е.. что все предметы помещаются в рюкзак размером С (рис. 13.10). ВХОД ВЫХОД Рис. 13.10. Задача о рюкзаке Обсуждение. Задача о рюкзаке возникает в ситуациях распределения ресурсов при на- личии финансовых ограничений. Например, как выбрать вещи, которые нужно купить на фиксированную сумму? Так как все предметы имеют свою стоимость и значимость, мы стремимся добиться максимальной значимости при данной стоимости. Само назва- ние задачи— задача о рюкзаке— вызывает в мыслях образ туриста, который из-за ограниченности размера рюкзака старается положить в него лишь самые нужные и компактные вещи. Самая распространенная постановка задачи имеет ограничение вида "0-1", подразуме- вающее. что каждый предмет должен быть либо положен в рюкзак целиком, либо не положен вовсе. В большинстве реальных случаев предметы нельзя разламывать на час- ти произвольным образом, поэтому нельзя взять только одну банку лимонада из шес- тибаночной упаковки или, вообще, часть содержимого одной банки. Вот это ограниче- ние "0-1" и делает задачу о рюкзаке такой сложной, т. к. при возможности разбиения предметов на части оптимальное решение находится "жадным" алгоритмом. Мы про- сто вычисляем "стоимость килограмма" каждого предмета и вкладываем в рюкзак са- мый дорогой предмет или наибольшую его часть и повторяем эту операцию с самым дорогим предметом из оставшихся до тех пор, пока не заполним весь рюкзак. Но. к сожалению, в большинстве приложений ограничение "0-1" присутствует.
Глава 13 Численные задачи 449 Перечислим вопросы, которые возникают при выборе наилучшего алгоритма для ре- шения задачи о рюкзаке. ♦ Имеют ли все предметы одинаковую стоимость или одинаковый размер? Когда стоимость всех предметов одинаковая, то. чтобы получить наибольшую стоимость, мы просто берем максимальное количество предметов. Поэтому оптимальное ре- шение— отсортировать все предметы в возрастающем порядке по размеру, после чего вкладывать их в рюкзак в этом порядке до тех пор. пока имеется свободное ме- сто. Задача решается подобным образом, когда все предметы одинакового размера, но разной стоимости. Предметы сортируются в возрастающем порядке по стоимо- сти и укладываются в рюкзак в этом порядке. Это легкие частные случаи задачи о рюкзаке. ♦ Одинакова ли стоимость килограмма для каждого предмета? В таком случае мы игнорируем цену и просто пытаемся минимизировать незаполненное пространство рюкзака. К сожалению, даже этот ограниченный случай задачи является NP- полным, поэтому нельзя ожидать найти эффективный алгоритм, который всегда выдает решение. Но не отчаивайтесь, т. к. задача о рюкзаке оказывается "легкой" сложной задачей, которую обычно можно решить с помощью представленных далее алгоритмов. Важным частным случаем варианта задачи о рюкзаке с одинаковой стоимостью килограмма каждого предмета является задача разбиения множества целых чисел. представленная графически на рис. 13.11. В данном случае нам нужно разделить элементы множества S на два подмножества А и В таким образом, чтобы ,еАа ~ ' 11ЛИ- хотя ^Ь1, ЧТО^Ь1 разность была минимальной. Задачу разбиения множества целых чисел можно рассматривать как задачу упаковки двух рюкзаков одинаковой вместимости или одного рюкзака с ем- костью, вдвое меньшей, чем исходная, поэтому все три задачи тесно связаны и яв- ляются NP-полными. Вариант задачи с одинаковой стоимостью килограмма для всех элементов часто на- зывается задачей о сумме подмножества (subset sum problem), т. к. мы стремимся найти такое подмножество элементов, общая стоимость которых будет равной определенному целевому значению С. т. е. емкости нашего рюкзака. ♦ Размеры всех предметов представлены относительно небольшими целыми числа- ми? Для варианта задачи, когда размеры предметов и емкость рюкзака представле- ны целыми числами, существует эффективный алгоритм поиска оптимального ре- 15 Зак 3741
450 Часть II Каталог алгоритмических задач шения с временной сложностью О(пС) и сложностью по памяти Подойдет ли этот алгоритм для решения вашей конкретной задачи, зависит от размера С. Он дает отличные результаты для С < 1 000, но не очень хорошие для С> 10 000 000. Алгоритм работает таким образом. Пусть S' будет набором элементов, a C[z, SO бу- дет иметь значение ИСТИНА тогда и только тогда, когда существует подмножество множества S', общий размер всех элементов которого равен точно /. Отсюда следу- ет, что С[Л 0] имеет значение ЛОЖЬ для всех 1 < i< С. Дальше мы по одному до- бавляем элементы я, к S' и обновляем затронутые значения C[z, S']. Обратите внима- ние, что С[/, S'о .s,| = ИСТИНА тогда и только тогда, когда истинны С[/, S'] или C[i-s,, S'], поскольку мы либо используем s, в получении суммы, либо нет. Мы оп- ределяем все суммы, которые можно получить, перебрав п раз все элементы в С — по одному разу для каждого s,, где 1 <j<n, таким образом обновляя массив. Реше- нием задачи является наибольший индекс истинного элемента наибольшего разме- ра. Чтобы воссоздать подмножество, дающее решение, для каждого 1 < z < С мы должны сохранять имя элемента, который изменяет значение C[z] с ЛОЖЬ на ИСТИНА, после чего перебрать элементы массива в обратном направлении. Эта формулировка в стиле динамического программирования игнорирует значения элементов. Чтобы обобщить этот алгоритм, в каждом элементе массива сохраняется значение наилучшего текущего подмножества, для которого сумма размеров эле- ментов равна i. Теперь обновление выполняется, когда сумма стоимости C[z —S3 и стоимости 5, лучше, чем предыдущая стоимость C[z], ♦ Что делать, если имеется несколько рюкзаков? В таком случае задачу лучше рас- сматривать, как задачу разложения по контейнерам. Алгоритмы для решения задачи разложения по контейнерам и задачи раскроя (cutting-stock) описаны в разделе 17.9. А реализации алгоритмов решения задачи с несколькими рюкзаками рассматрива- ются в подразделе "Реализации". Точные решения для рюкзаков большого объема можно найти с помощью целочислен- ного программирования или поиска с возвратом. Наличие или отсутствие элемента z в оптимальном подмножестве обозначается целочисленной переменной х„ принимаю- щей значения 0 или 1. Мы максимизируем выражение ” .х/v, при ограничивающем условии, что У "^%,-s, <С . Реализации алгоритмов целочисленного программирова- ния рассматриваются в разделе 13.6. Когда получение точного решения оказывается слишком дорогим в вычислительном отношении, то возникает необходимость в использовании эвристических алгоритмов. Простой эвристический "жадный" алгоритм вкладывает предметы в рюкзак согласно ранее рассмотренному правилу максимальной стоимости килограмма. Часто такое эв- ристическое решение близко к оптимальному, но, в зависимости от конкретного эк- земпляра задачи, оно также может быть сколь угодно плохим. Правило стоимости ки- лограмма можно использовать, чтобы уменьшить размер задачи в алгоритмах исчер- пывающего перебора, чтобы в дальнейшем не рассматривать "дешевые, но тяжелые" предметы. Другой эвристический алгоритм основан на масштабировании. Динамическое про- граммирование хорошо подходит для тех случаев, когда емкость рюкзака выражена
Глава 13. Численные задачи 451 достаточно небольшим целым числом, скажем. С%. А если нам приходится иметь дело с рюкзаком, емкость которого больше, чем это значение, т. е., С > С8? В таком случае мы уменьшаем размеры всех элементов в С/С8 раз, округляем полученные размеры до ближайшего целого числа, после чего используем динамическое программирование с этими уменьшенными элементами. Масштабирование хорошо зарекомендовало себя на практике, особенно при небольшом разбросе размеров элементов. Реализации. Коллекцию реализаций алгоритмов на языке FORTRAN для разных вер- сий задачи о рюкзаке можно загрузить с веб-сайта http://www.or.deis.unibo.it/kp.html. Здесь же можно загрузить электронную версию книги [МТ90а]. Хорошо организованную коллекцию реализаций алгоритмов на языке С для решения разных видов задачи о рюкзаке и родственных вариантов задачи, таких как разложение по контейнерам и загрузка контейнера, можно загрузить с веб-сайта http:// www.diku.dk/~pisinger/codes.html. Самый мощный код основан на алгоритме динами- ческого программирования, описание которого дается в работе [МРТ99]. Алгоритм 632 на языке FORTRAN из коллекции ЛСМ предназначен для решения за- дачи о рюкзаке с ограничением вида "0-1". Кроме того, он поддерживает работу с не- сколькими рюкзаками. Подробности см. в разделе 19.1.5. Примечания Самым свежим справочником по задаче о рюкзаке и ее вариантах является книга [КРР04]. Книга [МТ90а] и обзорная статья [МТ87] представляют собой обычные справочные посо- бия по задаче о рюкзаке, содержащие как теоретические, так и экспериментальные ре- зультаты се решения. Замечательное описание алгоритмов целочисленного программиро- вания для решения задач о рюкзаке представлено в книге [SDK83]. Алгоритмы решения задачи о рюкзаке с ограничением вида "0-1" обсуждаются в работе [МРТОО]. Аппроксимирующий метод лает решение, близкое к оптимальному, за время, зависящее полиномиально от размера входа и коэффициента аппроксимации е. Такое очень строгое ограничение заставляет искать компромисс между временем испол- нения и качеством аппроксимации. Хорошие описания аппроксимирующих методов с по- линомиальным временем исполнения для решения задачи о рюкзаке и суммы подмноже- ства можно найти в [1К75], [BvG99], [CLRS01], [GJ79] и [Мап89]. Первый алгоритм для общего метода шифрования с открытым ключом был основан на сложности задачи о рюкзаке. Описание см. в книге [Sch96]. Родственные задачи. Разложение по контейнерам (см. раздел 17.9), целочисленное программирование (см. раздел 13.6). 13.11. Дискретное преобразование Фурье Вход. Последовательность из п действительных или комплексных значений /?, функ- ции /?. выбранных через одинаковые интервалы. Задача. Дискретное преобразование Фурье Нт = '^[\}^ке1тк""" для 0 <т<п-\ (рис. 13.12).
452 Часть II. Каталог алгоритмических задач Обсуждение. В то время как программисты обычно плохо разбираются в преобразова- ниях Фурье, инженеры-электронщики, работающие в области обработки сигналов, об- ращаются с ними вполне свободно. В функциональном отношении преобразования Фурье предоставляют способ для преобразования выборок стандартного временного ряда в частотную область. Таким образом получается двойственное представление функции, при котором определенные операции становятся проще для выполнения. Преобразования Фурье находят следующие применения: ♦ фильтрация. Выполнение преобразования Фурье для функции равносильно пред- ставлению ее в виде суммы синусов. Устранив "лишние" высокочастотные и/или низкочастотные компоненты (т. е. исключив некоторые слагаемые) и взяв обратное преобразование Фурье, мы можем отфильтровать шум и другие нежелательные явления. Например, всплеск на графике в рис. 13.12 соответствует периоду одной синусной составляющей и моделирует входные данные. Остальные компоненты являются шумом; ♦ сжатие изображении. Сглаженное, отфильтрованное изображение содержит меньший объем информации, чем исходное изображение, в то же самое время со- храняя похожий внешний вид. Удалив компоненты, вносящие сравнительно не- большой вклад в изображение, мы можем уменьшить размер изображения за счет незначительного понижения его точности; ♦ свертка и обращение свертки. Преобразования Фурье можно использовать для эф- фективного выполнения свертки двух последовательностей. Сверткой (convolution) называется попарное умножение элементов двух разных последовательностей, на- пример, при умножении двух многочленов / и g с п переменными или при сравне- нии двух текстовых строк. Реализация такой операции напрямую имеет квадратич- ное время исполнения (т. е. О(п )). в то время как время исполнения алгоритма на основе быстрого преобразования Фурье равно O(n\gn). Приведем другой пример из области обработки изображений. Так как сканер изме- ряет уровень освещенности участка изображения, а не отдельной его точки, то от- сканированное изображение всегда немного смазано. Исходный сигнал можно вос- становить. выполнив обращение свертки входного сигнала посредством гауссовой функции рассеяния точки;
Глава 13. Численные задачи 453 ♦ вычисление корреляционной функции. Корреляционная функция двух функций /(/) и g(/) определяется как: z(/)= £^/(т)^(/ + т)<7т Эту функцию можно с легкостью вычислить, используя преобразования Фурье. Ко- гда две функции имеют похожую форму, но сдвинуты по отношению друг к другу (например, как функции /(/) = sin(/) и g(t) = cos(/)), значение л(/о) будет большим для этого смещения /о- В качестве примера практического применения допустим, что мы хотим определить, нет ли в нашем генераторе случайных чисел каких-либо нежела- тельных периодичных последовательностей. Для этого мы можем сгенерировать длинную последовательность случайных чисел, преобразовать их во временную по- следовательность (в которой /-е число соответствует моменту времени i), после чего взять преобразование Фурье этой последовательности. Любой всплеск будет инди- катором возможной периодичности. Дискретное преобразование Фурье принимает на входе п комплексных чисел /?*-, где D<k<n- I, соответствующих равномерно распределенным точкам временной после- довательности, и выдает на выходе п комплексных чисел кф, где 0 <к<п- I. каждое из которых описывает синусоидальную функцию данной частоты. Дискретное преоб- разование Фурье определяется следующей формулой: н1Н= к=0 А обратное преобразование Фурье определяется следующей формулой: к=0 Таким образом мы можем с легкостью перемещаться между h и Н. Так как выход дискретного преобразования Фурье состоит из п чисел, каждое из кото- рых вычисляется по формуле, содержащей п элементов, то его время исполнения равно О(п~У А алгоритм быстрого преобразования Фурье вычисляет дискретное преобразо- вание Фурье за время (?(nlogw). Вероятно, это самый важный известный алгоритм, т. к. он лежит в основе всей современной обработки сигналов. Под общим названием быст- рого преобразования Фурье известно несколько алгоритмов, использующих метод "разделяй и властвуй". По сути, задача вычисления дискретного преобразования Фурье по п точкам сводится к вычислению двух преобразований, каждое по nil точкам, кото- рые потом применяются рекурсивно. Алгоритм быстрого преобразования Фурье обычно предполагает, что число п является степенью двойки. Если ваше значение п не отвечает этому условию, то будет лучше дополнить данные нулями, чтобы создать п = 2к элементов, чем искать более общий код. Многие системы обработки сигналов имеют строгие условия работы в реальном вре- мени, поэтому алгоритмы быстрого преобразования Фурье часто реализуются аппарат- но или. по крайней мере, на языке ассемблера, настроенного под конкретную машину. Учтите это обстоятельство, если используемый вами код окажется слишком мед- ленным.
454 Часть II. Каталог алгоритмических задач Реализации. На первом месте среди открытых кодов алгоритма быстрого преобразо- вания Фурье стоит библиотека FFTW. Это библиотека процедур на языке С для вычис- ления дискретного преобразования Фурье в одном или нескольких измерениях, под- держивающая вход произвольного размера и данные, как действительного, так и ком- плексного типа. Результаты обширного тестирования доказывают, что это самое быстрое известное преобразование Фурье. Библиотека снабжена интерфейсами для FORTRAN и C++. В 1999 г. библиотеке был присужден приз Дж. X. Вилкинсона в об- ласти численного программного обеспечения (J. Н. Wilkinson Prize for Numerical Software). Библиотеку FFTW можно загрузить с веб-сайта http://www.fftw.org/. Пакет FFTPACK содержит процедуры на языке FORTRAN для вычисления быстрого преобразования Фурье периодических и других симметрических последовательностей. Пакет включает комплексные, действительные, синусоидальные и четвертьволновые преобразования. Загрузить его можно с веб-сайта http://www.netlib.org/fftpack. Науч- ная библиотека GNU для C/C++ предоставляет версию библиотеки FFTPACK. Под- робности см. по адресу http://www.gnii.org/software/gsl/. Алгоритм 545 (см. [Fra79]) из коллекции АСМ является реализацией на языке FORTRAN быстрого преобразования Фурье, оптимизирующей производительность виртуальной памяти. Дополнительную информацию см. в разделе 19.1.5. Примечания Хорошим введением в предмет преобразований Фурье и быстрых преобразований Фурье являются книги [Вга99] и [Bri88]. Описание преобразования можно найти в книге [PFTV07]. Изобретение быстрого преобразования Фурье обычно приписывают Кули (Cooley) и Туки (Tukey); см. [СТ65]. Подробную историю вопроса можно найти в [Вп88]. Нечувствительный к кэшированию алгоритм быстрого преобразования Фурье приведен в докладе [FLPR99]. Само понятие нечувствительных к кэшированию алгоритмов было впервые предложено в этом докладе. Библиотека FFTW основана на этом алгоритме. Подробности устройства библиотеки FFTW см. в [FJ05]. Интересный алгоритм типа "разделяй и властвуй" для умножения многочленов (см. [КО63]) с временем исполнения О(«159) рассматривается в книгах [AHU74] и [Мап89]. Алгоритм на основе быстрого преобразования Фурье, умножающий два числа длиной в п бит за время O(r;lgnlglgr;), был разработан Шонхаге (Schonhage) и Штрассеном (Strassen). Он представлен в [SS71] и [AHLI74]. Вопрос о том, действительно ли комплексные переменные являются необходимыми в бы- стрых алгоритмах для выполнения свертки, является открытым. К счастью, в большинст- ве приложений быструю свертку можно использовать в виде черного ящика. Быстрая свертка лежит в основе многих разновидностей алгоритмов для сравнения строк (см. [Ind98]). В последнее время было предложено вместо преобразования Фурье использовать в фильтрации специальные функции, называемые вейвлетами Введение в эту тему можно найти в [Wal99J. Родственные задачи. Сжатие данных (см. раздел 18.5), арифметические операции с высокой точностью (см. раздел 13.9).
ГЛАВА 14 Комбинаторные задачи В этой главе мы рассмотрим несколько алгоритмических задач чисто комбинаторного характера. В число этих задач входят задача сортировки и задача генерирования пере- становок, которые были первыми нечисловыми задачами, решенными с помощью электронных вычислительных машин. Сортировку можно рассматривать, как полное упорядочивание ключей, а поиск и выбор— как идентификацию ключей по их поло- жению в упорядоченной последовательности. Здесь также рассматриваются другие комбинаторные объекты, такие как перестановки, разбиения, подмножества, календари и расписания. Особый интерес представляют ал- горитмы, которые ранжируют комбинаторные объекты, т. е. устанавливают соответст- вие между объектом и уникальным целым числом, или выполняют обратную опера- цию, возвращая объект по числу. Операции ранжирования упрощают многие другие задачи, например генерирование случайных объектов (выбирается произвольное число и выполняется операция, обратная ранжированию) или перечисления всех объектов по порядку (в цикле генерируются числа от I до и, для каждого из которых выполняется операция, обратная ранжированию). В конце главы обсуждается задача генерирования графов. Алгоритмы на графах пред- ставлены более подробно в последующих разделах каталога задач. Среди книг по общим комбинаторным алгоритмам я порекомендую следующие: ♦ [NW78]— эта книга посвящена алгоритмам создания элементарных комбинатор- ных объектов, таких как перестановки, подмножества и разбиения. Такие алгорит- мы. как правило, трудно найти, и они нередко имеют много тонкостей. Для всех алгоритмов предоставляются программы на языке FORTRAN, а также обсуждаются их теоретические основы. Подробности см. в разделе 19.1.10', ♦ [KS99] — книга по комбинаторным алгоритмам: кроме этого, особое внимание уде- ляется алгебраическим задачам, таким как изоморфизм и симметрия; ♦ [Knu97a] и [Knu98] — общепризнанные справочники по сортировке и поиску; со- держат обширный материал по комбинаторным объектам, таким как перестановки. Отдельными изданиями выпущен материал (предположительно составляющий со- держимое мифического Тома 4) по генерированию перестановок (см. [Кпи05а]), подмножеств и разбиений (см. [Knu05b]), а также деревьев (см. [КпиОб]); ♦ [SW86a] — учебник по комбинаторике для студентов вузов, содержащий алгоритмы генерирования перестановок, подмножеств и разбиений множества. Содержит соот- ветствующие программы на языке Pascal; ♦ [PSO3]— описание библиотеки Combinatorica. содержащей свыше 400 функций системы Mathematica для генерирования комбинаторных объектов и объектов тео-
456 Часть II. Каталог алгоритмических задач рии графов (см. раздел 19.1.9}. Авторы книги придерживаются особого взгляда на взаимодействие разных алгоритмов. 14.1. Сортировка Вход. Множество из и элементов. Задача. Расположить элементы в возрастающем (или убывающем) порядке (рис. I4.1). ВХОД ВЫХОД Рис. 14.1. Сортировка Обсуждение. Сортировка является фундаментальной алгоритмической задачей теории вычислительных систем. Для программиста изучение разных алгоритмов сортировки подобно изучению гамм для музыканта. Как показано в разделе 4.2, сортировка являет- ся первым шагом в решении множества других алгоритмических задач, а правило "ес- ли не знаешь, как поступить, выполняй сортировку" является одним из главных при разработке алгоритмов. Сортировка также иллюстрирует все стандартные парадигмы разработки алгоритмов. Большинству программистов известно так много разных алгоритмов сортировки, что зачастую им трудно принять решение, какой из них выбрать в конкретном случае. Сле- дующие вопросы помогут вам в выборе алгоритма. ♦ Сколько элементов нужно отсортировать? Для небольших объемов данных (и < 100) не играет особой роли, какой из алгоритмов с квадратичным временем ис- полнения использовать. Сортировка вставками быстрее, проще и в меньшей степе- ни чревата ошибками реализации, чем пузырьковая сортировка. Сортировка мето- дом Шелла очень похожа на сортировку вставками, только намного быстрее. Одна- ко для ее реализации необходимо знание правильной последовательности вставок, изложенной в книге [Кпи98]. Если количество сортируемых элементов больше 100, то важно использовать алго- ритм с временем исполнения O(n\gn), такой как пирамидальная сортировка, быстрая сортировка или сортировка слиянием. Разные программисты выбирают разные алгоритмы, руководствуясь личными предпочтениями. Поскольку трудно сказать, какой алгоритм быстрее, выбор того или иного алгоритма не имеет особого значения.
Глава 14. Комбинаторные задачи 457 А когда количество элементов для сортировки превышает, скажем, 5 000 000, то пора начинать думать об использовании алгоритма сортировки данных, содержа- щихся на внешних запоминающих устройствах, который минимизирует обращения к этим устройствам. Алгоритмы обоих типов рассматриваются далее в этом разделе. ♦ Содержится ли в данных дубликаты? Порядок сортировки полностью определен, если все элементы имеют разные ключи. Но когда два элемента имеют одинаковый ключ, то для решения, какой элемент должен стоять первым, нужно применить до- полнительный критерий. Для многих приложений это не играет роли, и подходит любой алгоритм сортировки. А в случаях, когда это важно, очередность элементов с одинаковыми ключами определяется по вторичному ключу, например, по имени для одинаковы фамилий. А иногда очередность в отсортированном списке определяется по положению в ис- ходном списке. Допустим, что 5-й и 27-й элементы исходного набора данных имеют одинаковый ключ. В таком случае в отсортированном списке 5-й элемент должен находиться перед 27-м. В случае совпадения ключей стабильный алгоритм сорти- ровки сохраняет исходный порядок элементов в отсортированном списке. Боль- шинство алгоритмов сортировки с квадратичным временем исполнения являются стойкими, в то время как многие из алгоритмов с временем исполнения (?(nlgw)— нет. Если важна стабильность сортировки, то в функции сравнения, скорее всего, будет лучше использовать исходное положение элемента в качестве вторичного ключа, чем полагаться на реализацию алгоритма для обеспечения стабильности. ♦ Чти известно о данных? Для ускорения сортировки данных часто можно восполь- зоваться их особенностями. Конечно, сортировка, имеющая время исполнения O(rtlgn). является быстрой операцией, поэтому, если узким местом вашего приложе- ния является время, затрачиваемое на сортировку, можете считать, что вам повезло. • Данные уже частично отсортированы? В таком случае определенные алгорит- мы, например алгоритм сортировки вставками, работают быстрее, чем на полно- стью неотсортированных данных. • Что известно о распределении ключей? Если ключи распределены произвольно или равномерно, то имеет смысл использовать корзинную сортировку или сор- тировку распределением. Просто помещаем ключи в корзины по их первой бук- ве и выполняем рекурсию до тех пор, пока каждая корзина не достигнет размера, достаточно малого для сортировки полным перебором. Это очень эффективный подход, когда все ключи распределены по корзинам равномерно. Но в случае сортировки списка группы однофамильцев производительность корзинной сор- тировки будет очень плохой. • Очень длинные или трудные для сравнения ключи? Если ключи представлены в виде длинных текстовых строк, то, возможно, имеет смысл выполнить предва- рительную сортировку по сравнительно короткому префиксу (скажем, длиной в 10 символов), после чего выполнить сортировку по полному ключу. Этот метод особенно подходит для сортировки на внешних носителях (см. далее), при кото- рой нежелательно расходовать быструю память на хранение несущественных де- талей.
458 Часть II Каталог алгоритмических задач Другим подходом может быть использование поразрядной сортировки, имею- щей время исполнения линейное относительно количества символов в файле, что гораздо лучше времени O(nlg«), которое еще должно быть помножено на стоимость сравнения двух ключей. • Невелик диапазон возможных ключей? Если требуется отсортировать подмноже- ство из, скажем, п/2 разных целых чисел, значение каждого из которых лежит в диапазоне от 1 до п, то самым быстрым алгоритмом будет создать «-элементный битовый вектор, установить соответствующие ключам биты, а потом отсканиро- вать вектор слева направо и доложить о позициях с установленными битами. ♦ Необходимо ли обращаться к диску? При сортировке больших объемов информа- ции сортируемые данные могут не помещаться в память. В таких случаях мы имеем дело с задачей внешней сортировки (external sorting), для которой требуется приме- нение внешнего устройства хранения данных. Первоначально в качестве внешних устройств хранения данных использовались накопители на магнитной ленте, и в книге [Knu98] описываются разнообразные сложные алгоритмы для эффективного слияния данных с разных накопителей. В настоящее время используется виртуаль- ная память и обмен данными с диском. Любой алгоритм сортировки будет работать с виртуальной памятью, но у многих алгоритмов все время будет уходить на пере- мещение данных между памятью и диском. Самым простым подходом к внешней сортировке будет загрузка данных в В-дерево (см. раздел 12.1) с последующим симметричным обходом этого дерева, позволяю- щим читать ключи в отсортированном порядке. Однако, действительно высокопро- изводительные алгоритмы основаны на сортировке многоканальным слиянием. При этом подходе данные разбиваются на несколько файлов, каждый из которых сорти- руется с применением быстрой внутренней сортировки, после чего отсортирован- ные файлы сливаются поэтапно с использованием двух- или ^-канального слияния. Производительность можно оптимизировать, используя сложные схемы слияния и управления буфером, учитывающие свойства внешнего устройства хранения. ♦ Сколько времени отводится на разработку и отладку процедуры сортировки? Если бы мне нужно было реализовать рабочую процедуру в течение часа, то я, ско- рее всего, предпочел бы простую сортировку методом выбора. Но если бы в моем распоряжении был целый день, то я, скорее всего, использовал бы пирамидальную сортировку, т. к. этот алгоритм обладает надежной производительностью и не тре- бует дополнительной настройки. Самым лучшим алгоритмом внутренней сортировки является быстрая сортировка (см. раздел 4.2), хотя для получения максимальной производительности придется пово- зиться с настройкой. В действительности, будет намного лучше использовать для этого алгоритма библиотечную функцию вашего компилятора, чем пытаться разработать свою реализацию. Неудачная реализация быстрой сортировки будет работать медлен- нее, чем плохая реализация пирамидальной сортировки. Если же вы настроены на создание своей реализации алгоритма быстрой сортировки, то используйте следующие эвристические методы, которые на практике оказывают большое влияние на производительность.
Глава 14. Комбинаторные задачи 459 ♦ Используйте рандомизацию. Выполнив произвольную перестановку ключей перед сортировкой (см. раздел 14.4}, вы, возможно, добьетесь того, что время сортировки почти отсортированных данных не будет квадратичным. ♦ Выбирайте средний элемент из трех. Для выбора опорного элемента используйте средний по величине элемент из первого, последнего и центрального элемента мас- сива, чтобы повысить вероятность разбиения массива на приблизительно одинако- вые части. По результатам некоторых экспериментов для больших подмассивов следует составлять выборку из большего количества элементов, а для небольших — из меньшего. ♦ Для сортировки вставками выбирайте подмассивы небольшого размера. Переклю- чение с рекурсивной быстрой сортировки на сортировку вставками имеет смысл только при небольшом размере подмассивов, например, не превышающем 20 эле- ментов. Оптимальный размер подмассивов для данной реализации определяется экспериментальным путем. ♦ Обрабатывайте наименьшую порцию данных в первую очередь. Исходя из предпо- ложения, что ваш компилятор способен оптимизировать хвостовую рекурсию, вы можете минимизировать объем требуемой памяти, обрабатывая меньшие порции данных раньше. Так как каждый следующий рекурсивный вызов занимает в стеке память, превышающую память, отведенную под предыдущий вызов, максимум в полтора раза, то потребуется только O(lgn) стековой памяти. Прежде чем приступать к реализации алгоритма быстрой сортировки, прочитайте статью [Веп92Ь]. Реализации. Наилучшей программой сортировки с открытым кодом предположитель- но является GNU sort, которая является частью библиотеки основных утилит GNU. До- полнительную информацию см. по адресу http://www.gnu.org/software/coreutils/. Имейте в виду, что существуют также коммерческие поставщики высокопроизводи- тельных программ внешней сортировки, такие как Cosort (www.cosort.com), Syncsort (www.syncsort.com) и Ordinal Technology (www.ordinal.com). Современные языки программирования содержат библиотеки эффективных реализа- ций сортировки, поэтому вам никогда не придется разрабатывать собственную про- цедуру. Стандартная библиотека С содержит функцию qsort, являющуюся, предполо- жительно, обобщенной реализацией быстрой сортировки. Библиотека STL языка C++ предоставляет методы sort и stable sort. Библиотеку STL можно загрузить с веб-сайта http://www.sgi.com/tech/stl/. Более подробное руководство по использованию библио- теки STL и стандартной библиотеки C++ можно найти в книгах [Jos99], [MeyOl] и [MDS01], Стандартный пакет утилит Java java.util (http://java.sun.com/javase/) содержит неболь- шую библиотеку структур данных Java Collections (JC), предоставляющую, в частно- сти, классы SortedMap И SortedSet. Реализацию нечувствительного к кэшированию алгоритма funnelsort на языке C++ см. на веб-сайте http://kristoffer.vinther.name/projects/funnelsort/. Результаты тестов свидетельствуют о его высокой производительности.
460 Часть II. Каталог алгоритмических задач Существует множество веб-сайтов, содержащих анимационные апплеты для всех фун- даментальных алгоритмов сортировки, включая пузырьковую сортировку, пирами- дальную сортировку, быструю сортировку, поразрядную сортировку и сортировку ме- тодом Шелла. Безусловно, сортировка является классической задачей для анимации алгоритмов. Типичные примеры таких сайтов: http://www.cs.ubc.ca/spider/harrison/ Java/sorting-demo.html и http://www.cs.bell-labs.com/cin/cs/pearls/sortanini.html. Примечания Самой лучшей книгой по сортировке из числа написанных в прошлом и, пожалуй, в бу- дущем является книга [Кпи98]. Ей больше тридцати лет, но читать ее по-прежнему инте- ресно. Одной из областей сортировки, которые были разработаны после первого издания этой книги, является сортировка предварительно отсортированных данных, исследуемая в работе [ECW92]. Заслуживающим внимания справочником по сортировке является книга [GBY91], которая содержит ссылки на алгоритмы для частично отсортированных данных, а также реализации на языках С и Pascal всех фундаментальных алгоритмов. Описания основных алгоритмов внутренней сортировки содержатся в каждом учебнике по алгоритмам. Алгоритм пирамидальной сортировки был разработан в 1964 г. Дж. Виль- ямсом (J. Williams), см. [Wil64]. Алгоритм быстрой сортировки был разработан в I960 г. Чарльзом Хоаром (Charles Ноаге), см. [Ноа62]; а всесторонний анализ и реализация были осуществлены Р. Седжвиком (R. Sedgewick), см. [Sed78]. Первая реализация алгоритма сортировки слиянием (на компьютере EDVAC в 1945 г.) была выполнена фон Нейманом. Полное изложение истории сортировки, берущей начало во времена электронно- вычислительных машин, работавших на перфокартах, см. в книге [Кпи98]. Главным чемпионатом по высокопроизводительной сортировке являются ежегодные со- ревнования, начало которым положил Джим Грей (Jim Gray). На веб-сайте http:// sortbenchiiiark.org/ опубликована информация о текущих и предыдущих результатах, ко- торые могут вдохновлять или расстраивать, в зависимости от вашей точки зрения. Дос- тигнутый прогресс вдохновляет (тестовые экземпляры размером в миллион записей, ко- торые использовались в первых соревнованиях, теперь кажутся незначительными), но лично меня расстраивает тот факт, что вопросы управления системой и памятью оказы- ваются гораздо важнее комбинаторно-алгоритмических аспектов сортировки. В настоящее время создание высокопроизводительных программ сортировки включает в себя рабогу над алгоритмами как требовательными к кэшированию (см. [LL99]). так и нечувствительными к нему (см. [BFV07]). Для сортировки существует хорошо известная нижняя граница Q(iilgn) на модели алгеб- раического дерева решений (см. [ВО83]). Задача определения точного количества сравне- ний, необходимого для сортировки п элементов при небольших значениях п, вызвала зна- чительный интерес у исследователей. Описание задачи можно найти в [Aig88] и [Raw92], а последние результаты — в [РесО4] и [Рес07]. Но эта нижняя граница не соблюдается при других моделях вычислений. В работе [FW93] представляется алгоритм сортировки со сложностью O(«^lg«) по модели вычислений, позволяющей выполнять арифметические операции над ключами. Обзор алгоритмов быстрой сортировки по таким нестандартным моделям вычислений приведен в [And05]. Родственные задачи. Словари (см. раздел 12.1). поиск (см. раздел 14.2). топологиче- ская сортировка (см. раздел 15.2).
Глава 14 Комбинаторные задачи 461 14.2. Поиск Вход. Набор 5 из п ключей и ключ запроса q. Задача. Определить местоположение ключа q в наборе ключей S (рис. I4.2). G ? G ВХОД ВЫХОД Рис. 14.2. Поиск в наборе ключей Обсуждение. Для разных людей слово "поиск" означает разные понятия. Поиск гло- бального максимума или минимума функции является задачей безусловной оптимиза- ции, которая рассматривается в разделе 13.5. Программы для игры в шахматы выбира- ют свой лучший ход, выполняя исчерпывающий перебор/поиск всех возможных ходов, используя вариант поиска с возвратом (см. раздел 7.7). Мы же ставим перед собой задачу найти ключ в списке, массиве или дереве. Словар- ные структуры данных предоставляют эффективный доступ к наборам ключей для вставки и удаления (см. раздел 12.1). Типичными представителями словарных структур данных являются двоичные деревья и хэш-таблицы. Мы рассматриваем задачу поиска без привязки к словарям, поскольку в нашем распо- ряжении имеются более простые и эфффективные решения, когда мы занимаемся ста- тическим поиском без вставок и удалений. При правильном использовании во внут- реннем цикле такие простые структуры данных могут обеспечить заметное повышение производительности. Кроме того, идеи двоичного поиска и самоорганизации примени- мы при решении других задач, что оправдывает наше внимание к ним. Существуют два основных метода поиска: последовательный поиск и двоичный поиск. Оба достаточно просты, но у них имеются интересные варианты. При последователь- ном поиске мы движемся по списку ключей с самого начала и сравниваем каждый по- следующий элемент с ключом поиска, пока не найдем совпадающий или не достигнем конца списка. При двоичном поиске мы работаем с отсортированным списком или мас- сивом ключей. Чтобы найти ключ q, мы сравниваем значение q со значением среднего ключа массива 5„/2. Если значение ключа q меньше, чем значение ключа 5„,2, то данный ключ должен находиться в верхней половине массива .S': в противном случае он должен находиться в его нижней половине. Повторяя этот процесс на той части массива, кото-
462 Часть II. Каталог алгоритмических задач рая содержит элемент q, мы находим его за [lg«] сравнений, что намного лучше, чем ожидаемые п/2 сравнений при последовательном поиске. Дополнительную информа- цию по двоичному поиску см. в разделе 4.9. Последовательный поиск является простейшим алгоритмом и, скорее всего, будет са- мым быстрым, когда количество элементов не превышает 20. При ста и более элемен- тах двоичный поиск будет эффективнее, чем последовательный, и оправдает затраты на сортировку при большом количестве запросов. Но для принятия решения о выборе варианта алгоритма следует учитывать и другие факторы ♦ Сколько времени отводится на программирование? Известно, что алгоритм двоич- ного поиска сложен для программирования. Между изобретением алгоритма и опубликованием первой правильной реализации прошло целых семнадцать лет! По- этому я советую вам начать с одной из реализаций, описанных далее. Полностью протестируйте ее, создав программу, которая выполняет поиск каждого ключа в на- боре S, а также для промежуточных значений ключей. ♦ Какова частота обращений к разным ключам? Определенные английские слова (например, "the") встречаются намного чаще, чем другие (например, "defenestrate"). Количество сравнений в последовательном поиске можно уменьшить, поместив часто употребляемые слова вверху списка, а более редкие— внизу. Неравномер- ность запросов является скорее правилом, чем исключением. Слова во многих язы- ках распределяются согласно степенным законам. Классическим примеров являет- ся распределение слов в английском языке, которое довольно точно моделируется законом Ципфа. Согласно этому закону, /-й по частоте обращения ключ выбирается с вероятностью, которая в (т-1)// раз превышает вероятность выбора ключа. (/ - 1)-го по популярности, для всех 1 < / < п. Знание частоты обращений легко использовать в последовательном поиске. Но в двоичном дереве дело обстоит сложнее. Мы хотим, чтобы часто употребляемые ключи располагались возле корня (таким образом мы быстрее найдем их), но не за счет разбалансирования дерева и превращения двоичного поиска в последователь- ный. Решением этой проблемы будет применение алгоритма динамического про- граммирования для построения оптимального дерева двоичного поиска. Принципи- ально важным здесь является то обстоятельство, что каждый возможный корневой узел / делит пространство ключей на две части (слева и справа от /), каждую из ко- торых можно представить оптимальным деревом двоичного поиска для меньшего поддиапазона ключей Корень оптимального дерева выбирается таким образом, чтобы минимизировать ожидаемую стоимость поиска в получившемся разбиении. ♦ Может ли частота обращений измениться со временем? Для предварительной сортировки списка или дерева с целью воспользоваться асимметричной закономер- ностью доступа требуется знать эту закономерность доступа наперед. Для многих приложений получение такой информации может быть трудной задачей. Лучшим подходом будет использовать самоорганизующиеся списки, в которых порядок клю- чей изменяется в ответ на запросы. Самой лучшей схемой самоорганизации являет- ся перемещение ключа, к которому выполнялось самое последнее обращение, в на- чало списка. Таким образом ключи с наибольшим числом обращений продвигаются в начало списка, а с наименьшим— к концу. Отслеживать частоту обращений нет
Глава 14. Комбинаторные задачи 463 необходимости, мы просто перемещаем ключ при обращении к нему. В самоорга- низующихся списках также используется принцип временной локальности, т. к. су- ществует вероятность повторных обращений к данному ключу. Таким образом, пользующийся спросом ключ будет удерживаться вверху списка на протяжении по- следовательности обращений, даже если в прошлом другие ключи пользовались большим спросом. Принцип самоорганизации может расширить размер полезного диапазона последо- вательного поиска, но когда количество ключей превышает 100, то следует пере- ключаться на двоичный поиск. Но также можно рассмотреть возможность исполь- зования косых деревьев, являющихся деревьями двоичного поиска, в которых эле- мент. к которому было выполнено обращение, перемещается в корневой узел. Эти деревья дают гарантию отличной амортизированной производительности. ♦ Известно ли примерное местоположение ключа? Допустим, мы знаем, что требуе- мый ключ находится справа от позиции р. причем на небольшом расстоянии. Если мы правы в нашем предположении, то последовательный поиск быстро возвратит нужный элемент, но за ошибку нам придется дорого заплатить. Гораздо лучше про- верять ключи через возрастающие интервалы справа отр (р + 1, р + 2. р + 4, р + 8. р + 16,...), пока мы не дойдем до ключа, расположенного справа от целевого. Тогда у нас появится окно, содержащее целевой элемент, и мы сможем найти его, приме- нив двоичный поиск. Такой односторонний двоичный поиск возвращает целевой ключ в позиции р + 1 за, самое большее, 2pg/ ]сравнений, т. е. он работает быстрее двоичного поиска, если /«и, и такая производительность никогда не станет намного хуже. Односторонний двоичный поиск особенно полезен при неограниченном поиске, например, при из- влечении корня. ♦ Находится ли структура данных на внешнем устройстве? Когда количество клю- чей слишком велико, двоичный поиск теряет свой статус лучшего метода поиска. Мы будем метаться по пространству ключей в поисках средней точки для сравнения с целевым ключом и для каждого из этих сравнений потребуется загрузить новую страницу из внешнего устройства. Намного лучше будез использовать такие струк- туры данных, как В-деревья (см. раздел 12.Г) или деревья Емде Боаса (см. примеча- ния далее), которые собирают ключи в страницы, чтобы свести к минимуму количе- ство обращений к диску в каждом поиске. ♦ Можно ли угадать местоположение ключа? В интерполяционном поиске мы ис- пользуем свое знание распределения ключей, чтобы угадать следующую позицию для сравнения. Кстати, при поиске в телефонной книге более уместно было бы го- ворить об интерполяционном, а не о двоичном поиске. Допустим, что мы ищем но- мер телефона человека по фамилии Washington. Мы можем уверенно делать первое сравнение где-то в третьей четверти книги, фактически выполняя два сравнения по цене одного. Хотя идея интерполяционного поиска кажется привлекательной, мы рекомендуем не использовать его по трем причинам. Во-первых, нужно будет выполнить боль- шую работу, чтобы оптимизировать алгоритм поиска, прежде чем можно будет на-
464 Часть II. Каталог алгоритмических задач деяться получить лучшую производительность, чем при двоичном поиске. Во- вторых, даже если вы и повысите производительность по сравнению с двоичным поиском, то вряд ли настолько, чтобы оправдать вложенные усилия. И наконец, в-третьих, ваша программа будет намного менее надежна и эффективна при смене докали. например, при поиске французских слов вместо английских. Реализации. Элементарные алгоритмы последовательного и двоичного поиска доста- точно просты, что можно рассматривать возможность их самостоятельной реализации. Тем не менее, стандартная библиотека С содержит процедуру bsearch, являющуюся, предположительно, обобщенной реализацией двоичного поиска. Библиотека STL язы- ка C++ предоставляет итераторы find (последовательный поиск) и binary search (дво- ичный поиск). Стандартный пакет утилит Java java.util (http://java.sun.coni/javase) предоставляет процедуру binarysearch. Многочисленные реализации приводятся во многих учебниках по структурам данных. Реализации косых деревьев и других поисковых структур на языке C++ и Java даются в книгах [Sed98] (http://www.cs.princeton.edu/~rs) и [Wei06] (http://www.cs.fiu.edu/ ~weiss). Примечания В книге ([MS05]) приводится обзор современного состояния дел в области словарных структур данных. Другие обзоры представлены в книгах [МТ90Ь] и [GBY91]. В книге [Knu97a] представлен детальный анализ и описание основных алгоритмов поиска и сло- варных структур данных, однако в ней отсутствуют такие современные структуры дан- ных, как красно-черные и косые деревья. Очередная позиция сравнения при линейном интерполяционном поиске в массиве отсор- тированных чисел определяется по следующей формуле: next = (low -1) + q — .ST/он’ - 11 —--------------------------= x (high - low + 1) •S’[/»g// + 1]- .Spoil’- I где q — числовой ключ запроса, a S' — отсортированный массив числовых значений. Если ключи распределены равномерно и выбираются независимым образом, то ожидаемое время поиска равно O(lglg«) (см. [DJP04] и [PIA78]). Неравномерность распределения обращений можно использовать в деревьях двоичного поиска, организовав их таким образом, чтобы часто запрашиваемые ключи находились возле корня, что позволит минимизировать время поиска. Такие оптимальные деревья по- иска можно создавать за время (7(«)g«) посредством динамического программирования (см. [Кгш98]). В книге [SW86b] описан нетривиальный алгоритм для эффективного пре- образования двоичного дерева в дерево минимальной высоты (оптимально сбалансиро- ванное) посредством операций ротации. Метод Емде Боаса для создания двоичного дерева (или отсортированного массива) обес- печивает более высокую производительность при работе с внешними устройствами хра- нения данных, чем двоичный поиск, но за счет более сложной реализации. Информацию об этой и других нечувствительных к кэшированию структурах см. в [ABF05], Родственные задачи. Словари (см. раздел 12.1). сортировка (см. раздел 14.1).
Глава 14. Комбинаторные задачи 465 14.3. Поиск медианы и выбор элементов Вход. Набор из п чисел (или ключей), целое число к. Задача. Найти ключ такой, что ровно к ключей больше нею (рис. 14.3). ВХОД ВЫХОД Рис. 14.3. Сортировка по среднему значению Обсуждение. Поиск медианы (элемента, среднего по местоположению) является важ- ной задачей в статистике, поскольку он возвращает более устойчивое понятие, чем среднее значение. На среднее значение доходов всех людей, опубликовавших научные работы по сортировке, оказывает сильное влияние факт присутствия среди них Билла Гейтса (см. [GP79]). но его влияние на медиану доходов сводится к нейтрализации од- ного малообеспеченного аспиранта. Задача поиска медианы является частным случаем общей задачи о выборе, в которой требуется выбрать z-й элемент в отсортированном списке. Задача выбора возникает во многих приложениях, примеры которых приводятся в следующем списке: ♦ Фильтрация элементов с резко отклоняющимися значениями. При работе с зашум- ленными данными будет полезно отбросить, скажем, 10% самых больших и самых меньших значений. Для этого выбираются элементы, соответствующие 10-му и 90-му процентилям, после чего значения, не попадающие в диапазон, ограниченный этими точками, отбрасываются. ♦ Идентификация наиболее перспективных кандидатов. В компьютерной программе для игры в шахматы мы можем быстро оценить все возможные следующие ходы и затем рассмотреть 25% самых лучших ходов более пристально. В результате фильт- рации мы получим оптимальный вариант. ♦ Применение децилей и аналогичные способы деления диапазона. Удобным способом представления распределения доходов населения является отображение зарплат людей на графике по равномерным интервалам, границы которых устанавливаются, например, по 10-му перцентилю, по 20-му перцентилю и т. д. Вычисление этих зна- чений просто сводится к выбору элементов из соответствующего интервала.
466 Часть II. Каталог алгоритмических задач ♦ Статистика порядков. Особенно интересные частные случаи задачи выбора — это поиск наименьшего элемента (&=1), наибольшего элемента (к = п) и медианного элемента (к = п/2). Среднее значение п чисел можно вычислить за линейное время, сложив все элементы и разделив получившуюся сумму на п. Но поиск медианы представляет более трудную задачу. Алгоритмы вычисления медианы можно легко обобщить к произвольной вы- борке. Перечислю вопросы, на которые следует ответить при поиске медианы и выборе элементов. ♦ Какой должна быть скорость работы? Самый простой алгоритм вычисления ме- дианы сортирует элементы за время O(nlgn). после чего возвращает элемент, зани- мающий (н/2)-ю позицию. Бесспорное достоинство этого алгоритма состоит в том, что он выдает намного больше информации, чем одна лишь медиана, позволяя вы- брать k-й элемент (для любого 1 <к<п) за постоянное время после сортировки. Но если вам требуется только медиана, то для ее поиска существуют более быстрые ал- горитмы. В частности, одним из таких алгоритмов является алгоритм на основе быстрой сор- тировки с ожидаемым временем исполнения ()(п). Этот алгоритм работает сле- дующим образом. Выбираем в наборе данных произвольный элемент и используем его для разбиения данных на два набора, один из которых содержит элементы меньшие, чем элемент-разделитель, а другой — большие. Зная размер этих наборов, мы знаем позицию элемента-разделителя в исходном наборе, а следовательно, и расположение медианы — слева или справа от разделителя. Рекурсивно выполняем эту процедуру на соответствующем наборе данных, пока не найдем медиану. Нам понадобится, в среднем, 0(1 g«) итераций, а затраты на выполнение каждой после- дующей итерации примерно равны половине стоимости предыдущей. Времена вы- полнения каждой итерации образуют геометрическую прогрессию, которая сходит- ся к линейному времени исполнения. Впрочем, если нам очень не повезет, то время исполнения может оказаться таким же, как и для быстрой сортировки, т. е. С)(гГ). Более сложные алгоритмы находят медиану за линейное время в наихудшем случае. Однако для практического применения лучше всего подойдет алгоритм с ожидае- мым линейным временем исполнения. Только обязательно выбирайте случайные элементы-разделители, чтобы избежать наихудшего случая. ♦ Как поступить, если мы видим каждый элемент только один раз? На больших на- борах данных операции выбора элементов и вычисления медианы становятся слиш- ком дорогими, т. к. для них обычно требуется выполнить несколько проходов по данным, хранящимся на внешних устройствах. А у приложений, работающих с по- токовыми данными, объем данных слишком велик для того, чтобы их сохранять. В результате их повторное рассмотрение (и, следовательно, вычисление точного значения медианы) невозможно. В таких случаях лучше построить менее объемную модель данных для дальнейшего анализа, например, на основе децилей или момен- тов распределения (где k-й момент потоках определяется как Fk = )• Одним из решений такой задачи будет произвольная выборка. Решение, сохранять ли значение, принимается на основе броска монеты, у которой вероятность выпаде-
Глава 14. Комбинаторные задачи 467 ния стороны, соответствующей сохранению, достаточно низка, чтобы не перепол- нить буфер. Скорее всего, медиана сохраненных выборок будет близкой к медиане исходного набора данных, В качестве альтернативного варианта можно выделить определенную область памяти для хранения, например, значений децилей больших блоков, после чего объединить децили, чтобы получить более точные границы. ♦ Насколько быстро можно найти моду? Кроме среднего и медианы существует еще одно понятие средней величины — мода. Мода определяется как наиболее часто встречающийся элемент набора данных. Наилучший алгоритм для вычисления мо- ды сортирует набор данных за время O(nlgn), в результате чего одинаковые элемен- ты оказываются рядом. Перебирая отсортированные данные слева направо, мы мо- жем найти длину самой протяженной последовательности одинаковых элементов и вычислить моду за общее время (?(nlg«). Необходимо заметить, что более быстрого алгоритма для вычисления моды не су- ществует, т. к. можно доказать, что задача проверки наличия двух идентичных эле- ментов в наборе имеет нижнюю временную границу Q(rrlgn). Задача поиска одина- ковых элементов эквивалентна задаче поиска нескольких мод. Существует возмож- ность, по крайней мере, теоретическая, улучшить время исполнения для больших значений моды за счет использования быстрых вычислений медианы. Реализации. Библиотека STL на языке C++ содержит универсальный метод выбора элемента (nth_eiement), реализованный на основе алгоритма с ожидаемым линейным временем исполнения. Библиотеку STL можно загрузить вместе с документацией с веб- сайта http://vvvvw.sgi.com/tech/stl/. Библиотека STL и стандартная библиотека C++ подробно описаны в книгах [Jos99], [MeyOl] и [MDS01]. Примечания Алгоритм с ожидаемым линейным временем исполнения для вычисления медианы и вы- бора элемента был разработан Хоаром (Ноаге); см. [Ноа61]. В работе [FR75] представлен алгоритм, выполняющий в среднем меньшее число сравнений. Хорошие описания алго- ритма выбора с линейным временем исполнения приведены в книгах [AHU74], [BvG99], [CLRSOI] и [Raw92], среди которых [Raw92] является наиболее информативной. Потоковые алгоритмы широко применяются для работы с большими наборами данных; подробный обзор этих алгоритмов можно найти в книге [MutO5]. Значительный теоретический интерес представляет задача определения точного количе- ства сравнений, достаточных для вычисления медианы п элементов. Алгоритм с линей- ным временем исполнения, описываемый в работе [BFP172], доказывает, что достаточно сп сравнений, но мы хотим знать, чему равно с. В работе [DZ99] доказано, что для вычис- ления медианы достаточно выполнить 2.95и сравнений. Эти алгоритмы стремятся мини- мизировать количество сравнений элементов, но не общее количество операций, и поэто- му на практике не оказываются намного быстрее существующих. Кроме того, они остав- ляют на месте наилучшую на данный момент нижнюю границу времени вычисления медианы, равную (2 + е) сравнениям. Исследование пределов комбинаторных алгоритмов для решения задач выбора представ- лено в книге [Aig88]. Оптимальный алгоритм для вычисления моды приведен в работе [DM80], Родственные задачи. Очереди с приоритетами (см. раздел 12.2), сортировка (см. раз- дел 14.1).
468 Часть II. Каталог алгоритмических задач 14.4. Генерирование перестановок Вход. Целое число п. Задача. Создать: все возможные перестановки, или случайную перестановку, или оче- редную перестановку размером п (рис. 14.4). { 1,2,3 ) { 3, 1,2 } ВХОД выход Рис. 14.4. Перестановки Обсуждение. Перестановкой называется упорядоченный набор элементов. Во многих алгоритмических задачах из этого каталога требуется найти наилучший способ упоря- дочения набора объектов. В качестве примеров можно назвать задачу коммивояжера (определение порядка посещения п городов, имеющего наименьшую стоимость), зада- чу уменьшения ширины ленты (упорядочивание вершин графа в линию таким образом, чтобы минимизировать длину самого длинного ребра) и задачу изоморфизма графа (упорядочивание вершины графа таким образом, чтобы он был идентичным другому графу). Любой алгоритм для предоставления точного решения таких задач должен соз- давать в процессе решения последовательность перестановок. Из п элементов можно создать п\ перестановок. С увеличением п количество переста- новок возрастает так быстро, что не стоит надеяться сгенерировать все перестановки для п> 12, т. к. 12! = 479 001 600. Подобные числа должны охладить пыл любого чело- века, пытающегося решить задачу методом исчерпывающего перебора, и помочь объ- яснить важность генерирования случайных перестановок. При обсуждении генерирования перестановок основным является понятие порядка, т. е. последовательности, в которой они создаются. Наиболее естественным порядком генерирования перестановок является лексикографический, т. е. такой, в котором они находились бы после сортировки. Например, лексикографический порядок перестано- вок для и = 3 будет {1. 2, 3}, {1,3,2}, {2, 1,3}. {2, 3, 1}. {3, 1,2}, {3, 2. 1}.Хотя лекси- кографический порядок эстетически более привлекателен, часто он не дает никаких определенных преимуществ. Например, при поиске в коллекции файлов порядок, в котором рассматриваются имена файлов, не имеет значения, при условии, что в ко- нечном итоге вы просмотрите их все. Более того, если лексикографический порядок не требуется, то нередко получаются более быстрые и простые алгоритмы. Для создания перестановок существуют две парадигмы: метод ранжирования и метод инкрементального изменения. Второй метод более эффективен, но первый применим
Глава 14. Комбинаторные задачи 469 для решения более широкого класса задач. Ключевым моментом метода ранжирования является определение функций Rank и Unrank на всех перестановках р и целых числах п, т, где |/?| = п и 0 < in < /?!. ♦ Функция Rank(p) определяет позицию перестановки р в данном порядке генериро- вания. Типичная функция ранжирования является рекурсивной. Например, при базовом выражении Rank) [ I}) = 0 общая формула выглядит так: Rank(p) = = (Р\ - I Н|р| - I)! + Rank{p2. .... р^). Чтобы функция была правильной, необходимо присвоить новые метки элементам меньшей перестановки, чтобы отображать удаленный первый элемент. Таким обра- зом: Rank({2, I. 3})= 1-2! + ЯаШТ({1, 2}) = 2 + 0-1! + Rank({\]) = 2. ♦ Функция Unrankfm, п) возвращает перестановку в позиции т из п\ перестановок из п элементов. Типичная функция рекурсивно определяет, сколько раз (/7-1)! встречается в т. Выражение Unrank(2,3) говорит нам, что первый элемент переста- новки должен быть "2", т. к. (2 — 1)-(3 — 1)! < 2, но (3 — 1 )-(3 — I)! > 2. Удаление (2- 1)-(3- 1)! из т оставляет меньшую задачу Unrank(G,2). Ранг 0 соответствует полному упорядочению. Полным упорядочением на двух оставшихся элементах (т. к. 2 уже было использовано) является [1, 3}, поэтому Unrank{2, 3) = {2,1, 3}. Что собой представляют фактически функции Rank и Unrank, не имеет такого большо- го значения, как то обстоятельство, что они должны быть обратными друг другу. Ины- ми словами, р = Unrank(Rank(p), п) для всех перестановок р. Определив функции ран- жирования для перестановок, мы можем решать многие родственные задачи. В част- ности: ♦ генерирование следующей перестановки Чтобы определить следующую по порядку после р перестановку, мы можем выполнить Rank{p), добавить к результату 1, после чего выполнить Unrank(p). Аналогичным образом вычисляется и перестановка перед /?: Unrank(Rank(p) — \,\р\). Выполнение функции Unrank над последователь- ностью целых чисел от 0 до п! - 1 будет равнозначно генерированию всех переста- новок; ♦ генерирование случайных перестановок Выбрав случайное целое число от 0 до п\ -1, а потом вызвав для него функцию Unrank, мы получим действительно слу- чайную перестановку: ♦ отслеживание набора перестановок. Допустим, мы хотим генерировать случайные перестановки и предпринимать некоторое действие только в том случае, когда по- лучаем новую перестановку. Для отслеживания уже созданных перестановок можно создать битовый вектор (см. раздел 12.5) из п\ битов и устанавливать в нем бит / при генерировании перестановки Unrank^i, п). Подобный метод использовался с под- множествами из к элементов в приложении для игры в лото, описанного в разде- ле 1.6. Этот метод ранжирования лучше всего подходит для небольших значений п, т. к. вы- числение п\ быстро приведет к переполнению, если только не используются арифмети- ческие операции с произвольной точностью (см. раздел 13.9). Итеративные методы включают в себя определение операции next и previous для преобразования одной пе- рестановки в другую, обычно путем обмена местами двух элементов. Здесь сложной
470 Часть II. Каталог алгоритмических задач частью является планирование таких обменов местами, чтобы перестановки не повто- рялись. пока не будут сгенерированы все перестановки. На рис. 14.4 показано упорядо- чение шести перестановок множества {I, 2, 3} с использованием одного обмена места- ми между соседними перестановками. Алгоритмы, созданные на основе итеративных методов для генерирования последова- тельности перестановок, довольно сложны, но настолько лаконичны, что их можно реализовать программой, состоящей из десятка строк. Ссылки на существующий код см. в подразделе "Реализации". Так как при инкрементальном изменении выполняется только один обмен, то эти алгоритмы могут быть чрезвычайно быстрыми — в среднем, с постоянным временем исполнения— что не зависит от размера перестановки! Сек- рет заключается в использовании «-элементного массива для представления переста- новки. чтобы облегчить обмен местами. В некоторых приложениях важно только раз- личие между перестановками. Например, в программе поиска оптимального маршрута коммивояжера методом исчерпывающего перебора стоимостью маршрута, связанного с новой перестановкой, будет стоимость предшествующей перестановки с добавлением или удалением четырех ребер. До сих пор мы предполагали, что все элементы перестановок отличаются друг от дру- га. Однако если их множество содержит дубликаты (иными словами, является муль- тимножеством), то можно сэкономить значительное время и усилия, не рассматривая одинаковые перестановки. Например, для множества {1, 1. 2, 2, 2} имеется не 120 раз- ных перестановок, а только десять. Чтобы избежать появления дубликатов, переста- новки нужно генерировать в лексикографическом порядке методом поиска с воз- вратом. Генерирование случайных перестановок представляет собой несложную, но важную задачу, с которой разработчикам приходится часто сталкиваться и с которой они не всегда справляются. Для правильного решения этой задачи нужно использовать сле- дующий алгоритм, состоящий из двух строчек кода, с линейным временем исполнения Предполагается, что функция Random[i. п] генерирует случайное целое число в диапа- зоне между z и п включительно. for i = 1 to п do cz[z] = i; for i = 1 to (z? - 1) do 5H’<7p[cz[z'], a[Random[i. «]]; Далеко не очевидно, что этот алгоритм генерирует все перестановки случайным обра- зом с равномерным распределением. Если вы так не думаете, предоставьте убедитель- ное объяснение, почему следующий алгоритм не генерирует равномерно распределен- ные перестановки: for i = 1 to п do a[z] = z; for z = 1 to (n - I) do ^M’a/?[<7[z], a\Random[ 1. «]]; Такие тонкости демонстрируют, почему нужно быть очень осторожным с алгоритмами генерации случайных чисел. Более того, мы рекомендуем подвергнуть достаточно дли- тельному испытанию любой генератор случайных чисел, прежде чем по-настоящему доверять выдаваемым им результатам. Например, сгенерируйте четырехэлементные случайные перестановки 10 000 раз и убедитесь, что количество появлений всех
Глава 14. Комбинаторные задачи 471 24 разных перестановок примерно одинаковое. Если же вы знаете, как измерять стати- стическую значимость, вы сможете оценить надежность вашего алгоритма еще точнее. Реализации. Библиотека STL на языке C++ содержит две функции для генерирования перестановок в лексикографическом порядке— next permutation и prev_permutation. В книге [KS99] описывается реализация алгоритмов для генерирования перестановок итеративным методом и в лексикографическом порядке. Коды на языке С можно за- грузить с веб-сайта http://vvww.math.mtu.edu/~kreher/cages/Src.html. Профессор Фрэнк Раски (Frank Ruskey) из университета г. Виктория в Канаде разрабо- тал сервер комбинаторных объектов (Combinatorial Object Server) для генерирования перестановок, подмножеств, разбиений, графов и других комбинаторных объектов. Сервер доступен по адресу http://theory.cs.uvic.ca/. Интерактивный интерфейс сервера позволяет указывать объекты, которые вы хотите получить. Для некоторых типов объ- ектов имеются реализации на языках С. Pascal и Java. Веб-сайт http://vvvvvv.jjj.de/fxt/ содержит процедуры на языке C++ для генерирования удивительно широкого диапазона комбинаторных объектов, включая перестановки и циклические перестановки. Книга [NW78] уже многие годы является замечательным источником информации по генерированию комбинаторных объектов. Она содержит эффективные реализации ал- горитмов (на языке FORTRAN) для генерирования случайных перестановок и переста- новок методом минимального изменения порядка. Кроме того, в ней приводятся про- цедуры для выявления циклов в перестановке. Подробности см. в разделе 19.1 10. Библиотека Combinatorica (см. [PS03]) предоставляет реализации алгоритмов (на языке программирования пакета Mathematica) для генерирования случайных перестановок и последовательностей перестановок с минимальными различиями и в лексикографиче- ском порядке. Эта библиотека также содержит процедуру поиска с возвратом для соз- дания всех несовпадающих перестановок мультимножества и поддерживает разнооб- разные групповые операции над перестановками. Подробности см. в разделе 19.1.9. Примечания Самым лучшим на сегодня справочным материалом по генерированию перестановок является одна из последних работ Дональда Кнута [КпиО5а]. Обзор, представленный в [Sed77], написан раньше, но прогресс в этой области весьма незначителен. Хорошие опи- сания можно найти в [KS99], [NW78] и [RusO3]. Методы быстрого генерирования перестановок выполняют только один обмен элементов местами для получения очередной перестановки. Алгоритм Джонсона-Троттера (см. [Joh63] и [Тго62]) удовлетворяет еще более строгому условию, а именно, что местами все- гда обмениваются только смежные элементы. Простые функции с линейным временем исполнения для ранжирования перестановок описаны в работе [MR01]. В прежние времена, когда компьютеры не были так распространены, многие пользова- лись таблицами случайных перестановок, печатаемых в специальных книгах, например, [МО63]. Рассмотренный ранее алгоритм для генерирования случайных перестановок ме- тодом обмена местами двух элементов перестановки был впервые описан в этой же книге. Родственные задачи. Генерирование случайных чисел (см. раздел 13.7), генерирова- ние подмножеств (см. раздел 14.5). генерирование разбиений (см. раздел 14 6).
472 Часть II. Каталог алгоритмических задач 14.5. Генерирование подмножеств Вход. Целое число п. Задача. Сгенерировать: все возможные подмножества, или случайное подмножество, или очередное подмножество целых чисел {I,п} (рис. 14.5). ВХОД ВЫХОД Рис. 14.5. Подмножества Обсуждение. Подмножеством называется выборка объектов, порядок расположения которых не имеет значения. Целью многих алгоритмических задач является найти наи- лучшее подмножество набора объектов. Например, в задаче о вершинном покрытии требуется найти наименьшее подмножество вершин, покрывающее все ребра графа; в задаче о рюкзаке требуется найти наиболее дорогостоящее подмножество объектов в пределах заданного общего веса; а в задаче упаковки множества требуется найти наи- меньшее подмножество подмножеств, которые в совокупности содержат ровно по од- ному вхождению каждого элемента. Множество из п элементов имеет 2" разных подмножеств, включая пустое множество и само исходное множество. Количество подмножеств возрастает экспоненциально по отношению к значению п, но значительно медленнее, чем п\ перестановок из п элемен- тов. В самом деле. т. к. 2“° = 1 048 576. то задача просмотра всех подмножеств множе- ства из 20 элементов легко выполнима. Так как 230= 1 073 741 824, то предел, несо- мненно, будет при немного больших значениях п. По определению подмножества, относительный порядок его членов не играет роли при выяснении различий между подмножествами. Таким образом, подмножества {1,2, 5} и {2, 1. 5} являются одинаковыми. Но всегда полезно поддерживать отсортированный или канонический порядок элементов подмножества, чтобы ускорить выполнение та- ких операций, как проверка двух подмножеств на идентичность. Так же, как и в случае с перестановками (см. раздез 14 /), принципиальным моментом в задаче генерирования подмножеств является установление числовой последователь- ности среди всех 2" подмножеств. Для этого существуют три основных подхода: ♦ лексикографический порядок. Лексикографический порядок означает, что элементы отсортированы, и часто является наиболее естественным способом генерирования
Глава 14 Комбинаторные задачи 473 комбинаторных объектов. Восемь подмножеств множества {I. 2. 3J располагаются в лексикографическом порядке таким образом: {}, {I}, {1,2}, {1,2.3}, {1.3}, {2}. {2, 3} и {3}. Но генерирование подмножеств в лексикографическом порядке оказы- вается на удивление трудной задачей. Поэтому не стоит использовать этот подход, если только у вас нет какой-либо важной причины для этого; ♦ код Грея. Особенно интересной и полезной последовательностью подмножеств яв- ляется порядок с минимальными различиями, в котором соседние подмножества отличаются присутствием (или отсутствием) ровно одного элемента. Такое упоря- дочение подмножеств, называющееся кодом Грея (Gray code), показано на рис. 14.5. Генерирование подмножеств в порядке кода Грея выполняется очень быстро, т. к. для этого существует удачная рекурсивная процедура. Создаем код Грея из п- 1 элементов G’„ Создаем второй код Грея, являющийся обращенной копией перво- го кода, и добавляем п к каждому подмножеству этой копии. Потом выполним кон- катенацию всех подмножеств, чтобы получить G„. Чтобы лучше понять этот про- цесс, изучите пример, представленный на рис. 14.5. Кроме этого, так как разница между подмножествами состоит только в одном эле- менте, то алгоритмы поиска методов исчерпывающего перебора на основе кодов Грея могут быть довольно эффективными. Например, при решении задачи вершин- ного покрытия таким методом нужно будет обновлять изменение в покрытии, уда- лив или добавив только одно подмножество; ♦ двоичный счет. Самый простой подход к генерированию подмножеств основан на том обстоятельстве, что любое подмножество S' определяется элементами множест- ва .S'. входящими в S'. Мы можем представить подмножество S” битовым вектором из п элементов, в котором бит / установлен в том случае, если f-й элемент множест- ва S входит в подмножество S'. Таким образом определяется взаимно однозначное соответствие между 2” двоичными последовательностями длиной п и 2" подмноже- ствами п элементов. Для и = 3 метод двоичного счета генерирует подмножества в следующем порядке: {}, {3}, {2}. {2. 3}. {1}, {1.3}, {1.2}, {1. 2. 3}. Это двоичное представление является ключом к решению всех задач генерирования подмножеств Чтобы генерировать все подмножества, мы просто ведем счет от 0 до 2" - 1. Для каждого целого числа последовательно маскируем биты и составляем подмножество из элементов, соответствующих установленным битам. Чтобы соз- дать следующее или предшествующее подмножество, просто увеличиваем или уменьшаем на единицу число текущего подмножества. Процедура маскирования как раз и является функцией, обратной ранжированию подмножества, в то время как функция ранжирования создает двоичное число, в котором установленные биты соответствуют элементам X, после чего преобразует это двоичное число в целое число. Чтобы создать случайное подмножество, можно сгенерировать случайное целое число в диапазоне от 0 до 2” — 1 и выполнить по нему операцию, обратную ранжи- рованию. Но в зависимости от способа округления, применяемого в генераторе слу- чайных чисел, некоторые подмножества могут никогда не появиться. Будет намного лучше бросить монету п раз и по результату /-го броска решить, включать или нет
474 Часть II. Каталог алгоритмических задач элемент i в подмножество. Бросок монеты можно надежно смоделировать, генери- руя случайное действительное или очень большое целое число и проверяя, превы- шает ли оно центральное значение своего диапазона. Таким образом, для представ- ления подмножеств можно использовать массив булевых значений из п элементов, соответствующий целому числу с заранее наложенной маской. На практике необходимость генерирования подмножеств часто возникает в работе с двумя тесно связанными друг с другом структурами данных — /^-подмножествами и строками. ♦ Вместо создания всех подмножеств нам могут требоваться только подмножества из к элементов. Число таких подмножеств равно (£), что значительно меньше, чем 2", особенно для небольших значений к. Самым лучшим способом конструирования всех /r-подмножеств будет генерирова- ние их в лексикографическом порядке. Функция ранжирования основана на том об- стоятельстве. что существует ('£_{) /^-подмножеств, у которых наименьший элемент равен f. Отсюда мы можем определить наименьший элемент в т-м /^-подмножестве п элементов. Далее рекурсивно делаем то же самое с последующими элементами подмножества. Подробности см. в подразделе "Реализации". ♦ Генерирование всех подмножеств равнозначно генерированию всех 2" строк из зна- чений ИСТИНА и ЛОЖЬ. Те же самые основные методы применимы к задаче гене- рирования всех случайных строк на алфавитах размером а, за исключением, что общее количество строк будет равным а". Реализации. Генераторы как подмножеств, так и А-подмножеств в лексикографиче- ском порядке и в порядке кода Грея рассматриваются в книге [K.S99], Реализации этих генераторов на языке С можно загрузить с веб-сайта http://wvvvv.math.mtu.edu/ ~kreher/cages/Src.htmI. Профессор Фрэнк Раски (Frank Ruskey ) из университета г. Виктория в Канаде разрабо- тал сервер комбинаторных объектов для генерирования перестановок, подмножеств, разбиений, графов и других комбинаторных объектов. Сервер доступен по адресу http://theory.cs.uvic.ca. Интерактивный интерфейс сервера позволяет указывать объек- ты. которые вы хотите получить. Для некоторых типов объектов имеются реализации на языках С, Pascal и Java. Веб-сайт http://wvvvv.jjj.de/fxt/ содержит процедуры на языке C++ для генерирования удивительно широкого диапазона комбинаторных объектов, включая подмножества и ^-подмножества. Книга [NW78] уже многие годы является замечательным источником информации по генерированию комбинаторных объектов. Она содержит эффективные реализации ал- горитмов (на языке FORTRAN) для создания случайных подмножеств и для генериро- вания подмножеств в порядке кода Грея и в лексикографическом порядке. Кроме того, в ней приводятся процедуры для создания случайных /г-подмножеств и для генериро- вания подмножеств в лексикографическом порядке. Алгоритм 515 (см. [BL77]) из кол- лекции АСМ реализован на языке FORTRAN и предназначен для генерирования /r-подмножеств в лексикографическом порядке. Подробности см. в разделе 19.1.3.
Глава 14. Комбинаторные задачи 475 Библиотека Combinatorica (см. [PS03J) предоставляет реализации алгоритмов (на языке программирования пакета Mathematica) для создания случайных подмножеств и для генерирования последовательностей подмножеств с помощью Грея, методом двоично- го счета и в лексикографическом порядке. Также предоставляются процедуры для создания случайных ^-подмножеств и для генерирования последовательностей подмножеств в лексикографическом порядке. Дополнительную информацию по Combinatorica см. в разделе 19.1.9. Примечания Самым лучшим на сегодня справочным материалом по генерированию подмножеств яв- ляется одна из последних работ Дональда Кнута [КпиО5Ь]. Хорошие описания можно найти в [KS99], [NW78] и [Rus03]. В книге [Wil89] приводится более свежая информация, чем в [NW78], в том числе содержится подробное обсуждение современных проблем ге- нерирования кода Грея. Первоначально коды Грея были разработаны для обеспечения надежности передачи циф- ровых сигналов по аналоговому каналу (см. [Gra53]). При установке кодовых слов в по- рядке кода Грея i-e слово лишь слегка отличается от (/ +• 1)-го слова, так что незначитель- ные колебания уровня аналогового сигнала искажают лишь небольшое количество бит. Коды Грея довольно точно соответствуют гамильтоновым циклам на гиперкубе. В работе [Sav97] представлен отличный обзор кодов Грея (упорядочения по принципу минималь- ного изменения) для большого класса комбинаторных объектов, включая подмножества. Популярная головоломка Spinout, выпускаемая компанией ThinkFun (ранее носившей на- звание Binary Arts Corporation), решается на основе идей кодов Грея. Родственные задачи. Генерирование перестановок (см. раздел 14.4). генерирование разбиений (см. раздел 14.6). 14.6. Генерирование разбиений Вход. Целое число п. Задача. Создать: все возможные разбиения целого числа или множества размером п. или случайное разбиение, или очередное разбиение (рис. 14.6). Обсуждение. Существует два типа комбинаторных объектов, обозначаемых словом "разбиение", а именно разбиения целых чисел и разбиения множеств. Вам необходимо понимать разницу между ними. ♦ Разбиения целого числа п представляют собой мультимножества ненулевых целых чисел, сумма которых равна п. Например, число 5 имеет семь разных разбиений: {5}, {4, 1}, {3, 2}. {3. 1. 1}, {2, 2, 1}, {2, 1, 1, 1} и {1. 1, 1, 1, 1}. Одним из интерес- ных приложений, требующим генерирования разбиений целых чисел, с которым мне пришлось сталкиваться, было моделирование расщепления ядра. При расщеп- лении атома его ядро разбивается на множество кластеров меньшего размера. Об- щее количество частиц в этом множестве кластеров должно быть равным первона- чальному количеству частиц ядра. Таким образом, целочисленные разбиения этого первоначального количества частиц представляют все возможные способы расщеп- ления атома.
47 6 Часть II. Каталог алгоритмических задач вход ВЫХОД Рис. 14.6. Разбиения ♦ Разбиения множества разделяют множество {1, .... и} на непустые подмножества. Например, для множества п = 4 существует 15 разных разбиений: {1234}, {123,4}. {124,3}, {12.34}, {12,3,4}, {134,2}, {13,24}, {13,2.4}, {14,23}, {1.234}. {1.23.4}. {14.2,3}, {1.24.3}, {1,2,34} и {I.2.3.4}. Некоторые алгоритмические задачи, включая задачу раскраски вершин/ребер и задачу поиска компонент связности, имеют в ка- честве результатов разбиения множества. Для краткости мы будем называть разбиения целых чисел просто разбиениями, а для другого вида употреблять полное название — разбиение множеств. А полное название разбиения целого числа будем употреблять в тех случаях, где нужно избежать недора- зумения, какое именно из разбиений имеется в виду. Количество разбиений возрастает экспоненциально по отношению к п. Так, для целого числа п = 20 существует всего лишь 627 разбиений. Более того, возможно перечислить даже все разбиения числа л = 100, т. к. их количество равно 190 569 292. Самым простым способом создания разбиений будет генерирование их в порядке, об- ратном лексикографическому. Первым разбиением будет само число {и}. Общее пра- вило состоит в вычитании 1 из наименьшего элемента разбиения, большего единицы, с последующим объединением всех единиц с целью соответствия новому наименьшему элементу, большему единицы. Например, разбиением, следующим после {4. 3, 3.3, 1.1, 1. 1}. будет {4, 3. 3, 2, 2, 2, 1}, т. к. пять единиц, получаемые после операции вычита- ния 3-1=2, лучше всего объединяются в виде элемента {2. 2. 1}. Когда мы получаем разбиение, все члены которого являются единицами, мы завершаем первый проход по всем разбиениям. Этот алгоритм достаточно сложен для программирования, поэтому следует рассмот- реть использование одной из готовых реализаций, упоминаемых далее. В любом слу- чае, проверьте, что используемая реализация выдает ровно 627 разных разбиений для п = 20. Задача генерирования случайных разбиений с равномерным распределением более сложна, чем аналогичная задача для перестановок или подмножеств. Это потому, что
Глава 14 Комбинаторные задачи 477 выбор первого (т. е. самого большого) элемента разбиения сильно влияет на количест- во разбиений, которые можно сгенерировать. Обратите внимание, что независимо от размера л, существует только одно разбиение п, наибольший элемент которого равен I. Количество разбиений числа и с наибольшим элементом, не привышающим к, задается следующим рекуррентным соотношением: Рн,к ~ Рп kj< "* Рц,к - I с двумя граничными условиями Pf.K К = РУ УЛ и Р„ i = I. Посредством этой функции можно выбрать наибольший элемент случайного разбиения с требуемой вероятностью, после чего рекурсивно создать случайное разбиение целиком. При генерировании случайных разбиений наблюдается тенденция появления огромно- го количества сравнительно небольших элементов, как можно видеть в диаграмме Феррерса (Ferrers diagram) на рис. 14.7. Рис. 14.7. Диаграмма Феррерса для случайного разбиения числа п = 1000 Каждая строка диаграммы соответствует одному элементу разбиения, размер которого представлен количеством точек в данной строке. Разбиения множества можно генерировать, используя методы, подобные методам генерирования разбиений целых чисел. Каждое разбиение множества представляется в виде функции ограниченного роста at,..., a„, где Ы|=0 и a, < 1 + тах(<7|.а,) для / = 2,..., п. Каждая цифра определяет подмножество (или блок) разбиения, в то время как условие роста обеспечивает сортировку блоков в каноническом порядке по наи- меньшему элементу в каждом блоке. Например, функция ограниченного роста 0. 1, 1, 2,0,3, 1 определяет такое разбиение множества: {{1, 5}.{2, 3. 7},{4},{6}}. Так как между разбиениями множеств и функциями ограниченного роста существует взаимно однозначное соответствие, мы можем воспользоваться лексикографическим порядком функций ограниченного роста для упорядочивания разбиений множеств. В самом деле. 15 разбиений множества {1, 2, 3, 4}, перечисленных в определении раз- биения множества в начале рассмотрения, расположены в соответствии с лексикогра- фическим порядком их функций ограниченного роста (вы можете в этом убедиться). Для генерирования случайных разбиений множества можно использовать метод дво- ичного счета, аналогичный используемому для генерирования разбиений целых чисел. Числами Стирлинга второго рода (£) называется количество разбиений «-элементного
478 Часть II Каталог алгоритмических задач множества {1, .... и} на/: непустых подмножеств. Для их вычисления используется сле- дующее рекуррентное соотношение: {*}={*:!}+^7'} с краевыми условиями {”} = {"} = 1. Подробности см. в подразделе "Реализации". Реализации. Реализации, описываемые в книге [K.S99], генерируют разбиения как целых чисел, так и множеств в лексикографическом порядке, включая функции ранжи- рования. Реализации этих генераторов на языке С можно загрузить с веб-сайта http://www. math. mtu.edu/~kreher/cages/Src.html. Профессор Фрэнк Раски (Frank Ruskey) из университета г. Виктория в Канаде разрабо- тал сервер комбинаторных объектов (Combinatorial Object Server) для генерирования перестановок, подмножеств, разбиений, графов и других комбинаторных объектов. Сервер доступен по адресу http://theory.cs.uvic.ca/. Интерактивный интерфейс сервера позволяет указывать объекты, которые вы хотите получить. Для некоторых типов объ- ектов имеются реализации на языках С. Pascal и Java. Веб-сайт http://www.jjj.de/fxt/ содержит процедуры на языке C++ для генерирования удивительно широкого диапазона комбинаторных объектов, включая разбиения целых чисел. Книга [NW78] уже многие годы является замечательным источником информации по генерированию комбинаторных объектов. Она содержит эффективные реализации (на языке FORTRAN) алгоритмов для генерирования случайных и последовательных раз- биений целых чисел и множеств и таблиц Юнга. Подробности см. в разделе 19.1.10. Библиотека Combinatorica (см. [PS03] предоставляет реализации алгоритмов (на языке программирования пакета Mathematica) для генерирования случайных и последова- тельных разбиений целых чисел, строк и таблиц Юнга, а также средства для подсчета и манипулирования этими объектами. Подробности см. в разделе 19.1.9. Примечания Самым лучшим на сегодня справочным материалом по алгоритмам для генерирования разбиений как целых чисел, так и множеств является одна из последних работ Дональда Кнута [КпиО5Ь]. Хорошие описания можно найти, например, в [KS99], [NW78], [RusO3] и [PS03]. Основным справочником по разбиениям целых чисел и родственным темам явля- ется книга [And98], а книга [АЕ04] представляет собой доступное введение в данную об- ласть. Разбиения целых чисел и множеств являются частными случаями разбиений мульти- множеств или разбиений множеств с не обязательно разными элементами. В частности, разные разбиения мультимножества {1, 1, 1, ..., 1} в точности соответствуют разбиениям целых чисел. Разбиения мультимножеств обсуждаются в работе Дональда Кнута [КпиО5Ь]. Длинная история генерирования комбинаторных объектов подробно излагается в работе Доначьда Кнута [КпиОб]. Особенно интересна связь между разбиениями множеств и японским ритуалом сжигания благовоний, а также связь между всеми 52 разбиениями множества для п = 5 и разными главами самого старого известного романа "Повесть о Гэндзи".
Гпава 14 Комбинаторные задачи 479 Родственными комбинаторными объектами являются таблицы Юнга и композиции целых чисел, хотя их возникновение в реальных приложениях маловероятно. Алгоритмы гене- рирования объектов обоих типов представлены в [NW78], [Rus03] и [PS03], Таблицы Юнга представляют собой двумерные формирования целых чисел в диапазоне {I,п}, где количество элементов в каждой строке определяется одним из разбиений це- лого числа п. Элементы каждой строки и каждого столбца отсортированы в возрастаю- щем порядке, а строки выровнены по левому краю. Это понятие охватывает широкий на- бор разнообразных структур в виде частных случаев. Они обладают многими интересны- ми свойствами, включая взаимно однозначное соответствие между парами таблиц и перестановок. Композиции целых чисел представляют всевозможные способы распределения множества из и одинаковых шаров по к различным урнам. Например, три шара можно разместить по двум урнам таким образом: {3, 0}, {2, I}, {I, 2} или {0, 3}. Композиции легче всего гене- рировать последовательно в лексикографическом порядке. Для генерирования случайных композиций выбираем случайное (к I )-подмножество из п + к-\ элементов с помощью алгоритма, описанного в разделе 14.5. после чего подсчитываем количество оставшихся элементов между выбранными. Например, для к = 5 и п = 10, (5 I )-подмножество [1, 3, 7, 14} множества {1, ..., (n + А-1) = 14} определяет композицию {0. 1. 3, 6, 0}, т. к. нет элементов ни слева от элемента 1, ни справа от элемента 14. Родственные задачи. Генерирование перестановок (см. раздел 14.4), генерирование подмножеств (см. раздел 14.5). 14.7. Генерирование графов Вход. Параметры, описывающие граф, включая количество вершин п и количество ре- бер т или вероятность наличия ребрар. Задача. Сгенерировать: все графы, удовлетворяющие заданным параметрам, или слу- чайный граф, или очередной граф (рис. 14.8). N = 4 Connected unlabeled вход выход Рис. 14.8. Генерирование графов Обсуждение. Задача генерирования графов обычно возникает при создании тестовых данных для программ. Возможно, у вас имеются две разные программы для решения
480 Часть II. Каталог алгоритмических задач одной и той же задачи, и вы хотите узнать, какая из них работает быстрее, или удосто- вериться, что обе дают одинаковый результат. Другим приложением задачи является проверка, обладают ли данные графы определенным свойством, или насколько оно распространено среди них. В справедливость теоремы четырех цветов легче поверить после демонстрации четырехцветной раскраски всех планарных графов на 15 вер- шинах. Еще одно приложение задачи генерирования графов возникает при проектировании сетей. Допустим, что вы должны объединить в сеть десять компьютеров, использовав кабель как можно меньшей длины. Сеть при этом должна оставаться работоспособной после одновременного выхода из строя до двух узлов. Одним из подходов в данном случае будет проверка всех возможных конфигураций сети с данным числом соедине- ний/ребер, пока не будет найдена конфигурация, отвечающая требованиям. Но для се- тей с большим количеством узлов такой подход неприемлем и, скорее всего, потребу- ются эвристические методы, наподобие метода имитации отжига. Задача генерирования графов усложняется многими факторами. Прежде всего, нужно точно знать, граф какого типа требуется создать. Некоторые важные свойства графов изображены на рис. 5.2. При генерировании графов необходимо ответить на следую- щие вопросы. ♦ Требуются помеченные или непомеченные графы? Иными словами, имеют ли зна- чение при сравнении графов названия вершин. При генерировании помеченных графов мы создаем все возможные маркировки для всех возможных топологий гра- фов. А при генерировании непомеченных графов нам достаточно создать только по одному представителю каждой топологии и можно игнорировать метки. Например, из трех вершин можно создать только два связных непомеченных графа— тре- угольник и простой путь. Но из тех же трех вершин можно создать четыре связных помеченных графа— один треугольник и три трехвершинных пути, отличающиеся друг от друга названием центральной вершины. В общем, намного легче создавать помеченные графы. Но их так много, что очень быстро мы будем завалены изо- морфными копиями одних и тех же графов. ♦ Требуются ориентированные или неориентированные графы? Большинство естест- венных алгоритмов генерируют неориентированные графы. Эти графы можно пре- образовать в ориентированные, выполнив ориентацию ребер по броску монеты. Из любого графа можно сделать бесконтурный орграф, упорядочив случайным образом вершины в линию и направив каждое ребро слева направо. При таком разнообразии вариантов вы должны все обдумать и решить, генерируются ли все графы единооб- разно и случайным образом, а также насколько важен для вас способ генериро- вания Кроме прочего, нужно определиться с тем, что вы считаете случайным графом. Суще- ствуют три основные модели случайных графов, и все они генерируют графы согласно разным вероятностным распределениям: ♦ генерирование случайных ребер. Параметры этой модели задаются вероятностью наличия ребра р. Обычно значение р устанавливается равным 1/2, хотя для генери- рования случайных разреженных графов можно использовать меньшие значения. Для каждой пары вершин х и у решение о добавлении ребра (х. у) принимается по
Глава 14. Комбинаторные задачи 481 результату броска монеты. При р= 1/2 будут сгенерированы с одинаковой вероят- ностью все помеченные графы: ♦ выбор случайных ребер. Параметры этой модели задаются требуемым количеством ребер т. Модель генерирует граф, выбирая т разных ребер случайным образом с равномерным распределением. Одним из способов сделать это будет создавать случайные пары вершин (х, у) и соединять их ребрами, если данная вершина еще не в графе. Альтернативным подходом будет создать набор из (^возможных ребер и случайным образом выбрать из них /н-подмножество, как рассматривается в разде- ле 14.5\ ♦ избирательное присоединение (preferential attachment). Согласно модели "богатые становятся богаче", более вероятно, что новые ребра будут направлены к вершинам высокого уровня, а не низкого. Посмотрите, как добавляются новые ссылки (ребра) к графу веб-страниц. По любой реалистичной сетевой модели генерирования более вероятно, что очередная ссылка будет на Google, чем на http://www.cs.sunysb.cilu/ ~algorith . Выбор следующей соседней вершины с вероятностью, пропорциональ- ной ее степени, порождает графы со свойствами степенной зависимости, которыми обладают многие реальные сети. Какая из этих моделей подходит лучше всего для вашего приложения? Скорее всего, никакая. Случайные графы плохо структурированы по определению. Но графы часто используются для моделирования высокоструктурированных взаимоотношений. Инте- ресные и легко выполнимые эксперименты на случайных графах, как правило, не от- ражают реальное положение вещей. Альтернативой случайным графам являются "органические" графы, которые отобра- жают взаимоотношения между реальными объектами. Превосходным источником "органических" графов является рассматриваемая далее база графов Stanford GraphBase. В Интернете имеется много источников взаимоотношений, которые можно превратить в интересные "органические графы", приложив немного усилий по про- граммированию. Возьмем, например, граф, определяемый набором веб-страниц, в ко- тором ребро определяется гиперссылкой с одной страницы на другую. Другой при- мер— граф, определяемый железными дорогами или авиарейсами, где вершины пред- ставляют станции или аэропорты, а ребра— прямую связь между ними. Наконец, каждая большая компьютерная программа определяет граф вызовов процедур, в кото- ром вершины представляют процедуры, а ребра — вызовы одних процедур из других. Два класса графов имеют особенно интересные алгоритмы генерирования: ♦ деревья Коды Прюфера (Prufer codes) предоставляют простой способ ранжирования помеченных деревьев и, таким образом, решения всех стандартных задач генериро- вания (см. раздел 14.4). Для п вершин существует ровно п ~ помеченных деревьев и столько же строк длиной п - 2 существует на алфавите {1.2,.... п}. Ключом к взаимно однозначному соответствию Прюфера является то обстоятельст- во. что каждое дерево имеет, по крайней мере, две вершины степени 1. Таким обра- Пожалуйста. создайте ссылку с нашей домашней страницы на наш веб-сайт, чтобы исправить эту ситуацию. 16 Зак 3741
482 Часть II. Каталог алгоритмических задач зом, в любом помеченном дереве вершина v, соединенная с листом с наименьшей меткой, является четко определенной. Пусть 5Ь первый символ кода, — это верши- на V. Удалим связанный лист и будем повторять эту процедуру до тех пор, пока не останутся только две вершины. Таким образом мы получим уникальный код S для любого помеченного дерева, который можно использовать для ранжирования дере- ва. Чтобы построить дерево по коду, воспользуемся тем обстоятельством, что сте- пень вершины v дерева на один больше, чем количество вхождений вершины v в код S. Листом с наименьшей меткой будет целое число, отсутствующее в S, которое совместно с S'] определяет первое ребро дерева. Применяя индукцию, получаем все дерево; ♦ графы с фиксированной степенной последовательностью. Степенной последова- тельностью (degree sequence) графа G называется разбиение целого числа р = (р\,.... /?„), где р, является степенью /-й вершины с наивысшей степенью графа. Так как каждое ребро учитывается при подсчете степени двух вершин, тор является разбиением целого числа 2т, где т является количеством ребер в графе. Не все разбиения соответствуют степенным последовательностям графов. Но суще- ствует рекурсивный метод, который создает граф для данной степенной последова- тельности, если такой граф вообще существует. Если разбиение выполнимо, то вершину с наивысшей степенью V| можно соединить с вершинами со степе- нями, наивысшими из оставшихся, или вершинами, соответствующими частям p2,—,pPi+i. Удалив р, и уменьшив р2,...,рл+1 на единицу, мы получаем меньшее разбиение, которое мы обрабатываем рекурсивно. Если мы завершим процедуру, не создав отрицательных чисел, то разбиение является осуществимым. Так как мы все- гда соединяем вершину наивысшей степени с другими вершинами высокой степени, то важно после каждой итерации переупорядочить элементы разбиения по размеру. Хотя этот метод является детерминистическим, с его помощью из графа G можно создать полуслучайную коллекцию графов, используя операции обмена ребер. До- пустим, что ребра (х, у) и (и; z) входят в граф G, а ребра (х. vv) и (у, z) — нет. Обме- няв эти пары ребер, мы получим другой (не обязательно связный) граф, не меняя степень ни одной из вершин. Реализации. База графов Stanford GraphBase (см. [Knu94]) является, пожалуй, самым хорошим генератором экземпляров графов для использования их в качестве тестовых данных для других программ. Она содержит графы, полученные путем обработки взаимоотношений персонажей известных романов, статей из словаря синонимов Род- жета. визуальных характеристик "Моны Лизы", графов-расширителей, а также модели экономики США. Она также содержит процедуры генерирования двоичных деревьев, произведений графов и других операций на графах основных типов. Наконец, посколь- ку в ней используются аппаратно-независимые генераторы случайных чисел, создан- ные с ее помощью случайные графы можно воспроизвести на других машинах, что по- зволяет использовать их в качестве входных данных при экспериментальном сравне- нии алгоритмов. Дополнительную информацию см. в разделе 19.1.8. Библиотека Combinatorica (см. [PS03]) содержит генераторы (на языке пакета Mathematica) для таких типов графов, как "звезда" и "колесо", для полного графа, для
Глава 14. Комбинаторные задачи 483 случайных графов и деревьев, а также для графов с заданной степенной последова- тельностью. Кроме этого, библиотека включает в себя операции создания более слож- ных графов на основе этих, в том числе объединение и произведение графов. Сервер комбинаторных объектов (http://theory.cs.uvic.ca) предоставляет процедуры для генерирования как свободных, гак и корневых деревьев. В работе [VL05] описывается реализация на языке C++ алгоритма генерирования про- стых связных графов с указанной степенной последовательностью. Код можно загру- зить с веб-страницы http://www.liafa.jussieu.fr/~fabien/generation. Программа Nauty для тестирования графов на изоморфизм (см. раздел 16.9) содержит набор программ для генерирования неизоморфных графов, а также специальные гене- раторы для создания двудольных графов, ориентированных графов и мультиграфов. Все эти программы можно загрузить с веб-страницы http://cs.anu.edu.au/~bdm/nauty. Математики Брендан Маккей (Brendan McKay) (http://cs.anu.edu.au/~bdm/data) и Гор- дон Ройли (Gordon Royle) (http://people.csse.uwa.edu.au/gordon/data.html) предостав- ляют на своих веб-страницах исключительно полные каталоги нескольких семейств графов и деревьев, размер которых ограничен лишь разумным количеством вершин. В книге [NW78] предоставляются эффективные процедуры (на языке FORTRAN) пере- числения всех помеченных деревьев с помощью кодов Прюфера и создания случайных непомеченных корневых деревьев. Подробности см. в разделе 19.1.10. В книге [KS99] описывается алгоритм на языке С для генерирования помеченных деревьев. Реализа- ции можно загрузить с веб-сайта http://www.math.mtu.edu/~kreher/cages/Src.html. Примечания На тему генерирования случайных графов с равномерным распределением написано мно- жество книг. Среди прочих обзоров можно назвать [Gol93] и [Tin90J. С задачей генериро- вания классов графов тесно связана задача их подсчета. Обзор достижений в этой области приведен в книге [НР73]. Самым лучшим на сегодня справочным материалом по генерированию деревьев является одна из последних работ Дональда Кнута [КпиОб]. Взаимно однозначное соответствие между (п - 2)-строками и помеченными деревьями было установлено немецким матема- тиком Хайнцем Прюфером (Heinz Prufer) в 1918 г. (см. [Prul8]). В теории случайных графов пороговые законы определяют плотность ребер, при которой становится высокой вероятность существования таких свойств, как связность. Изложение теории случайных графов можно найти в книгах [BolO 1 ] и [JLROO]. Модель избирательного присоединения графов возникла сравнительно недавно при изу- чении сетей. Введение в эту интересную область содержится в книгах [ВагОЗ] и [Wat04]. Разбиение целого числа является графическим, если существует простой граф с такой степенной последовательностью. В работе [EG60] приводится доказательство, что сте- пенная последовательность является графической тогда и только тогда, когда для любого целого числа г < п соблюдается следующее условие: Г п У'ф, <r(r-l)+ min(r<7,) ;=1 >=r+l Родственные задачи. Генерирование перестановок (см. раздел 14.4), изоморфизм гра- фов (см. раздел 16.9).
484 Часть II. Каталог алгоритмических задач 14.8. Календарные вычисления Вход. Календарная дата d, заданная месяцем, днем и годом. Задача. На какой день недели выпадаете/ в данной календарной системе (рис. 14.9)? 5773 Tevcth 8 (Hebrew) December 21, 2012 ? 1434 saiar 7 (isiamio (Gregorian) 1934 Agrahayana 30 (Indian Civil) 13.0.0.0 (Mayan Long Count) ВХОД ВЫХОД Рис. 14.9. Календарные вычисления Обсуждение. Во многих бизнес-приложениях требуется выполнять календарные вы- числения. Например, возможно нужно вывести календарь для определенного месяца и года или вычислить, в какой день недели или года произойдет какое-то событие. Важ- ность календарных вычислений была ярко продемонстрирована "ошибкой 2000 года", присутствовавшей в старых программах, в которых для обозначения года использова- лись только две последние цифры. Более сложные вопросы возникают в международных приложениях, т. к. разные наро- ды и этнические группы используют разные календарные системы. Некоторые из этих календарных систем, такие как используемый в большинстве стран мира григориан- ский календарь, основаны на солнечном цикле, в то время как другие календари, на- пример иудейский календарь, основаны на лунных циклах. А смогли бы вы назвать сегодняшнюю дату по китайскому календарю? Календарные вычисления отличаются от других задач в этой книге, потому что кален- дари являются историческими объектами, а не математическими. В области календар- ных вычислений алгоритмические вопросы касаются установления правил календар- ной системы и правильной их реализации, а не разработки эффективных методов вы- числения. Подход, лежащий в основе всех календарных систем, заключается в выборе начальной точки и ведении отсчета от этой точки. Разные календарные системы отличаются друг от друга специфическими правилами, определяющими конец одного месяца или года и начало другого. Для реализации календаря требуются две функции, одна из которых по данной дате возвращает количество дней, прошедших от начальной точки, а другая по данному целому числу п возвращает дату календаря, отстоящей ровно на п дней от точки начала отсчета. Эти функции аналогичны функциям ранжирования для комбина- торных объектов, таких как перестановки (см. раздел 14 4). Главным источником проблем в календарных системах является то обстоятельство, что в солнечном году не целое число дней. Чтобы поддерживать синхронизацию календар-
Глава 14. Комбинаторные задачи 485 ного года с реальным, время от времени в него нужно вносить поправки в виде висо- косных дней. Но так как продолжительность солнечного года составляет 365 дней, 5 часов, 49 минут и 12 секунд, то добавление високосного дня через каждые четыре года делает корректируемый календарный год на 10.8 минут длиннее. В первоначальном юлианском календаре эти лишние минуты не принимались во вни- мание, и к 1582 году накопилось отставание в Ю дней. Тогда Папа Григорий XIII ввел в действие исправленный календарь, названный григорианским, который используется по сей день. Исправления состояли в одноразовом переносе даты календаря на Ю дней вперед и нового правила добавления високосных дней в годы столетий — в годы, кратные 400, но не 100. Годы, кратные четырем, по-прежнему оставались високосны- ми. По некоторым сведениям, введение нового календаря сопровождалось бунтами среди населения. Люди считали, что их жизнь укоротилась на десять дней. Вне преде- лов влияния католической церкви принятие нового календаря проходило медленно. В Англии и Америке григорианский календарь был принят в 1752 г., а в Турции — в 1927 г. Правила для большинства календарных систем достаточно сложны, вследствие чего будет разумным воспользоваться уже готовым кодом из надежного источника, вместо того, чтобы пытаться писать свою реализацию. Существует много алгоритмов типа "удиви своих друзей", посредством которых можно в уме высчитать день недели для определенной даты. Но такие алгоритмы часто рабо- тают правильно только в определенном столетии, и их не следует реализовывать на компьютере. Реализации. Библиотеки процедур для работы с календарями широко доступны как на языке С, так и Java. Библиотека Boost предоставляет надежную реализацию григориан- ского календаря на языке C++. Код можно загрузить с веб-страницы http:// www.boost.org/doc/libs/l_43_0/doc/html/date_time.htnil. Стандартный пакет утилит Java java.util содержит класс calendar, который реализует григорианский календарь. Любой из этих реализаций будет, скорее всего, достаточно для большинства прило- жений. В книге [RD01] описаны алгоритмы для разных календарей, включая григорианский, китайский, индийский, мусульманский и иудейский, а также календари, представляю- щие исторический интерес. Реализация этих календарей на языке Common Lisp. Java и Mathematica называется Calendrical и содержит процедуры преобразования дат между разными календарными системами, вычисления дня недели, а также определения свет- ских и религиозных праздников. Пакет Calendrical является, пожалуй, самым обшир- ным и надежным источником существующих календарных процедур. Подробности см. на веб-сайте http://calendarists.com. Реализации интернациональных календарей, написанные на языках С и Java, доступны на веб-сайте SouceForge (http://sourceforge.net), но об их надежности ничего неизвест- но. Чтобы найти требуемый календарь, выполните поиск по соответствующему ключу, например "Gregorian calendar" для григорианского календаря. Примечания Всесторонний обзор алгоритмов для календарных вычислений представлен в работах [DR90] и [RDC93], на основе которых была написана книга [DR01], содержащая алгорит-
486 Часть II. Каталог алгоритмических задач мы, по крайней мере, для 25 национальных и исторических календарей. В книге [DR02] представлены календарные таблицы на 300 лет, с 1900 г. по 2200 г. Некоторые люди обеспокоены тем, что 21 декабря 2012 г. может настать конец света. Опасение основано на том обстоятельстве, что эта дата представляет дату окончания цик- ла по календарю майя и переход на новый цикл 13.0.0.0.0. Читателям моей книги не сле- дует переживать по этому поводу, т. к. я не посвятил бы столько усилий ее написанию, если бы конец света был так близок. Авторитетное описание календаря майя дается в кни- ге [RD01]. Родственные задачи. Арифметические операции с произвольной точностью (см. раз- дел 13.9), генерирование перестановок (см. раздел 14.4). 14.9. Календарное планирование Вход. Бесконтурный орграф G = (V, Е), в котором вершины представляют задания, а ребро (и, v) отражает тот факт, что задание и должно быть завершено перед заданием v. Задача. Составить такое расписание выполнения всех заданий, которое минимизирует время выполнения или количество процессоров (рис. 14.10). ВХОД ВЫХОД Рис. 14.10. Задача календарного планирования Обсуждение. Задача разработки расписания, удовлетворяющего набору ограничиваю- щих условий, является фундаментальной для многих приложений. Распределение за- даний между процессорами является критическим аспектом любой системы парал- лельной обработки данных. Плохое планирование может привести к простаиванию всех процессоров, кроме того, который выполняет задание, представляющее собой уз- кое место. Другими примерами задач календарного планирования являются выдача заданий рабочим, размещение студенческих групп по аудиториям или составление расписания экзаменов. Задачи календарного планирования различаются по типам ограничивающих условий и видам составляемых расписаний. Некоторые из рассматриваемых в каталоге задач имеют отношение к различным типам календарного планирования. Говоря более кон- кретно: ♦ посредством топологической сортировки можно создать расписание, отвечающее ограничивающим условиям очередности. Подробности см. в разделе 15.2',
Глава 14. Комбинаторные задачи 487 ♦ паросочетание в двудольных графах можно использовать для сопоставления набора заданий с рабочими, имеющими навыки, требуемые для их выполнения. Подробно- сти см. в разделе 15.6; ♦ с помощью раскраски ребер и вершин можно сопоставить набор заданий с времен- ными периодами таким образом, чтобы избежать попадания конфликтных заданий в один и тот же временной интервал. Подробности см. в разделах 16.7 и 16.8; Ь алгоритм коммивояжера можно использовать, чтобы выбрать оптимальный мар- шрут для посещения указанных адресов разносчиком пиццы. Подробности см. в разделе 16.4; ♦ алгоритм поиска эйлерова цикла можно использовать для построения наиболее эф- фективного пути следования снегоуборочной машины, проходящей по всем задан- ным улицам. Подробности см. в разделе 15.7. Рассмотрим задачу календарного планирования с условиями очередности для бескон- турных орграфов. Допустим, что вы разбили большую работу на несколько меньших заданий. Для каждого задания известно время, требуемое для его выполнения (или, возможно, верхний временной предел). Кроме этого, для каждой пары заданий извест- но, необходимо ли, чтобы одно из них выполнялось перед другим. Чем меньше огра- ничивающих условий, тем лучшее расписание можно получить. Эти ограничивающие условия должны определять бесконтурный орграф, т. к. цикл при ограничивающих условиях очередности создает конфликт, который невозможно разрешить. Нас интересуют следующие задачи: ♦ критический путь. Критический путь— это самый длинный путь от начальной вершины до конечной. Эта информация может быть важной, т. к. единственным способом сократить общее время выполнения проекта будет сокращение времени выполнения одного задания в каждом критическом пути. Задания в критических пу- тях можно определить за время О(п + т), используя методы динамического про- граммирования, рассматриваемые в разделе 15.4'. ♦ минимальное время завершения. Самое короткое время, за которое можно завер- шить данный проект, соблюдая при этом условия очередности и предполагая нали- чие неограниченного количества исполнителей. При отсутствии условий очередно- сти все задания можно было бы выполнять одновременно, а общее время было бы равно времени, требуемого для выполнения самого длительного из них. А если на отдельные задания наложены строгие ограничивающие условия, согласно которым каждое задание должно выполняться только после выполнения его непосредствен- ного предшественника, то минимальное время завершения будет равно сумме вре- мен выполнения всех заданий. Для бесконтурного орграфа минимальное время выполнения можно вычислить за время О(п + т). Это время зависит от критического пути. Чтобы составить расписа- ние, перебирайте задания в топологическом порядке. Каждое задание должно за- пускаться на новом процессоре, как только завершится выполнение непосредствен- но предшествующего ему задания; ♦ компромисс между количеством исполнителей и временем завершения? Мы заин- тересованы в быстрейшем выполнении работы при заданном количестве исполни- телей К сожалению, эта и большинство подобных ей задач являются NP-полными.
488 Часть II. Каталог алгоритмических задач Любое реальное приложение календарного планирования будет, скорее всего, иметь комбинацию ограничивающих условий, смоделировать которые посредством описан- ных методов будет трудно или невозможно. Существуют два приемлемых подхода к решению таких задач. В первом случае мы игнорируем некоторые ограничивающие условия, пока задача не сведется к одному из описанных выше типов, решаем задачу, а потом оцениваем полученное решение с учетом других ограничивающих условий. Возможно, что полученное расписание можно будет легко подправить вручную, чтобы оно удовлетворяло таким негласным ограничениям, как запрещение совместной рабо- ты двум работникам, недолюбливающим друг друга. Другий подход— сформулиро- вать задачу календарного планирования во всей ее сложности в терминах линейного целочисленного программирования (см. раздел 13.6). Я рекомендую сначала приме- нить самый простой способ и посмотреть, что из этого получится, и только потом пе- реходить к более сложным методам. Другая фундаментальная задача календарного планирования заключается в распреде- лении набора заданий без ограничивающих условий очередности между идентичными исполнителями таким образом, чтобы минимизировать общее время исполнения. На- пример, в типографии требуется распределить несколько работ между несколькими копировальными машинами таким образом, чтобы завершить работу к концу рабочего дня. Такую задачу можно смоделировать как задачу разложения по контейнерам (см. раздел 17.9). где каждому заданию присваивается числовое значение, равное количест- ву часов, требуемому для ее завершения, а каждая машина представляется контейне- ром с емкостью, равной количеству рабочих часов. В более сложных вариантах задачи календарного планирования для каждого задания указывается время, раньше которого нельзя начинать выполнение, и крайний срок за- вершения. Для решения задач календарного планирования существуют эффективные эвристические алгоритмы, основанные на сортировке заданий по размеру и требуемо- му времени завершения. Дополнительную информацию см. в подразделах "Реализа- ции" и "Примечания". Обратите внимание на то, что эти задачи календарного планиро- вания становятся сложными только в том случае, если задания нельзя распределить по нескольким машинам или прерывать их исполнение с последующим возобновлением. Если ваше приложение дает вам высокую степень свободы, то ею следует воспользо- ваться. Реализации. JOBSHOP представляет собой коллекцию программ на языке С для ре- шения задач календарного планирования на основе вычислительного изучения, пред- ставленного в работе [АС91]. Коллекцию можно загрузить с веб-сайта http:// www2.isye.gatech.edu/~wcook/jobshop. Программа с открытым кодом Tablix (http://tablix.org) предназначена для решения задач создания расписаний в настоящих учебных заведениях. Поддерживает парал- лельные и кластерные вычисления. Программа календарного планирования LEK1N (см. [Pin02]) предназначена для обра- зовательный целей. Программа поддерживает планирование как для одной, так и для нескольких машин. Загрузить программу можно с веб-сайта http://www.stem. nyu.edu/om/software/lekin/.
Глава 14. Комбинаторные задачи 489 Для коммерческих приложений планирования доступна современная программа ILOG СР. Дополнительную информацию можно найти на веб-сайте http://www.ilog.com/ products/cp. Алгоритм 520 (см. [WBCS77]) из коллекции алгоритмов АСМ представляет собой про- грамму на языке FORTRAN для планирования расписания сети с множественными ре- сурсами. Алгоритм является частью библиотеки NetLib (см. развел 19.1.5). Примечания Существует огромное количество литературы об алгоритмах планирования. Подробный обзор достижений в этой области дается в книгах [Вги07] и [Pin02]. Подборка свежих об- зоров по всем аспектам календарного планирования приводится в кише [LA04], Имеется хорошо определенная классификация нескольких тысяч вариантов расписаний по машинной среде, по особенностям обработки и ограничивающим условиям, а также по параметрам, подлежащим минимизации. Обзоры содержатся в книгах [Bru07], [CPW98], [LLK83] и [Pin02], Диаграммы Ганта предоставляют визуальные представления решений распределения ра- бот по машинам, где ось абсцисс представляет время, а разные машины отображаются в разных рядах. Пример диаграммы Ганта показан на рис. 14.10. На этом рисунке каждая запланированная работа представлена в виде прямоугольника, определяющего ее испол- нителя и время начала и окончания. Методы календарного планирования проектов с огра- ничивающими условиями очередности часто называют системой PERT/CPM (Program Evaluation and Review Technique/Critical Path Method, система оценки и пересмотра пла- нов/Метод критического пути). Как диаграммы Ганта, так и система PERT/CPM обсуж- даются в большинстве учебников по исследованию операций, включая [Pin02]. При рассмотрении задач календарного планирования уроков и других подобных задач часто используется термин составление расписания (timetabling). На проводящейся дваж- ды в год конференции РАТАТ (Practice and Theory of Automated Timetabling, практика и теория автоматического составления расписания) обсуждаются новые результаты в этой области. Протоколы конференции доступны на веб-сайте http://www.asap.cs.nott.ac.uk/ patat/patat-index.shtml. Родственные задачи. Топологическая сортировка (см. раздел 15.2). паросочетание (см. раздел 15.6). вершинная раскраска (см. раздел 16.7). реберная раскраска (см. раз- дел 16.8). разложение по контейнерам (см. раздел 17.9). 14.10. Выполнимость Вход. Набор дизъюнкций в конъюнктивной нормальной форме. Задача. Определить, существует ли такой набор значений истинности булевых пере- менных, для которого одновременно выполняются все дизъюнкции (рис. 4.11). Обсуждение. Задача выполнимости возникает в ситуации, когда нужно найти конфи- гурацию или объект, который должен удовлетворять набору логических ограничений. Задача выполнимости, в основном, применяется для проверки того, чго проектируемая аппаратная или программная система работает должным образом для всех входных экземпляров. Допустим, что данная логическая формула S(X ) обозначает указанный результат на входных переменных X=Х\. ....Х„. а другая формула С(Х ) обозначает
490 Часть II. Каталог алгоритмических задач (XI or Х2 (XI or Х2 (XI or Х2 (XI or Х2 or ХЗ) or ХЗ ) or ХЗ ) or ХЗ ) вход выход Рис. 4.11. Задача выполнимости логику предлагаемой схемы вычисления S(X). Схема будет правильной в том случае, если нет такого значения X , для которого S( X )^С( X ). Задача выполнимости (или задача SAT) является самой первой NP-полной задачей. Хо- тя она применяется на практике в таких областях, как выполнимость ограничивающих условий, логика и автоматическое доказательство теорем, она имеет большую теорети- ческую значимость как корневая задача, на основе которой строятся все другие доказа- тельства NP-полноты. В создание лучших из современных систем решения задач SAT было вложено столько усилий, что их следует использовать, когда вам действительно нужно точно решить NP-полную задачу. Однако самым практичным подходом обычно является применение эвристических алгоритмов, которые дают хорошие, хотя и не оп- тимальные результаты. При проверке выполнимости необходимо найти ответы на следующие вопросы. ♦ В какой форме представлена ваша формула? В задаче выполнимости ограничи- вающие условия указываются в виде логической формулы. Существуют два основ- ных способа выражения логических формул — в конъюнктивной нормальной фор- ме (КНФ) и дизъюнктивной нормальной форме (ДНФ). Формулы в КНФ состоят из конъюнкций (операция И) дизъюнкций (операция ИЛИ), как в следующем примере: (v, ИЛИ vj) И (v2 ИЛИ v3) Для выполнения формулы в КНФ нужно выполнить все дизъюнкции. Формулы в ДНФ состоят из дизъюнкций (операция ИЛИ) конъюнкций (операция И). Предыдущую формулу можно записать в ДНФ таким образом: (v, И v2 И v3) ИЛИ (vi И v2 И v3) ИЛИ (v, И v2 И v3) ИЛИ (V| И v2 И v2) Для выполнения формулы в ДНФ нужно выполнить одну из конъюнкций. Решение задачи разрешимости формулы в ДНФ является тривиальным, т. к. любая формула ДНФ является выполнимой, если только каждая конъюнкция не содержит как литерал, так его дополнение (отрицание). Но задача выполнимости формулы в КНФ является NP-полной. Это кажется парадоксальным, т. к. посредством законов де Моргана можно преобразовать формулу из КНФ в ДНФ и наоборот. Причина за-
Глава 14 Комбинаторные задачи 491 ключается в том, что в процессе такого преобразования приходится создавать новые члены, количество которых может возрастать экспоненциально, вследствие чего само преобразование не может быть выполнено за полиномиальное время. ♦ Каков размер дизъюнкций? Задача Zr-SAT является частным случаем задачи выпол- нимости, в которой каждая дизъюнкция состоит из, самое большее, к переменных. Задача 1-SAT является тривиальной, т. к. для ее решения достаточно, чтобы истин- ное значение имела любая переменная в любой дизъюнкции. Задача 2-SAT триви- альной не является, но все же ее можно решить за линейное время. Этот факт пред- ставляет интерес, т. к. при определенной изобретательности удается моделировать некоторые задачи в виде задачи 2-SAT. Впрочем, задачи с дизъюнкциями, содер- жащими три переменные (т. е. 3-SAT), являются NP-полными. ♦ Достаточно ли выполнить только одну из дизъюнкций? Если требуется точное ре- шение, то вам в любом случае придется использовать какой-нибудь алгоритм пере- бора с возвратом, например, метод Дэвиса-Путнама (Davis-Putnam). В наихудшем случае потребуется проверить 2™ наборов значений истинности, но, к счастью, су- ществует много способов сократить пространство поиска. Хотя теоретически задача выполнимости является NP-полной, трудность конкретной задачи зависит от спосо- ба создания экземпляров. Естественно определенные "случайные" экземпляры часто решаются с неожиданной легкостью; более того, задача генерирования по- настоящему сложных экземпляров является нетривиальной. Тем не менее, иногда полезно сделать задачу менее строгой, поставив в качестве цели выполнимость наибольшего количества дизъюнкций. Качество решений, по- лученных случайным методом или с помощью эвристического алгоритма, можно улучшить с помощью оптимизационных методов, таких как метод имитации отжи- га. В самом деле, при присвоении переменным любого случайного набора значений истинности вероятность выполнимости каждой &-SAT дизъюнкции составляет I -(I/2)*, так что, скорее всего, с первой попытки выполнится большинство дизъ- юнкций. Однако завершить работу будет труднее. Задача поиска набора значений истинности, удовлетворяющего наибольшему количеству дизъюнкций, является NP-полной даже для экземпляров, не имеющих решения. Когда мы сталкиваемся с задачей неизвестной сложности, доказательство ее NP- полноты может оказаться важным первым шагом в ее решении. Если вы думаете, что ваша задача может быть сложной, посмотрите, нет ли ее в списке сложных задач в кни- ге [GJ79], Если нет, то я рекомендую попробовать самостоятельно доказать сложность задачи, используя базовые задачи 3-SAT, вершинного покрытия, независимого множе- ства, разбиения целого числа, клики и гамильтонова цикла. Я советую начинать имен- но с этих задач, а не со сложных задач из книги [GJ79]. Кроме того, это поможет вам лучше разобраться в вопросе, поскольку сложность вашей задачи не будет заслонена доказательством сложности задачи из книги. Стратегии доказательства сложности рас- сматриваются в главе 9. Реализации. В последние годы наблюдался заметный рост производительности средств решения задач SAT. На ежегодном соревновании по решению задач SAT опре- деляются лучшие средства решения для трех категорий экземпляров задач — промыш- ленного, ручного и случайного.
492 Часть II. Каталог алгоритмических задач В соревновании SAT в 2007 г. наилучшими решателями промышленных задач были Rsat (http://reasoning.cs.ucla.edu/rsat), PicoSAT (http://fmv.jku.at/picosat) и MiniSAT (http://www.es.chalniers.se/Cs/Research/FormalMethods/MiniSat). Исходный код этих трех решателей и другая информация доступны на веб-странице http:// www.satconipetition.org. Веб-сайт SAT Live! (http://www.satlive.org)— источник самых свежих научных работ, программ и тестовых наборов для задач выполнимости и родственных задач оптимиза- ции логических функций. Примечания Наиболее полный обзор тестирования выполнимости представлен в книге [KSBD07]. Ал- горитм обхода с возвратом DPLL (Davis-Putnam-Logemann-Loveland) для решения задач выполнимости был представлен в 1962 г. Методы локального поиска работаю! лучше на определенном классе задач, трудных для решателей DPLL. Особенно популярным являет- ся решатель Chaff (см. [MMZ 01]), который можно загрузить с веб-сайта http://www. princeton.edu/~chaff/. Обзор последних достижений в области тестирования выполнимо- сти представлен в работе [KS07]. В работе [DGU+02] описывается алгоритм для решения задачи 3-SAT за время О*( 1,4802") в наихудшем случае. Обзор эффективных (но имеющих не полиномиальное время исполнения) алгоритмов решения NP-полных задач приводится в работе [\Voe03]. Основным справочником по NP-полноте является книга [GJ79], содержащая список около 400 NP-полных задач. Уже долгие годы эта книга остается чрезвычайно полезным спра- вочником. Я обращаюсь к ней чаще, чем к какой-либо другой. Обновления к книге пре- доставляются в колонке Дэвида Джонсона (David Johnson), публикующейся время от вре- мени в журналах "Journal of Algorithms" и (в последнее время) "ACM Transactions on Algorithms" Обсуждению теоремы Кука (см. [Соо71]), доказывающей сложность задачи выполнимо- сти, посвящены работы [CLRS01], [GJ79] и [КТ06]. Важность результатов работы Кука становится ясной из доклада Карпа (Karp) (см. [Каг72]), в котором доказывается слож- ность свыше 20 разных комбинаторных задач. В работе [АРТ79] представлен алгоритм решения задач 2-SAT за линейное время. Инте- ресное применение задачи 2-SAT для нанесения обозначений на карты излагается в рабо- те [WW95]. Самый лучший известный эвристический алгоритм выдает приблизительное решение задачи MAX-2-SAT с точностью до 1,0741 (см. [FG95]). Родственные задачи. Условная оптимизация (см. раздел 13.5). задача коммивояжера (см. раздел 16.4).
ГЛАВА 15 Задачи на графах с полиномиальным временем исполнения Алгоритмические задачи на графах составляют приблизительно треть всех задач в этом каталоге. Многие задачи из других разделов можно было бы сформулировать с тем же успехом в виде графов, например задачу минимизации ширины ленты и задачу оптимизации конечного автомата. Одним из основных навыков хорошего алгориста является способность определить название задачи теории графов. Если вы знаете на- звание своей задачи, вы найдете в каталоге точные инструкции, как приступить к ее решению. В этой главе рассматриваются только те задачи, для решения которых существуют эф- фективные алгоритмы. 1 ак как часто существует несколько способов моделирования приложения, то имеет смысл, прежде чем использовать одну из более трудных поста- новок решаемой задачи, поискать в этой главе способ ее моделирования. Время исполнения представленных в этой главе алгоритмов возрастает полиномиально с увеличением размера графа. Буквой п обозначается количество вершин графа, а бук- вой т — количество его ребер. Графы легче всего воспринимать, когда они представлены в виде рисунков. Поэтому мы также рассмотрим алгоритмы рисования графов, деревьев и планарных графов. Большинство сложных алгоритмов на графах трудно программировать. Но существуют уже готовые реализации, нужно только знать, где их можно найти. Лучшие источники реализаций алгоритмов на графах— библиотека LEDA (см. [MN99]) и библиотека Boost Graph Library (см. [SLL02]). Но для многих задач существуют специальные про- граммы. Самые свежие обзоры разнообразных алгоритмов для работы с графами приведены в книге [TNX08], Другие интересные обзоры можно найти в книге [vL90a] и некоторых главах из книги [Ata98]. Среди специализированных книг по алгоритмам обработки графов можно порекомендовать следующие: ♦ [Sed98] — том этого учебника, посвященный алгоритмам для работы с графами, содержит всеобъемлющее, но доступное введение в эту область; ♦ [АМО93] — хотя эта книга посвящена, в основном, потокам в сетях, в ней рассмат- ривается целый набор алгоритмов, причем особое внимание уделяется исследова- нию операций; ♦ [Gib85] — хорошая книга по разнообразным алгоритмам для работы с графами, включая проверку планарности, поиск паросочетаний. эйлеровых и гамильтоновых циклов, а также более простые темы;
494 Часть II Каталог алгоритмических задач ♦ [Eve79a] — уже не новый, но от этого не менее ценный учебник по алгоритмам для работы с графами, содержащий подробное обсуждение алгоритмов для проверки планарности графов. 15.1. Компоненты связности Вход. Ориентированный или неориентированный граф G. Задача. Разбить граф (J на части, или компоненты, так, чтобы вершины х и у принад- лежали разным компонентам, если изх в у нет пути (рис. 15.1). 9 ВХОД выход Рис. 15.1. Компоненты связности Обсуждение. Компоненты связности графа представляют части графа. Две вершины находятся в одной и той же компоненте графа G тогда и только тогда, когда между ни- ми существует какой-либо путь. Задача поиска компонент связности является центральной во многих приложениях на графах. Рассмотрим, например, задачу определения естественных кластеров в наборе элементов. Для решения задачи мы представляем каждый элемент в виде вершины и соединяем ребром каждую пару элементов, которые считаются "подобными". Компо- ненты связности этого графа соответствуют разным классам элементов. Проверка графа на наличие в нем компонент связности является важным шагом пред- варительной обработки. При исполнении алгоритма только на одной компоненте не- связного графа часто удается выявить скрытые, труднообнаруживаемые ошибки. Про- верка на связность выполняется так легко и быстро, что следует всегда проводить ее для графа входа, даже если вы уверены, что он должен быть связным. Связность любого неориентированного графа можно проверить посредством обхода графа в глубину или в ширину (см. главу 5). В действительности, не так важно, какой из этих методов обхода вы выберете. При любом обходе поле "номер компоненты" для каждой вершины инициализируется нулевым значением, а потом начинается поиск
Глава 15. Задачи на графах с полиномиальным временем исполнения 495 первой компоненты из вершины v,. При открытии каждой вершины этому полю при- сваивается значение номера текущей компоненты. По завершении первоначального обхода номер компоненты увеличивается на единицу, и поиск выполняется снова, на- чиная с первой вершины, у которой номер компоненты все еще равен нулю. При пра- вильной реализации с использованием списков смежности (как описано в разде- ле 5.7.7) этот алгоритм имеет время исполнения О(п + т). На практике также возникают другие вопросы, касающиеся связности. ♦ Является ли граф ориентированным? Для ориентированных графов существует два понятия компонент связности. Ориентированный граф является слабо связным (weakly connected), если при игнорировании ориентации ребер он окажется связ- ным. Таким образом, слабо связный граф состоит из одной-единственной компо- ненты. Ориентированный граф является сильно связным (strongly connected), если между любой парой его вершин существует ориентированный путь. Различие меж- ду ними легко понять при рассмотрении сети улиц с односторонним и двусторон- ним движением. Сеть является сильно связной, если возможно проехать между лю- быми двумя точками, не нарушая правил. Если между любыми двумя точками мож- но проехать, не нарушая или нарушая правила, то сеть является слабо связной. Если же попасть из точки А в точку В не является возможным, то сеть является несвяз- ной. Слабо и сильно связные компоненты определяют однозначные разбиения вершин. На рис. 15.1 представлен ориентированный граф, состоящий из двух слабо или пяти сильно связных компонент (также называющихся блоками графа G). Проверку орграфа на слабую связность можно с легкостью провести за линейное время. Просто сделаем все ребра графа неориентированными и выполним на нем алгоритм поиска компонент связности на основе обхода в глубину. Проверку на сильную связность провести сложнее. Самый простой алгоритм с линейным време- нем исполнения осуществляет поиск, начиная с какой-либо вершины v, чтобы про- демонстрировать достижимость всех вершин графа из этой вершины. Затем мы соз- даем граф G', поменяв направление всех ребер графа G на обратное. Чтобы узнать, достижимы ли все вершины графа G из вершины v, достаточно выполнить обход графа G' из этой вершины. Граф G является сильно связным тогда и только тогда, когда вершина v достижима из любой вершины в графе G и все вершины в графе G достижимы из вершины v. Все сильно связные компоненты графа G можно извлечь за линейное время, ис- пользуя более сложные алгоритмы типа обхода в глубину. Обобщение только что описанной идеи двойного обхода в глубину с легкостью поддается программирова- нию. но понять, как оно работает, несколько сложнее. 1. Выполняем обход в глубину, начав с произвольной вершины графа G и нумеруя вершины в порядке завершения их обработки (а не открытия). 2. Изменяем направление каждого ребра графа G, создав, таким образом, граф G'. 3. Выполняем обход в глубину графа G', начиная с вершины графа G, имеющей наи- больший номер. Если не удается обойти граф G' полностью, продолжаем обход с еще не посещенной вершины с наибольшим номером.
496 Часть II. Каталог алгоритмических задач 4. Каждое созданное на шаге 3 дерево обхода в глубину является сильно связной ком- понентой. В разделе 5.10.2 приводится моя реализация однопроходного алгоритма. В любом случае, проще использовать готовую реализацию, чем изучать описание алгоритма в учебнике. ♦ Какое самое слабое место в моем графе/сети? Прочность цепочки измеряется прочностью ее самого слабого звена. Потеря одного или нескольких внутренних звеньев разорвет цепочку на части. Связность графа измеряет его "прочность" — количество ребер или вершин, которое нужно удалить, чтобы разбить граф на части. Связность является важным понятием в проектировании сетей и других задачах стру кту ри ро ван и я. Алгоритмические задачи связности графов рассматриваются в разделе 15.8. В част- ности, двусвязные компоненты представляют собой части графа, получаемые в ре- зультате разрыва ребер, имеющих общую вершину. Все компоненты двусвязности можно найти за линейное время посредством обхода в глубину. Реализацию этого алгоритма см. в разделе 5.9.2. Вершины, удаление которых нарушает связность гра- фа, принадлежат нескольким компонентам двусвязности, ребра которых однозначно распределяются между компонентами. ♦ Является ли граф деревом? Как найти цикл, если он существует? Задача поиска цикла возникает часто, особенно с ориентированными графами. Например, провер- ка. возможна ли взаимная блокировка последовательности условий, часто сводится к задаче выявления цикла. Если я жд> Фреда, а Фред ждет Мэри, а Мэри ждет меня, то налицо цикл и взаимная блокировка. Для неориентированных графов аналогичной задачей является идеигификация де- рева. По определению дерево является неориентированным связным бесконтурным графом. Как описано ранее, обход в глубину можно использовать для проверки графа на связность. Если граф является связным и имеет п- 1 ребер для п вершин, то этот граф является деревом. Обход в глубину можно использовать для поиска циклов как в ориентированных, так и в неориентированных графах. Всякий раз, когда в процессе обхода в глубину мы обнаруживаем обратное ребро (т. е. ребро к вершине-предшественнику в дереве обхода в глубину), это обратное ребро и дерево совместно определяют ориентиро- ванный цикл. Существования другого такого цикла в ориентированном графе не- возможно. Ориентированные графы, не содержащие контуров (циклов), называются бесконтурными. Фундаментальной операцией на бесконтурных орграфах является топологическая сортировка. Реализации. Все реализации структур данных для работы с графами, описанные в раз- деле 12.4. включают реализацию обхода в ширину и обхода в глубину и, соответствен- но, определенную степень проверки на связность. Библиотека Boost Graph Library для C++ (см. [SLL02]) предоставляет реализации компонент связности. Библиотеку можно загрузить с веб-сайта littp://wvvvv.boost.org/libs/graph/doc. Библиотека LEDA (см. раз- дел 19.1.1) содержит реализации на языке C++ этих компонент, а также двусвязных и трехсвязных компонент, и обходов в глубину и ширину.
Глава 15. Задачи на графах с полиномиальным временем исполнения 497 Для программистов, пишущих на языке Java, библиотека JUNG (http:// jung.sourceforge.net/) содержит реализации алгоритмов компонент двусвязности, а библиотека JGraphT (http://jgrapht.sourceforge.net/)— компоненты сильной двусвяз- ности. По моему предвзятому мнению, лучшей реализацией всех основных алгоритмов про- верки на связность (включая компоненты сильной связности и компоненты двусвязно- сти) на языке С является библиотека, разработанная для этой книги. Подробности см. в разделе 19.1.10. Примечания Обход в глубину был впервые использован в девятнадцатом столетии для поиска выхода из лабиринтов (см. [Luc91] и [Таг95]). А обход в ширину был впервые использован в 1957 г. для поиска кратчайшего пути (см. [Моо59]). В работах [НТ73Ь] и [Таг72] было установлено, что обход в глубину является фундамен- тальным методом построения эффективных алгоритмов на графах. Алгоритмы обхода в глубину и обхода в ширину приводятся в каждой книге, посвященной алгоритмам, причем книга [CLRSOl] содержит, пожалуй, наиболее полное описание этих алгоритмов. Первый алгоритм для поиска компонент сильной связности был предложен в работе [Таг72], а его описание приведено в книгах [BvG99], [Eve79a] и [Мап89]. Еще один алго- ритм для поиска компонент сильной связности, более легкий для программирования и бо- лее изящный, был разработан Шариром (Sharir) и Косараю (Kosaraju). Хорошие описания этого алгоритма приведены в книгах [AHU83] и [CLRSOl]. В своей работе [СМ96] Чериян (Cheriyan) и Мэлхорн (Mehlhorn) представляют улучшенные алгоритмы для некоторых за- дач на плотных графах, включая поиск компонент сильной связности. Родственные задачи. Связность ребер и вершин (см. раздел 15.8), поиск кратчайшего пути (см. раздел 15 4). 15.2. Топологическая сортировка Вход. Бесконтурный (ациклический) орграф G = (V, Е), также называющийся частично упорядоченным множеством (partially ordered set). Задача. Найти линейное упорядочение вершин Г', такое, что для каждого ребра (/,/) вершина i находится слева от вершины j (рис. 15.2). Обсуждение. Подзадача топологической сортировки возникает в большинстве алго- ритмов на ориентированных ациклических графах. Топологическая сортировка упоря- дочивает вершины и ребра бесконтурного орграфа простым и непротиворечивым обра- зом, и поэтому играет ту же самую роль для бесконтурных орграфов, что и обход в глубину для общих графов. Топологическую сортировку можно использовать для создания расписания работ с ог- раничивающими условиями очередности. Если у нас имеется набор задач, которые должны выполняться в определенной последовательности, то эти ограничивающие условия очередности формируют бесконтурный орграф, и любая топологическая сор- тировка на этом графе, иногда называемая линейным расширением (linear extension), определяет такой порядок выполнения этих работ, при котором каждая из них выпол- няется только после удовлетворения ее ограничивающих условий.
498 Часть II. Каталог алгоритмических задач В отношении топологической сортировки справедливы следующие три важных факта. ♦ Топологическую сортировку можно выполнять только на бесконтурных орграфах, т. к. любой ориентированный цикл принципиально противоречит линейному поряд- ку выполнения работ. ♦ Топологическую сортировку можно выполнить на любом бесконтурном орграфе, поэтому для любого разумного набора ограничивающих условий очередности работ должно существовать, по крайней мере, одно расписание их выполнения. ♦ Для бесконтурного орграфа можно выполнить несколько разных топологических упорядочений, особенно при небольшом количестве ограничивающих условий. Так, для п работ, не имеющих никаких ограничивающих условий, любая из п\ переста- новок работ является допустимым топологическим упорядочением. Самый концептуально простой алгоритм топологической сортировки выполняет обход в глубину бесконтурного орграфа, чтобы найти полный набор вершин-истоков, т. е. вершин, у которых количество входящих дуг равно нулю. Любой бесконтурный орграф должен иметь, по крайней мере, одну такую вершину-исток. Вершины-истоки могут находиться в начале любого расписания, не нарушая никаких ограничивающих усло- вий. Удаление всех исходящих ребер этих вершин-истоков создаст новые вершины- истоки, которые будут располагаться справа от первого набора. Эта процедура повто- ряется до тех пора, пока не будут учтены все вершины. При разумном выборе структур данных (списков смежности и очередей) этот алгоритм будет иметь время исполнения О(п + т). В альтернативном алгоритме используется то обстоятельство, что упорядочение всех вершин по времени завершения обхода в глубину в убывающем порядке дает линейное расширение. Реализация этого алгоритма с доказательством его правильности приво- дится в разделе 5.10.1. В топологической сортировке следует принимать во внимание следующие два аспекта: ♦ Как получить все линейные расширения? В некоторых приложениях важно создать все линейные расширения бесконтурного орграфа. Вы должны отдавать себе отчет
Глава 15 Задачи на графах с полиномиальным временем исполнения 499 в том, что количество линейных расширений может возрастать экспоненциально по отношению к размеру графа. Даже сама задача подсчета количества всех линейных расширений является NP-полной. Алгоритмы для перечисления всех линейных расширений бесконтурного орграфа основаны на обходе с возвратом. В них создаются все возможные слева направо упорядочения, где кандидатами для следующей вершины являются все вершины с нулевой степенью захода. Прежде чем продолжать обход, из выбранной вершины удаляются все исходящие ребра. Оптимальный алгоритм для перечисления линей- ных расширений рассматривается в подразделе "Примечания". Алгоритмы для создания случайных линейных расширений начинают с произволь- ного линейного расширения. В этом линейном расширении последовательно выби- раются пары вершин, которые меняются местами, если получающаяся в результате такого обмена перестановка остается топологической сортировкой. При выборе достаточного количества случайных пар вершин этот процесс дает случайное ли- нейное расширение. Подробности см. в подразделе "Примечания”. ♦ Как поступать, когда граф не является бесконтурным? Когда набор ограничи- вающих условий содержит внутренние противоречия, естественно возникает задача поиска наименьшего набора условий, удаление которого устраняет все конфликты. Наборы непрестижных работ (вершин) или ограничений (ребер), после удаления которых орграф становится бесконтурным, называются разрывающими множест- вами вершин (feedback vertex set) и разрывающими множествами дуг (feedback arc set) соответственно. Эти структуры рассматриваются в разделе 16.11. К сожалению, задачи поиска обоих наборов являются НР-полными. Так как алгоритм топологической сортировки зацикливается, как только он находит вершину в ориентированном контуре, мы можем удалить проблемное ребро или вершину и продолжать работу. В конечном итоге этот простой и эффективный эвристический подход делает орграф бесконтурным, но может удалить больше эле- ментов, чем необходимо. Лучший подход к решению этой задачи описывается в разделе 9.10.3. Реализации. По сути, все упоминаемые в разделе 12.4 реализации структур данных для представления графов, включая библиотеку Boost Graph Library (http://www. boost.org/libs/graph/doc) и библиотеку LEDA (см. раздел 19.1.1), содержат реализации топологической сортировки. Для языка Java библиотека JDSL (http://vvvvw.jdsl.org/) содержит реализацию специальной процедуры для выполнения взвешенной топологи- ческой нумерации. Советую также обратить внимание на библиотеку JGraphT (http://jgrapht.sourceforge.net). Сервер комбинаторных объектов на веб-сайте http://theory.cs.uvic.ca предоставляет программы на языке С для генерирования линейных расширений как в лексикографи- ческом порядке, так и в порядке кода Грея, а также средства для их перечисления. Сер- вер оснащен интерактивным интерфейсом. По моему предвзятому мнению, лучшей реализацией всех основных алгоритмов реше- ния задач на графах (включая топологическую сортировку), на языке С, является биб- лиотека, разработанная для этой книги. Подробности см. в разделе 19.1.10.
500 Часть II. Каталог алгоритмических задач Примечания Хорошие описания топологической сортировки включают книги [CLRSOl] и [Мап89] В своей работе [BW91] Брайтвэл (Brightwell) и Уинклер (Winkler) доказали, что задача подсчета количества линейных расширений частичного порядка является #Р-полной. Класс сложности #Р включает класс сложности NP, следовательно, любая #Р-полная зада- ча является, по крайней мере, NP-полной. Прусс (Pruesse) и Раски (Ruskey) в своей работе ([PR86]) представляют алгоритм для ге- нерирования линейных расширений бесконтурного орграфа за постоянное амортизиро- ванное время. Кроме этого, каждое расширение отличается от своего предшественника перестановкой только одного или двух смежных элементов. Этот алгоритм можно ис- пользовать для подсчета линейных расширений e(G) „-вершинного графа G за время О(п~ + e(G)). Альтернативным методом перечисления линейных расширений является метод поиска в обратном направлении Ависа (Avis) и Фукуды (Fukuda), представленный в работе [AF96]. Программа обхода с возвратом для генерирования всех линейных пере- числений описана в работе [KS74] Дональда Кнута. В работе [НиЬОб] Губера (Huber) представлен алгоритм случайной выборки с равномер- ным распределением линейных расширений из произвольного частичного порядка за время G(w lg„), что улучшает результаты работы алгоритма, представленного в работе [BD99], Родственные задачи. Сортировка (см. раздел 14.1), разрывающее множество вершин и ребер (см. раздел 16.11). 15.3. Минимальные остовные деревья Вход. Граф G = (V, Е) со взвешенными ребрами. Задача. Найти подмножество ребер Е' е Е, которое формирует дерево на вершинах Г и имеет минимальный вес (рис. 15.3). ВХОД ВЫХОД Рис. 15.3. Поиск минимального остовного дерева Обсуждение. Минимальное остовное дерево графа определяет подмножество ребер с минимальным весом, которое удерживает граф в одной компоненте связности. В ре- шении задачи построения минимального остовного дерева могул быть заинтересованы.
Глава 15. Задачи на графах с полиномиальным временем исполнения 501 например, телефонные компании, поскольку минимальное остовное дерево набора объектов определяет схему для соединения этих объектов в сеть с использованием ка- беля наименьшей длины. Задача построения минимального остовного дерева является основной задачей проектирования сетей. Важность минимальных остовных деревьев объясняется несколькими причинами: ♦ они легко и быстро реализуются на компьютере и образуют разреженные подграфы, которые содержат много информации о первоначальных графах; ♦ они предоставляют способ определить кластеры в наборах точек. Удаление длин- ных ребер из минимального остовного дерева оставляет компоненты связности, ко- торые определяют естественные кластеры в наборе данных, как показано на рис. I5.3; ♦ с их помощью можно получить приблизительные решения сложных задач, таких как задача построения дерева Штейнера и задача коммивояжера; ♦ они могут быть использованы в учебных целях, поскольку алгоритмы для построе- ния минимального остовного дерева дают графическую иллюстрацию того, как "жадные" алгоритмы выдают доказуемо оптимальные решения. Известны три классических алгоритма для эффективного создания минимальных ос- товных деревьев. Подробная реализация двух из них (алгоритма Прима и алгоритма Крускала) совместно с доказательством их правильности дается в разделе 6.1. Третий алгоритм менее известен, хотя он был изобретен первым, прост в реализации и более эффективен. Далее даются краткие описания и псевдокод этих алгоритмов. ♦ Алгоритм Крускала. В начале каждая вершина представляет отдельное дерево, и эти деревья сливаются в одно путем последовательного добавления ребер с наи- меньшим весом, которые соединяют по два поддерева (т. е. не создают цикл). Псев- докод этого алгоритма представлен в листинге 15.1. Листинг 15.1. Алгоритм Крускала Kruskal (G) Сортируем ребра по весу в возрастающем порядке count = О while (count • n — 1) do get next edge (v, w) if (component (v) / component(w)) add to T component(v) component(w) Проверку, какой компонент выбирать, можно эффективно реализовать с помощью структуры данных "объединение-поиск" (см. раздел 12.5), что даст нам алгоритм с временем исполнения OOnlg/n). ♦ Алгоритм Прима. Начинает работу с произвольной вершины v и "выращивает" де- рево из этой вершины, в цикле выбирая ребро с наименьшим весом, которое при- соединяет к дереву какую-либо новую вершину. В процессе исполнения каждая вершина помечается или как входящая в дерево, или как открытая, но еще не до-
502 Часть II. Каталог алгоритмических задач бавленная в дерево вершина, или как не увиденная, (т. е.. расположенная на рас- стоянии более одного ребра от дерева). Псевдокод этого алгоритма представлен в листинге 15.2. Листинг 15.2. Алгоритм Прима Prim(G) Выбираем произвольную вершину, с которой надо начинать построение дерева while (имеются вершины, не включенные в дерево) выбираем ребро минимального веса между деревом и вершиной вне дерева добавляем выбранное ребро и вершину в дерево ооновляем стоимость для всех затрагиваемых ребер вне дерева Этот алгоритм создает остовное дерево для любого связного графа, т. к. циклы не могут быть внесены через ребра между вершинами в дереве и вне его. То. что это дерево действительно имеет минимальный вес. можно доказать методом "от про- тивного". Для простых структур данных можно реализовать алгоритм Прима с вре- менем исполнения О(п2). ♦ Алгоритм Борувки. Основан на том обстоятельстве, что инцидентные каждой вер- шине ребра наименьшего веса должны быть в минимальном остовном дереве. Ре- зультатом объединения этих ребер будет остовной лес. содержащий, как минимум, nil деревьев. Теперь для каждого из этих деревьев Т выбираем такое ребро (х, у) наименьшего веса, для которого х е Т и у ё Т. Каждое из этих ребер опять должно быть в минимальном остовном дереве, а результатом их объединения опять будет остовной лес, содержащий, самое большее, половину предыдущего количества де- ревьев. Псевдокод этого алгоритма представлен в листинге 15.3 Листинг 15.3. Алгоритм Борувки Boruvka(G) Инициализируем остовной лес F, состоящий из п одновершинных деревьев while '.лес F содержит больше, чем одно дерево) for each 1дерева Т в лесу F' находим ребро с наименьшим весом от Т к G - Т добавляем все выбранные ребра в лес F, спивая вместе пары деревьев В каждой итерации количество деревьев уменьшается, по крайней мере, вдвое, что дает нам минимальное остовное дерево после, самое большее, logn итераций, каж- дая из которых выполняется за линейное время. В целом это дает нам алгоритм с временем исполнения O(/nlog«). причем без использования каких-либо хитроумных структур данных. Построение минимального остовного дерева является только одной из нескольких за- дач об остовном дереве, возникающих на практике. Чтобы разобраться в них, попытай- тесь найти ответы на следующие вопросы. ♦ Одинаковый чи вес у всех ребер графа? Любое остовное дерево из п вершин имеет ровно п — 1 ребер. Таким образом, если граф невзвешенный, то любое остовное де-
Глава 15. Задачи на графах с полиномиальным временем исполнения 503 рево будет минимальным остовным деревом. Корневое остовное дерево можно най- ти за линейное время посредством обхода в глубину или в ширину. Обход в глубину обычно выдает высокие и тонкие деревья, а обход в ширину отражает структуру расстояний графа (см. главу 5). ♦ Какой алгоритм использовать. Прима или Крускала? В том виде, в каком они реа- лизованы в разделе 6.1, алгоритм Прима имеет время исполнения О(и"), а алгоритм Крускала— O(wlogm). Таким образом, алгоритм Прима эффективен на плотных графах, а алгоритм Крускала— на разреженных. Вместе с тем, используя более сложные структуры данных, можно создать реализа- ции алгоритма Прима с временем исполнения О(т + nlgn). а реализация алгоритма Прима с использованием парных пирамид будет самой быстрой на практике как для разреженных, так и для плотных графов. ♦ Что предпринять, если входные данные — не граф, а точки на плоскости? Геомет- рические экземпляры, состоящие из п точек в d измерениях, можно решать, создав граф полного расстояния за время О(п~}, а затем построив минимальное остовное дерево этого графа. Но для точек на плоскости более эффективный подход— непо- средственно решить геометрическую версию задачи. Чтобы построить минимальное остовное дерево для п точек, сначала создадим триангуляцию Делоне (Delaunay triangulation) для этих точек (см. разделы 17.3 и 17.4). На плоскости это даст граф с О(п) ребрами, который содержит все ребра минимального остовного дерева нашего множества точек. Алгоритм Крускала находит минимальное осговное дерево в этом разреженном графе за время O(nlgn). ♦ Как найти остовное дерево, не имеющее вершин с высокой степенью? Другой рас- пространенной целью задач об остовном дереве является минимизация степени вершин, обычно чтобы минимизировать исходящие разветвления в сети. К сожале- нию. задача построения остовного дерева с максимальной степенью вершин, равной 2. является NP-полной задачей, т. к. она идентична задаче поиска гамильтонова пу- ти. Но существуют эффективные алгоритмы для создания остовных деревьев с мак- симальной степенью вершин на единицу больше, чем требуемая, что, скорее всего, будет достаточно на практике. Реализации. Среди реализаций структур данных для представления графов из разде- ла 12 4 обязательно должны присутствовать реализации алгоритма Прима и/или алго- ритма Крускала. Этому требованию отвечают библиотеки Boost Graph Library (см. [SLL02]) (http://vvvvw.boost.org/Iibs/graph/doc) и LEDA (см. раздел 19.1.1). По неиз- вестной причине библиотеки графов Java, ориентированные преимущественно на со- циальные сети, не содержат реализаций этих алгоритмов, зато в библиотеке JDSL (http://vvvvvv.jdsl.org) есть реализации обоих алгоритмов. Эксперименты с измерением времени исполнения алгоритмов для построения мини- мального остовного дерева дают противоречивые результаты, дающие основание пола- гать, что разница в их производительности слишком незначительная. Реализации алго- ритмов Прима, Крускала и Черитона-Тарьяна (Cheriton-Tarjan) на языке Pascal описы- ваются в книге [MS91]. Здесь же приводится подробный анализ экспериментальных данных, доказывающий, что на большинстве графов самым быстрым будет алгоритм Прима, реализованный с правильно построенной очередью с приоритетами. В про-
504 Часть II. Каталог алгоритмических задач грамме Standford GraphBase самым быстрым из четырех реализаций разных алгорит- мов построения минимального остовного дерева оказался алгоритм Крускала (см. раз- дел 19.1.8). Библиотека Combinatorica (см. [PS03]) содержит реализацию (на языке пакета Mathematica) алгоритма Крускала для построения минимального остовного дерева и быстрого подсчета количества остовных деревьев в графе. Подробности см. в разде- ле 19.1.9. Но моему предвзятому мнению, лучшей реализацией на языке С всех основных алго- ритмов на графах, включая алгоритмы для построения минимальных остовных деревь- ев, является библиотека, разработанная для этой книги. Подробности см. в разде- ле 19.1.10. Примечания Задача построения минимального остовного дерева была впервые решена в 1926 г., когда был разработан алгоритм Борувки. Алгоритмы Прима (см. [Pri57]) и Крускала для реше- ния этой задачи появились только в середине 1950-х годов. Алгоритм Прима был заново открыт Дейкстрой (см. [Dij59])_ Дополнительную информацию об истории алгоритмов построения минимального остовного дерева можно найти в [GH85]. By (Wu) и Чао (Chao) написали монографию [WC04b] о задаче построения минимального остовного дерева и родственным задачам. Самые быстрые реализации алгоритмов Прима и Крускала используют пирамиды Фибо- наччи (см. [FT87]). Однако позже были предложены парные пирамиды, которые обеспе- чивают такую же производительность, но при меньших накладных расходах Экспери- менты с парными пирамидами описаны в журнале [SV87J. Алгоритм, получаемый в результате простой комбинации алгоритма Борувки с алгорит- мом Прима, имеет время исполнения равное O(/«lglgn). Выполните Iglgn итераций алго- ритма Борувки, чтобы получить лес из, самое большее, я/lgn деревьев. Теперь создайте граф G', в котором каждое дерево этого леса представлено одной вершиной, а вес ребра между деревьями Т, и 7} установите равным весу самого легкого ребра (л,у), где.с е Т„ а у е 7,. Минимальное остовное дерево графа С совместно с ребрами, выбранными алго- ритмом Борувки, образуют минимальное остовное дерево графа G. Время исполнения ал- горитм Прима (реализованного с использованием пирамид Фибоначчи) на этом графе из ц/lgw вершин и m ребер составляет О(п + го). История выяснения оптимального времени построения минимальных остовных деревьев выглядит примерно так. В своей работе [ККТ95] Каргер (Karger). Кляйн (Klein) и Тарьян (Tarjaii) предложили рандомизированный алгоритм на основе алгоритма Борувки для по- строения минимального остовного дерева за линейное время. В свою очередь, Шазель (Chazelle) в работе [ChaOOJ представил алгоритм с временем исполнения O(na(m,»)), где а(т, п) является функцией, обратной функции Аккермана. Наконец, Петти (Pettie) и Рама- чандран (Ramachandran) в работе [PR02] описали доказуемо оптимальный алгоритм, точ- ное время исполнения которого неизвестно (как бы странно это ни звучало), но лежит в диапазоне между Q(« + in) и O(na(m, п)). Остовным подграфом S(G) графа G называется подграф, обеспечивающий эффективный компромисс между двумя противоречащими друг другу целями проектирования сетей. Говоря конкретно, общий вес остовного подграфа близок к весу минимального остовного дерева полного графа G, и в то же самое время гарантируется, что кратчайший путь меж- ду вершинами х и г в графе 5(G) близок к кратчайшему пути в полном графе G. Самый свежий исчерпывающий обзор достижений в этой области представлен в монографии [NS07],
Глава 15 Задачи на графах с полиномиальным временем исполнения 505 Алгоритм для вычисления евклидова минимального остовного дерева был разраоотан Шамосом (Shamos). Он обсуждается в учебниках по вычислительной геометрии, таких как [dBvKOSOO] и [PS85]. В работе [FR94] Фурера (Furer) и Рагхавачари (Raghavachari) представлен алгоритм для создания остовного дерева почти минимальной степени — всего лишь на единицу боль- ше, чем степень остовного дерева наименьшей степени. Данная ситуация аналогична тео- реме Визинга (Vizing) для раскраски ребер, которая позволяет построить аппроксими- рующий алгоритм, выдающий решение с точностью до единицы. В работе [SL07] пред- ставлен полиномиальный алгоритм построения остовного дерева с максимальной степенью, равной или меньшей, чем к з 1, стоимость которого не больше стоимости оп- тимального минимального остовного дерева, с максимальной степенью, равной или меньшей, чем к. Алгоритмы построения минимального остовного дерева можно рассматривать в виде матроидов. которые представляют собой системы подмножеств, замкнутые по включе- нию. Максимальное взвешенное независимое множество матроида можно найти, исполь- зуя "жадный" алгоритм. Связь между "жадными" алгоритмами и матроидами была уста- новлена Эдмондсом (Edmonds) в его работе [Edm71], Теория матроидов обсуждается в [Law76] и [PS98]. Динамические алгоритмы на графах стремятся сохранять инвариант (например, мини- мальное остовное дерево) при операциях вставки или удаления ребер. В работе [HdITO 1 ] представлен эффективный детерминистический алгоритм для поддержания минимальных остовных деревьев (и нескольких других инвариантов) за амортизированное полилога- рифмическое время при каждом обновлении. Алгоритмы генерирования остовных деревьев в порядке возрастания веса обсуждаются в работе [Gab77] Полный набор остовных деревьев невзвешенного графа можно сгенери- ровать за постоянное амортизированное время. Обзор алгоритмов генерирования, ранжи- рования и остовных деревьев представлено в работе [Rus03]. Родственные задачи. Дерево Штейнера (см. раздел 16.10), задача коммивояжера (см. раздел 16.4). 15.4. Поиск кратчайшего пути Вход. Граф G со взвешенными ребрами и две вершины, s и /. Задача. Найти кратчайший путь от вершины л к вершине / (рис. 15.4). Обсуждение. Задача поиска кратчайшего пути в графе имеет несколько применений, иногда довольно неожиданных: ♦ Транспортировка или связь. Задача состоит в том, чтобы найти наилучший мар- шрут, например, для отправки грузовика с товаром из Чикаго в Феникс или мар- шрут для направления сетевых пакетов к месту назначения. ♦ Сегментация изображения— разбиение оцифрованного изображения на области, содержащие отдельные объекты. Задача состоит в том, чтобы провести линии, раз- граничивающие эти области. При одном из возможных решений такие линии со- единяют точки х и у отрезками, не проходящими через пикселы объекта, насколько это возможно. Сетку пикселов можно смоделировать в виде графа, стоимость ребер которого отражает изменение цвета между соседними пикселами. Кратчайший путь
506 Часть II. Каталог алгоритмических задач ВХОД ВЫХОД Рис. 15.4. Поиск кратчайшего пути от точки х к точке у в таком взвешенном графе определяет наилучшую линию раз- дела. ♦ Различение омофонов. Омофонами называются слова, которые произносятся одина- ково, но пишутся по-разному и имеют разные значения. Различение омофонов представляет собой одну из основных задач в области распознавания речи. Ключом к решению этой задачи является внедрение грамматических ограничиваю- щих условий в процесс интерпретации слов в предложении. Для этого каждая по- следовательность фонем (опознанных звуков) сопоставляется со словами, которые, вероятно, им соответствуют. Потом создается граф, в котором вершины представ- ляют эти возможные трактовки слова, а смежные трактовки соединяются ребрами. Если для каждого ребра установить вес, пропорциональный вероятности перехода, то кратчайший путь будет определять наилучшую трактовку предложения. Подроб- ное описание подобного приложения изложено в разделе 6.4. ♦ Визуальное представление графа. На рисунке "центр" графа должен находиться в центре страницы. Хорошим определением центра графа будет вершина с мини- мальным расстоянием до всех других вершин графа. Чтобы найти эту центральную вершину, нужно знать расстояние (т. е. кратчайший путь) между всеми парами вер- шин. Основным алгоритмом поиска кратчайшего пути является алгоритм Дейкстры, кото- рый эффективно вычисляет кратчайший путь отданной начальной вершины х ко всем п- 1 другим вершинам. При каждой итерации алгоритм находит новую вершину v. для которой известен кратчайший путь из вершины х. Мы сопровождаем набор вершин S', к которым в настоящее время имеется кратчайший путь из вершины v. и с каждой итера- цией этот набор увеличивается на одну вершину. При каждой итерации алгоритм нахо- дит ребро (и, v), где и, и' е S и v, v' е V— S, такие, что: dist(x,и) + weight(u, v) = min dist(x,и') + weight(u'v’), (и’, vie/;’ где disi — расстояние, weight — вес, min dist — минимальное расстояние.
Глава 15. Задачи на графах с полиномиальным временем исполнения 507 Это ребро (w. v) добавляется к дереву кратчайшего пути с корнем в вершине х, которое описывает все кратчайшие пути из этой вершины. В разделе 6.3.1 приводится реализация алгоритма Дейкстры с временем исполнения О(г?Д. Как показано ниже, можно получить лучшее время исполнения, используя более сложные структуры данных. Если нам достаточно найти кратчайший путь от х к г. то выполнение алгоритма следует прервать, как только вершина у окажется в мно- жестве S. Алгоритм Дейкстры применяется для поиска кратчайшего пути из одной начальной точки на графах с положительным весом ребер. Но существуют ситуации, когда нужно воспользоваться другим алгоритмом. ♦ Является ли граф взвешенным? Если граф невзвешенный, то простой обход в ши- рину, начиная с вершины-источника, предоставит кратчайший путь ко всем другим вершинам за линейное время. Более сложный алгоритм требуется только тогда, когда ребра имеют разный вес. Обход в ширину проще и быстрее, чем алгоритм Дейкстры. ♦ Есть ли в графе ребра с отрицательным весом? Алгоритм Дейкстры предполагает, что все ребра графа имеют положительный вес. Для графов, в которых есть ребра с отрицательным весом, требуется использовать более общий, но менее производи- тельный алгоритм Беллмана-Форда (Bellman-Ford algorithm). Графы, содержащие циклы с отрицательным весом, представляют еще большую проблему. Обратите внимание, что в таком графе кратчайший путь из х в у не определен, г. к. имеется возможность пойти из вершины х по циклу с отрицательным весом и кружить по нему, делая общую стоимость сколь угодно низкой. Заметим, что добавление фиксированного значения к весу каждого ребра с целью сделать веса всех ребер положительными не решает проблему. В таком случае алго- ритм Дейкстры будет выбирать пути с наименьшим количеством ребер, даже если эти пути не были кратчайшими взвешенными путями в первоначальном графе. ♦ Как поступить, если входные данные представлены не графом, а набором геомет- рических объектов? Во многих приложениях требуется найти кратчайший путь ме- жду двумя точками при наличии геометрических объектов, например, в комнате, обставленной мебелью. Самое простое решение в таком случае заключаезся в пре- образовании входного экземпляра задачи в граф расстояний для последующей его обработки алгоритмом Дейкстры. Вершины будут соответствовать узлам на грани- цах препятствий, а ребра будут определены только между такими парами вершин, что из одной вершины видна другая. Существуют и более эффективные геометрические алгоритмы, вычисляющие крат- чайший путь непосредственно по расположению препятствий. Информацию о таких геометрических алгоритмах см. в разделе 17.14 и в подразделе "Примечания". ♦ Является ли граф бесконтурным орграфом? В бесконтурных орграфах кратчайший путь можно найти за линейное время. Сначала выполняем топологическую сорти- ровку, упорядочивая вершины таким образом, чтобы все они расположились слева направо, начиная с исходной вершины х. Очевидно, что расстояние от вершины х до
508 Часть II. Каталог алгоритмических задач самой себя. d(s. л), равно 0. Остальные вершины обрабатываются слева направо. Обратите внимание, что d(s,j) = min d(s.i) + w(i.j) (.r.l )€/'.' т. к. мы уже знаем кратчайший путь d(s. i) для всех вершин слева оз вершины I. В самом деле, большинство задач динамического программирования можно сфор- мулировать в виде задачи поиска кратчайшего пути в бесконтурном орграфе. Обра- тите внимание, что, заменив min на max, мы сможем применить тот же самый алго- ритм для поиска самого длинного пути в бесконтурном орграфе. Эту особенность можно использовать во многих приложениях, например, при составлении расписа- ний (см. раздел 1-1.9). ♦ Требуется ли найти кратчайший путь между всеми парами точек? Если требуется найти кратчайший путь между всеми парами вершин, то одно из решений — вы- полнить алгоритм Дейкстры п раз, по одному разу для каждой начальной вершины. Также можно использовать алгоритм Флойда-Варшалла, имеющий время исполне- ния О(п ). Этот алгоритм работает быстрее, чем алгоритм Дейкстры, и его легче программировать. Он также работает с отрицательным весом ребер, но не с цикла- ми. Описание алгоритма вместе с реализацией представлено в разделе 6.3.2. а в лис- тинге 15.4 приводится его псевдокод. Переменная М обозначает матрицу расстоя- ний, в которой Мч = оо, если ребро (i.j) не существует. Листинг 15.4. Алгоритм Флойда-Варшалла С' = м for k to n do for 1 to n do for j 1 to n do £>* = min(Df/, of"' + D*?1) Return Dn Ключевым обстоятельством к пониманию алгоритма Флойда-Варшалла является то, что £>,* обозначает "длину кратчайшего пути от вершины i к вершине /. который проходит через вершины 1..... k. как возможные промежуточные вершины." Обра- тите внимание, что сложность по памяти составляет всего лишь О(п~), т. к. на этапе к нам требуется знать только расстояния D* и '. ♦ Как найти кратчайший цикл в графе? Одним из применений алгоритма поиска кратчайшего пути между всеми парами вершин является поиск кратчайшего цикла графа, называющегося обхватом (girth) (рис. 15.5). Алгоритм Флойда можно использовать для вычисления расстояния du, где 1 <i <п. что является кратчайшим путем из вершины i в вершину I, или. иными словами, са- мым коротким циклом через вершину i. Возможно, это вам и требуется. Кратчайший цикл через вершину х вероятно может проходить от вершины х к вершине у и обратно к вершине х. используя дважды од- но и то же ребро. Цикл называется простым, если каждое его ребро и вершина по-
Глава 15. Задачи на графах с полиномиальным временем исполнения 509 сещаюгся только один раз. Очевидным подходом к поиску кратчайшего простого цикла является вычисление кратчайших путей от вершины i ко всем другим верши- нам с последующей проверкой наличия приемлемого ребра от каждой вершины до вершины i. Задача поиска самого длинного цикла графа включает задачу поиска гамильтонова цикла как частный случай (см. раздел 16.5). так что она является NP-полной. Рис. 15.5. Обхват, или кратчайший цикл, графа Матрицу кратчайшего пути для всех пар вершин можно использовать для вычисления нескольких полезных инвариантов, связанных с центром графа G. Эксцентриситетом вершины v графа называется кратчайший путь к самой дальней вершине от v. На этом понятии основаны некоторые другие инварианты. Радиусом графа называется мини- мальный из эксцентриситетов вершин графа, а вершина, на которой достигается этот минимум,— центральной вершиной. Диаметром называется наибольшее расстояние между вершинами графа. Реализации. Самые эффективные программы поиска кратчайшего пути написаны Эн- дрю Голдбергом (Andrew Goldberg) и его соавторами. Загрузить эти программы можно с веб-сайта http://www.avglab.com/andrew/soft.html_ В частности, программа MLB на языке C++ предназначена для поиска кратчайшего пути в графах с ребрами положительного веса, выраженного целыми числами. Под- робности об этом алгоритме и его реализации см. в [GolOl]. Время исполнения этого алгоритма обычно только в 4—5 раз больше, чем время обхода в ширину, и он может обрабатывать графы, содержащие миллионы вершин. Также существуют высокопроиз- водительные реализации на языке С как алгоритма Дейкстры. так и алгоритма Беллма- на-Форда(см. [CGR99]). Вес библиотеки графов на C++ и Java, упомянутые в разделе 12.4. содержат, как мини- мум, реализацию алгоритма Дейкстры. Библиотека Boost Graph Library на языке C++ (см. [SLL02]) содержит обширную коллекцию алгоритмов поиска кратчайшего пути, включая алгоритмы Беллмана-Форда и Джонсона. Загрузить библиотеку можно на веб- сайте http://www.boost.org/libs/graph/doc. Библиотека LEDA (см. раздел 19.1.1) со- держит хорошие реализации на языке C++ всех рассмотренных здесь алгоритмов по- иска кратчайшего пути, включая алгоритмы Дейкстры, Беллмана-Форда и Флойда- Варшалла. Библиотека JGraphT (http://jgrapht.sourceforge.net) содержит реализации на языке Java как алгоритма Дейкстры, так и алгоритма Беллмана-Форда. Библиотека
510 Часть II. Каталог алгоритмических задач алгоритмов для этой книги содержит реализации на языке С алгоритмов Дейкстры и Флойда-Варшалла. Подробности см. в разделе 19.1.10. Соревнования D1MACS в октябре 2006 г. проводились по алгоритмам поиска кратчай- шего пути. Во время соревнований обсуждались реализации эффективных алгоритмов для различных аспектов поиска кратчайшего пути. Соответствующие материалы мож- но найти на веб-сайте http://dimacs.rutgers.edu/Challenges/. Примечания Одним из удачных описаний алгоритмов Дейкстры (см. [Dij59]), Беллмана-Форда (см. [Ве158] и [FF62]) и Флойда-Варшалла для поиска кратчайшего пути для всех пар вершин (см. [Flo62]) является книга [CLRS01]. Относительно свежий обзор алгоритмов поиска кратчайшего пути представлен в работе [ZwiOl], а обзор геометрических алгоритмов кратчайшего пути — в книге [PN04]. Самый быстрый известный алгоритм — с временем исполнения О(т -. nlogn) — для по- иска кратчайшего пути из одной вершины в графах с ребрами с положительным весом яв- ляется алгоритм Дейкстры, использующий пирамиды Фибоначчи. Экспериментальные исследования алгоритмов поиска кратчайшего пути описаны в работах [DF79] и [DGKK79], Но эти эксперименты проводились до того, как были разработаны пирамиды Фибоначчи. Сравнительно недавнее исследование можно найти в работе [CGR99]. На практике производительность алгоритма Дейкстры можно повысить с помощью эвристи- ческих методов. В работе [HSWW05] предоставляется экспериментальное исследование взаимодействия четырех таких эвристических методов. Приблизительный кратчайший путь между двумя точками разветвленных дорожных сетей нетрудно найти с помощью таких служб, как Mapquest. Решение этой задачи несколько отличается от решения обсуждаемых здесь задач поиска кратчайшего пути. Во-первых, затраты на предварительную обработку можно амортизировать, распределив их среди многих запросов поиска пути от одной точки до другой. Во-вторых, высокоскоростные магистрали большой протяженности могут свести задачу поиска кратчайшего пути к по- иску наилучших точек для въезда на магистраль и съезда с нее. И, наконец, для практиче- ских целей достаточно получить приблизительные или эвристические результаты. Алгоритм А* выполняет поиск кратчайшего пути, выбирая первый оптимальный вариант, и сопровождает его анализом нижнего предела, чтобы установить, действительно ли най- денный путь является кратчайшим путем графа. • В работах [GK.W06] и [GKW07] описана реализация алгоритма А*, способного после двух часов предварительной обработки выполнять за одну миллисекунду запросы о кратчай- шем пути между двумя точками дорожной сети национального масштаба. Во многих приложениях, кроме оптимального пути, требуется найти альтернативные ко- роткие пути. Отсюда возникает задача поиска к кратчайших путей. Существуют варианты этой задачи, в зависимости от того, должен ли требуемый путь быть простым или он мо- жет содержать циклы. Приведенная в работе [Ерр98] реализация генерирует представле- ние этих путей за время О(т + wlogn + к). Из этого представления отдельные пути можно восстановить за время О(и). Новый алгоритм и экспериментальные результаты см. в рабо- те [HMS03], Существуют быстрые алгоритмы для вычисления обхвата как общих, так и планарных графов (см. [1R78] и [DjiOO] соответственно). Родственные задачи. Потоки в сетях (см. раздел 15.9), планирование перемещений (см. раздел 17.14).
Гпава 15. Задачи на графах с полиномиальным временем исполнения 511 15.5. Транзитивное замыкание и транзитивная редукция Вход. Ориентированный граф G = (К £). Задача. Для получения транзитивного замыкания создать граф G' = (К Е1), у которого (i,j) е £'тогда и только тогда, когда в графе G существует ориентированный путь от i до j. Для получения транзитивной редукции создать небольшой граф G' = (V, £'), ко- торый содержит ориентированный путь от i до j тогда и только тогда, когда ориентиро- ванный путь от i до J существует в графе G (рис. 15.6). ВХОД ВЫХОД Рис. 15.6. Транзитивное замыкание Обсуждение. Транзитивное замыкание можно рассматривать как создание структуры данных, позволяющей эффективно решать задачу достижимости (т. е„ можно ли по- пасть в пункт х из пункта у?). После создания транзитивного замыкания на любой во- прос относительно достижимости можно найти ответ за постоянное время, просто воз- вращая значение соответствующего элемента матрицы. Транзитивное замыкание лежит в основе распространения изменений атрибутов гра- фа G. Рассмотрим, например, граф, моделирующий электронную таблицу, у которого вершины представляют ячейки таблицы, а ребро (/.J) соединяет соответствующие ячейки, если результат ячейки J зависит от ячейки /. Когда изменяется значение данной ячейки, то также требуется обновить значения всех зависящих от нее (т. е. достижимых из нее) ячеек. Идентификация этих ячеек осуществляется с помощью транзитивного замыкания графа G. Аналогичным образом, многие задачи баз данных сводятся к по- строению транзитивных замыканий. Для построения транзитивного замыкания применяются три основных алгоритма: ♦ самый простой алгоритм выполняет обход графа в ширину или глубину из одной из вершин и запоминает все посещенные вершины. Время работы алгоритма, выпол- няющего п таких обходов, равно О(п(п + т)) и ухудшается до кубического для плотных графов. Этот алгоритм легко поддается реализации, имеет хорошую про- изводительность на разреженных графах и, скорее всего, подходит для вашего при- ложения;
512 Часть II. Каталог алгоритмических задач ♦ алгоритм Варшалла создает транзитивные замыкания за время O(n'). используя простой, изящный алгоритм, идентичный алгоритму Флойда для поиска кратчайше- го пути между всеми парами, рассматриваемый в разделе 15.4. Если нас интересует только само транзитивное замыкание, но не длина получившихся путей, то можно уменьшить объем требуемой памяти, используя только один бит для каждого эле- мента матрицы. Таким образом, D* = 1 тогда и только тогда, когда вершина j дос- тижима из вершины / при посещении в качестве промежуточных только вершин 1.....кг ♦ вычислить транзитивное замыкание можно также с помощью перемножения мат- риц. Пусть Л/1 — матрица смежности графа G. Ненулевые значения матрицы М' = М*М идентифицируют все пути длиной 2 графа G. Обратите внимание, что Л/2[/, /] = Л/[/,х]-Л/[х,у]. поэтому путь (/, х,у) содержится в ЛТ[/,у]. Таким обра- зом. объединение М' выдает транзитивное замыкание Т. Кроме этого, это объ- единение можно вычислить всего лишь за O(lgn) матричных операций, используя быстрый алгоритм возведения в степень, рассмотренный в разделе 13.9. Предположительно, для больших значений п можно получить лучшую производи- тельность, используя алгоритм Штрассена для быстрого перемножения матриц, но лично я бы не стал тратить на это свое время. Так как сложность задачи построения транзитивного замыкания такая же, как и у задачи перемножения матриц, надежда на получение значительно более быстрого алгоритма невелика. Для многих графов время исполнения этих трех процедур можно значительно улуч- шить. Вспомните, что компонентой сильной связности является набор вершин, в кото- ром любая вершина достижима из любой другой вершины. Например, любой цикл оп- ределяет сильно связный подграф. Из всех вершин в любой компоненте сильной связ- ности должно быть достижимо одно и то же подмножество графа G. Таким образом, мы можем выполнить сведение нашей задачи, создав транзитивное замыкание на графе компонент сильной связности, который должен иметь значительно меньшее количест- во ребер и вершин, чем граф G. Компоненты сильной связности графа G можно найти за линейное время (см. раздел 15.1). Транзитивная редукция (также называемая минимальным эквивалентным ориентиро- ванным графом) является операцией, обратной операции транзитивного замыкания, а именно уменьшением количества ребер при сохранении первоначальных свойств дос- тижимости. Транзитивное замыкание графа G идентично транзитивному замыканию транзитивной редукции этого графа. Основным применением транзитивной редукции является минимизация пространства, удаляя из графа G повторяющиеся ребра, которые не влияют на достижимость. Необходимость в транзитивной редукции также возникает при рисовании графов, когда требуется убрать как можно больше ненужных ребер, чтобы не загромождать рисунок. Хотя граф G может иметь только одно транзитивное замыкание, он может иметь не- сколько разных транзитивных редукций, в число которых входит он сам. Мы ищем наименьшую из этих редукций, но у такой задачи существуют разные формулировки. ♦ Простой линейный алгоритм для вычисления транзитивной редукции определяет компоненты сильной связности графа G. заменяет каждую из них ориентированным
Глава 15. Задачи на графах с полиномиальным временем исполнения 513 циклом, а потом добавляет эти ребра к ребрам, связывающим разные компоненты. Хотя эта редукция, вероятно, не является оптимальной, на типичных графах она будет довольно близка к таковой. Одним из возможных недостатков этого эвристического алгоритма состоит в том, что он может вставить в транзитивную редукцию графа G ребра, которых нет в ис- ходном графе. Впрочем, это может и не представлять проблему в вашем конкрет- ном приложении. ♦ Если все ребра нашей транзитивной редукции должны существовать в графе G, то найти редукцию минимального размера не удастся. Попробуем разобраться, почему это так. Рассмотрим ориентированный граф, состоящий из одной компоненты силь- ной связности, чтобы из любой вершины можно было попасть в любую другую вершину. Для такого графа наименьшей возможной транзитивной редукцией будет просто ориентированный цикл из л ребер. Это будет возможным тогда и только тогда, когда граф G является гамильтоновым графом, что доказывает, чго задача поиска наименьшего подмножества ребер является NP-полной. Эвристическим подходом к поиску такой транзитивной редукции будет последова- тельный перебор каждого ребра и его удаление, если это удаление не меняет дан- ную транзитивную редукцию. Эффективная реализация этого процесса позволит минимизировать время, затрачиваемое на проверку достижимости. Обратите вни- мание, что ориентированное ребро (/,/) можно удалить в любом случае, когда от вершины / к вершине j имеется другой путь, минующий данное ребро. ♦ Редукцию минимального размера, в которой допустимы ребра между произвольны- ми парами вершин, можно найти за время О(п). Но для большинства приложений, скорее всего, будет достаточно описанного выше простого, но действенного алго- ритма, который, к тому же, легко программировать и который является более эффективным. Реализации. Реализация алгоритма построения транзитивного замыкания, содержа- щаяся в библиотеке Boost Graph Library, создана на основе алгоритмов, предложенных в [Nuu95]. Библиотека LEDA (см. раздел 19.1.1) содержит реализации на языке C++ алгоритмов для вычисления как транзитивного замыкания, так и транзитивной редук- ции. Подробности о библиотеке LEDA см. в [MN99], Создается впечатление, что ни одна из обычных библиотек Java не содержит реализа- ции алгоритмов для поиска транзитивного замыкания или транзитивной редукции. Но пакет Graphlib содержит библиотеку Java Transitivity, которая имеет реализации обоих этих алгоритмов. Подробности можно найти на веб-сайте http://vvww-verimag.imag.fr/ ~cotton/. Библиотека Combinatorica (см. [PSO3]) содержит реализации (на языке пакета Mathematica) алгоритмов поиска транзитивного замыкания и транзитивной редукции. Подробности см. в разделе 19.1.9. Примечания Транзитивное замыкание и транзитивная редукция обсуждаются в работе [vL90a]. Экви- валентность перемножения матриц и вычисления транзитивного замыкания была доказана Фишером (Fischer) и Мейером (Meyer) в работе [FM71], а описания алгоритмов можно найти в книге [AHU74]. 17 Зак. 3741
514 Часть II. Каталог алгоритмических задач В последнее время наблюдался повышенный интерес исследователей к транзитивному за- мыканию, нашедший свое отражение в работе [Nuu95]. Пеннер (Penner) и Прасанна (Prasanna) (см. [РР06]) улучшили производительность алгоритма Варшалла (см. [War62]) приблизительно вдвое посредством реализации, чувствительной к кэшированию. Доказательство эквивалентности транзитивного замыкания и транзитивной редукции, а также алгоритм вычисления транзитивной редукции за время О(н') были представлены в работе [AGU72]. Результаты экспериментальных исследований алгоритмов вычисления транзитивного замыкания можно найти в [Nuu95], [РРО6] и [SD75]. Важным аспектом оптимизации запросов к базам данным является оценка размера тран- зитивного замыкания. Алгоритм решения этой задачи за линейное время представлен в работе [Coh94], Родственные задачи. Компоненты связности (см. раздел 15.1), поиск кратчайшего пути (см. раздел 15.4). 15.6. Паросочетание Вход. Граф (возможно, взвешенный) G = (У, Е). Задача. Найти наибольшее подмножество ребер Е' множества Е. для которого каждая вершина множества V инцидентна, самое большее, одному ребру из подмножества Е' (рис. 15.7). ВХОД ВЫХОД Рис. 15.7. Поиск паросочетаний Обсуждение. Допустим, что мы руководим группой рабочих, и каждый из них умеет выполнять некоторое подмножество операций, необходимых для выполнения работы. Для решения этой задачи мы создадим граф, в котором вершины представляют как ра- бочих, так и операции, и соединяем ребрами рабочих с операциями, которые они могут выполнять. Чтобы не перегружать рабочих, мы должны поручить каждому рабочему одну операцию. Нашей целью является наибольший набор ребер, при котором не по- вторяются ни рабочие, ни операции, т. е. паросочетание. Паросочетание — это мощный алгоритмический инструмент, и даже удивительно, что оптимальные паросочетания могут быть найдены достаточно эффективным образом. Необходимость в применении паросочетаний на практике возникают часто.
Гпава 15. Задачи на графах с полиномиальным временем исполнения 515 Переженить группу мужчин и женщин таким образом, чтобы каждая пара была счаст- ливой, — еще один пример практического применения задачи паросочетания. В соот- ветствующем графе каждая подходящая пара соединяется ребром. В синтетической биологии (см. [МРС+06]) требуется перемешать символы в строке 5 таким образом, чтобы максимизировать количество перемещенных символов. Например, символы строки aaabc можно перемешать так, что получится строка bcaaa, в которой только один символ занимает свою исходную позицию. Это вариант задачи паросочетания, в котором вместо мужчин фигурируют символы строки, а вместо женщин — позиции в строке (от I до |S|). Ребра соединяют символы с позициями в строке, в которых пер- воначально находились другие символы. Эта основная структура паросочетаний может быть улучшена несколькими способами, и при этом останется все той же задачей о назначениях. Вы должны ответить на сле- дующие вопросы. ♦ Является ли граф двудольным? В большинстве задач о назначениях используются двудольные графы, как, например, в классической задаче о назначении заданий ра- бочим. Это благоприятное обстоятельство, т. к. для поиска паросочетаний в дву- дольных графах существуют более быстрые и простые алгоритмы. ♦ Можно ли некоторым рабочим назначать не одно, а несколько заданий? Естест- венные обобщения задачи о назначениях включают поручение некоторым конкрет- ным рабочим нескольких заданий или (что эквивалентно) назначение нескольких рабочих на одно задание. В данном случае целью является не столько паросочета- ние. сколько постановка "галочек" в сводной таблице работ. Такие требования мож- но моделировать созданием стольких копий рабочего, сколько заданий мы хотим ему поручить. Как раз этот подход применялся в предыдущем примере с переста- новками символов строки. ♦ Является ли граф взвешенным? Многие приложения паросочетаний основаны на невзвешенных графах. Предположим, мы хотим максимизировать общее количест- во выполняемых работ, равнозначных друг другу. В этом случае мы ищем паросо- четание максимальной мощности, в идеале совершенное паросочетание, в котором каждая вершина паросочетания сопоставляется с другой. Но для других приложений нам нужно дополнить каждое ребро весом, например, отражающим способность рабочего к выполнению данного задания. Теперь задача превращается в задачу создания паросочетания с максимальным весом, т. е. набора независимых ребер с максимальным общим весом. Эффективные алгоритмы для построения паросочетаний работают, создавая в графах увеличивающие пути. Для частичного паросочетания М в графе G увеличивающим пу- тем является путь Р из ребер, которые то входят в паросочетание М, то выходят из не- го. Имея такой увеличивающий путь, мы можем расширить паросочетание на одно ребро, заменив четные ребра пути Р из М нечетными ребрами этого пути. Согласно теореме Берга, паросочетание является максимальным тогда и только тогда, когда оно не содержит ни одного увеличивающего пути. Поэтому мы можем создавать паросоче- тания максимальной мощности, выполняя поиск увеличивающих путей, прекращая поиск, когда больше нет таких путей.
516 Часть II. Каталог алгоритмических задач Задача поиска паросочетания в общих графах более сложная, т. к. в них возможны уве- личивающие пути, являющиеся циклами нечетной длины (т. е. с одной и той же на- чальной и конечной вершиной). Такие циклы невозможны в двудольных графах, кото- рые не содержат их по определению. Стандартные алгоритмы вычисления паросочетания в двудольном графе основаны на потоках в сети. В них применяется простое преобразование двудольного графа в экви- валентный потоковый граф. Реализация такого подхода приведена в разделе 6.5. Однако необходимо иметь в виду, что для решения задач паросочетания во взвешен- ных графах требуются другие подходы, наиболее известным из которых является вен- герский алгоритм. Реализации. Эндрю Голдберг (Andrew Goldberg) в соавторстве с другими исследова- телями разработал высокопроизводительные коды для вычисления паросочетаний как во взвешенных, так и в невзвешенных двудольных графах. В частности. Голдберг и Кеннеди написали на языке С программу CSA для вычисления паросочетаний во взве- шенных двудольных графах, основанную на потоках в сети, масштабирующих стои- мость (см. [GK.95]). Более быстрая программа В1М для вычисления паросочетаний в невзвешенных двудольных графах, основанная на методах с использованием увеличи- вающих путей, представлена в работе [CGM+98]. Обе программы можно загрузить для некоммерческого использования с веб-сайта http://www.avglab.com/andrew/soft.html. Первое соревнование по реализации алгоритмов DIMACS (см. [JM93]) было посвяще- но, в основном, потокам в сетях и паросочетаниям. На этом соревновании было выяв- лено несколько удачных генераторов экземпляров задач, а также ряд реализаций алго- ритмов для вычисления паросочетаний максимальной мощности и паросочетаний с максимальным весом. Эти программы можно загрузить по адресу ftp:// dimacs.rutgers.edu/pub/netflow/matching/. В их число входят следующие программы: ♦ решатель на языке FORTRAN 77 для вычисления паросочетаний максимальной мощности, разработанный Мэттингли (Mattingly) и Ричи (Ritchey); ♦ решатель на языке С, разработанный Эдвардом Розбергом (Edward Rothberg), для вычисления паросочетаний максимальной мощности, реализующий алгоритм Габо- ва (Gabow's algorithm) с временем исполнения О(и3): ♦ решатель, разработанный Эдвардом Ротбергом, для вычисления паросочетаний с максимальным весом. Это более медленный, но более общий решатель, чем вы- шеупомянутый решатель этого же автора. Обширная библиотека классов C++ GOBLIN (http://www.math.uni-augsburg.de/ ~fremuth/goblin.html) предназначена для решения всех стандартных задач оптимиза- ции графов, включая вычисление паросочетаний во взвешенных двудольных графах. Библиотека LEDA (см. раздел 19.1.1) содержит эффективные реализации на языке C++ для вычисления паросочетаний как максимальной мощности, так и максимального ве- са, как на двудольных, так и на общих графах. Эффективную программу Blossom IV (см. [CR99]) на языке С для вычисления совер- шенных паросочетаний минимального веса можно загрузить с веб-сайта http:// www2.isye.gatech.edu/~wcook/software.html. Реализацию для вычисления паросочета- ний максимальной мощности в общих графах за время О(тпа(т, и)) можно загрузить с веб-сайта http://www.cs.arizona.edu/~kece/Research/software.html.
Глава 15. Задачи на графах с полиномиальным временем исполнения 517 База графов Stanford GraphBase (см. раздел 19.1.8) содержит реализацию венгерского алгоритма для вычисления паросочетаний в двудольных графах. В целях визуализации взвешенных двудольных графов Дональд Кнут использует оцифрованную версию кар- тины Мона Лиза и осуществляет в ней поиск пикселов максимальной яркости в от- дельных строчках и столбцах. Паросочетание также используется для построения ори- гинальных портретов в стиле "домино". Примечания Книга Ловаша (Lovasz) и Пламмера (Plummer) [LP86J является исключительно полным справочником по теории паросочетаний и алгоритмам их вычисления Среди обзорных статей по алгоритмам вычисления паросочетаний особого внимания заслуживает работа [Са186]. Хорошие описания алгоритмов, использующих потоки в сети для вычисления па- росочетаний в двудольных графах, содержатся в работах [CLRSOI], [Eve79a] и [Мап89], а венгерского алгоритма— в работах [Law76] и [PS98]. Самый лучший алгоритм для вы- числения максимальных паросочетаний в двудольных графах, разработанный Хопкроф- том (Hopcroft) и Карпом (Karp) (см. [НК73]), последовательно находит кратчайшие уве- личивающие пути (вместо использования сетевого потока) и исполняется за время O(-Jnm). Время исполнения венгерского алгоритма равно О(п(т •- „log//)). Алгоритм Эдмондса (Edmonds) (см. [Edm65]) для вычисления паросочетаний максималь- ной мощности представляет большой исторический интерес, поскольку он вызывает во- просы о том, какие задачи можно решить за полиномиальное время. Описания алгоритма Эдмондса можно найти в [Law76], [PS98] и [Таг83]. Реализация алгоритма Эдмондса, принадлежащая Габову (см. [Gab76]), выполняется за время О(л). Время исполнения са- мого лучшего известного алгоритма вычисления общих паросочетаний равно О(у[пт) (см. [MV80]). Рассмотрим задачу паросочетания мужчин и женщин. Предположим, входной экземпляр содержит ребра (/?,, G',) и (Д2> G2), причем в действительности мужчина В, и женщина С2 предпочитают друг друга своим текущим партнерам. В реальной жизни эти двое, скорее всего, сойдутся, расторгнув свои браки. Паросочетание, не имеющее таких пар, называет- ся устойчивым (stable). Всестороннее изучение устойчивых паросочетаний содержится в книге [GI89]. Интересно, что независимо от многообразия взаимных предпочтений муж- чин и женщин, всегда существует хотя бы одно устойчивое паросочетание Более того, такое паросочетание можно найти за время О(п2) (см. статью [GS62]). В двудольных графах размер максимального паросочетания равен размеру минимального вершинного покрытия. Это означает, что на двудольных графах задачу о минимальном вершинном покрытии и задачу о максимальном независимом множестве можно решить за полиномиальное время. Родственные задачи. Эйлеров цикл (см. раздел 15.7), потоки в сети (см. раздел 15.09). 15.7. Задача поиска эйлерова цикла и задача китайского почтальона Вход. Граф G = (V, Е). Задача. Найти самый короткий маршрут, который проходит по каждому ребру графа G, по крайней мере, один раз (рис. 15.8).
518 Часть II. Каталог алгоритмических задач ВХОД ВЫХОД Рис. 15.8. Эйлеров цикл Обсуждение. Допустим, что вам предоставили карту города и дали задание разрабо- тать маршрут для мусоровозов, или для снегоуборочных машин, или почтальонов. В каждом случае необходимо целиком пройти каждую улицу, по крайней мере, один раз. Для эффективного выполнения задания нам нужно .минимизировать время прохо- ждения маршрута или (что равносильно) общее расстояние или количество ребер гра- фа. по которым осуществляется обход. Рассмотрим другой пример: проверка системы меню телефонного автоматизированно- го справочника. Каждая опция "Нажмите такую-то кнопку для получения такой-то ин- формации" рассматривается как ребро между двумя вершинами графа. Тестировщик меню ищет самый эффективный способ обхода этого графа, стараясь пройти по каж- дому ребру, по крайней мере, один раз. Такие задачи являются вариантами задачи поиска эйлерова цикла, которую лучше все- го сравнить с головоломкой, где нужно нарисовать геометрическую фигуру, не отрывая карандаш от бумаги и не проводя его по уже нарисованным линиям. В них нужно най- ти путь или цикл в графе, который проходит по каждому ребру ровно один раз. Перечислим признаки наличия в графе эйлерова цикла или пути (цепи). ♦ Неориентированный граф содержит эйлеров цикл тогда и только тогда, когда граф связный и все его вершины имеют четную степень. ♦ Неориентированный граф содержит эйлеров путь тогда и только тогда, когда граф связный и все (кроме двух) вершины имеют четную степень. Эти две вершины бу- дут начальной и конечной точками любого пути. ♦ Ориентированный граф содержит эйлеров цикл тогда и только тогда, когда граф сильно связный и все вершины имеют одинаковые степень захода и степень исхода. ♦ Ориентированный граф содержит эйлеров путь от вершины х к вершине у тогда и только тогда, когда граф связный и все другие вершины имеют одинаковые степень захода и степень исхода, причем вершинах имеет степень захода на единицу мень- ше. чем степень исхода, а вершина у имеет степень захода на единицу больше, чем степень исхода. Эти свойства эйлеровых графов позволяют с легкостью проверить наличие цикла: сна- чала проверяем граф на связность, выполнив обход в глубину или ширину, а потом подсчитываем количество вершин с нечетной степенью. Явный поиск цикла также за-
Гпава 15. Задачи на графах с полиномиальным временем исполнения 519 нимает линейное время и выполняется следующим способом. Посредством обхода в глубину находим в графе произвольный цикл Удаляем этот цикл из графа и повторяем процедуру до тех пор, пока все множество ребер не будет разбито на циклы, не имею- щие общих ребер. Так как удаление цикла уменьшает степень каждой вершины на чет- ное число, то оставшийся граф продолжает удовлетворять все тем же эйлеровым сте- пенным условиям. Эти циклы будут иметь общие вершины (т. к. граф является связ- ным) и поэтому их можно объединить в виде "восьмерок" по общим вершинам. Объединяя таким образом все извлеченные циклы, мы создаем один цикл, содержащий все ребра. Эйлеров цикл, если такой существует в графе, решает задачу построения маршрута снегоуборочной машины, т. к. длина любого пути, проходящего по каждому ребру только один раз, должна быть минимальной. Но на практике дорожные сети редко удовлетворяют эйлеровым степенным условиям. Тогда нам приходится решать более общую задачу — задачу китайского почтальона, в которой требуется найти минималь- ный цикл, проходящий через каждое ребро, по крайней мере, один раз. Этот мини- мальный цикл никогда не пройдет более двух раз ни по какому ребру, так что удовле- творительный маршрут можно найти для любой дорожной сети. Оптимальный маршрут китайского почтальона можно создать, добавив соответствую- щие ребра в граф G, чтобы сделать его эйлеровым. В частности, мы находим в графе G кратчайший путь между каждой парой вершин с нечетной степенью. Добавив путь ме- жду двумя вершинами с нечетной степенью, мы превращаем их в вершины с четной степенью, что приближает граф G к тому, чтобы он стал эйлеровым. Задача поиска наилучшего множества кратчайших путей для добавления к графу G сводится к поиску совершенного паросочетания с минимальным весом в специальном графе G'. Для неориентированных графов вершины графа G' соответствуют вершинам нечетной степени графа G, где вес ребра (z.y) определяется как длина кратчайшего пути от вер- шины i к вершине j в графе G. Для ориентированных графов вершины графа G' соот- ветствуют вершинам графа G, несбалансированным по степени, при этом все ребра в графе G' направлены из вершин с нехваткой степени исхода в вершины с нехваткой степени захода. Таким образом, для ориентированного графа G будет достаточно ис- пользовать алгоритмы для вычисления паросочетания в двудольных графах. Когда граф становится эйлеровым, то искомый цикл можно получить за линейное время, ис- пользуя только что описанную процедуру. Реализации. Реализацию алгоритма поиска эйлерова цикла содержат многие библио- теки графов, но реализации решений задачи китайского почтальона встречаются менее часто. Мы рекомендуем реализацию на языке Java для решения задачи китайского поч- тальона, разработанную Тимблби (Thimbleby); см. [ThiO3], Программу можно загрузить с веб-сайта http://www.cs.swan.ac.uk/~csharold/cpp/index.html. Библиотека GOBLIN содержит обширную коллекцию процедур на языке C++ для всех стандартных задач оптимизации на графах, включая процедуру решения задачи китай- ского почтальона как на ориентированных, так и на неориентированных графах. Загру- зить библиотеку можно с веб-сайта http://www.math.uni-augsburg.de/~fremuth/ goblin.html. Библиотека LEDA (см. раздел 19.1.1) предоставляет все инструменты для эффективной реализации решателей для задач поиска эйлерова цикла, паросочетаний, и кратчайших путей в двудольных и общих графах.
520 Часть II. Каталог алгоритмических задач Библиотека Combinatorica (см. [PS03]) содержит реализации (на языке пакета Mathematica) для решения задач поиска эйлеровых циклов и последовательностей де Брейна (de Bruijn sequences). Подробности см. в разделе 19.1.9. Примечания История теории графов начинается в 1736 г., когда Леонард Эйлер взялся за решение за- дачи о семи мостах Кенигсберга. Город Кенигсберг (в настоящее время Калининград) расположен на берегах реки. Во времена Эйлера оба берега и два острова соединялись семью мостами. Эту планировку можно смоделировать в виде мультиграфа с четырьмя вершинами и семью ребрами. Эйлер захотел узнать, возможно ли пройти по всем мостам ровно по одному разу и возвратиться в исходную точку (и впоследствии такой маршрут получил название эйлерова цикла). Он доказал, что искомый маршрут невозможен, т. к. все четыре вершины имели нечетную степень. Мосты были разрушены во время Второй мировой войны. История возникновения задачи рассказана в книге [BLW76], Описания алгоритмов с линейным временем исполнения для создания эйлеровых циклов (см. [ЕЬе88]) можно найти в работах [Eve79a] и [Мап89]. Простым и элегантным методом создания эйлеровых циклов является алгоритм Флери (Fleury); см. [Luc91j. Начинаем об- ход с любой вершины и удаляем пройденные ребра. Единственным критерием выбора следующего ребра является отказ от прохода по мосту (т. е. ребру, удаление которого разъединит граф) до тех пор, пока не останется других вариантов. Метод поиска эйлерова пути играет важную роль в параллельных алгоритмах на графах. Многие параллельные алгоритмы начинаются с построения остовного дерева, для которо- го потом устанавливается корень по методу поиска эйлерова пути. Описания параллель- ных алгоритмов см. в таких учебниках, как [J92], а последние результаты практического применения— в работе [СВ04]. Для подсчета эйлеровых циклов в графах существуют эффективные алгоритмы (см. [НР73]). Задача пбиска кратчайшего маршрута, проходящего через все ребра графа, была впервые представлена китайским ученым Кваном (Kwan) (см. [Kwa62]), что и определило ее на- звание — задача китайского почтальона. Решение задачи китайского почтальона посред- ством алгоритма поиска паросочетаний в двудольных графах было разработано Эдмонд- сом (Edmonds) и Джонсоном (Johnson); см. [EJ73]. Этот алгоритм работает как с ориенти- рованными. так и с неориентированными графами, но для смешанных графов задача является NP-полной (см. [Рар76а]) Смешанные графы содержат как ориентированные, так и неориентированные ребра. Описание алгоритма решения задачи китайского поч- тальона приводится в [Law76]. Последовательность де Брейна (de Bruijn) S протяженностью п на алфавите Е из а симво- лов представляет собой циклическую строку длиной а", содержащую все строки длиной п, как подстроки последовательности S, каждую ровно один раз. Например, для п = 3 и S = {0. 1} циклическая строка 00011101 содержит по порядку подстроки 000. 001, 011, 111, НО, 101, 010, 100. Последовательности де Брейна можно рассматривать как "руко- водство для вскрытия сейфа", которое предоставляет самый короткий набор поворотов ручки кодового замка, причем а дает количество позиций, достаточное для перебора всех комбинаций длины п. Эти последовательности можно генерировать, создав ориентированный граф, все верши- ны которого представляют строки а" длиной п- 1, и в котором ребро (и, у) существует тогда и только тогда, когда и = sts2...s„ t и г = s2...s„^s„. Любой эйлеров цикл на таком гра- фе описывает последовательность де Брейна. Описания последовательностей де Брейна и способы их построения приведены в [Eve79a] и [PS03],
Гпава 15. Задачи на графах с полиномиальным временем исполнения 521 Родственные задачи. Паросочетание (см. раздел 15.6), гамильтонов цикл (см. раз- дел 16.5). 15.8. Реберная и вершинная связность Вход. Граф G и. возможно, пара вершин з и /. Задача. Найти наименьшее подмножество вершин (или ребер), удаление которых разъединит граф G или, как вариант, отделит вершину х от вершины t (рис. 15.9). ВХОД ВЫХОД Рис. 15.9. Выяснение числа реберной связности Обсуждение. Вопрос о связности графов часто возникает в задачах, имеющих отноше- ние к надежности сетей. Например, в контексте телефонных сетей число вершинной связности— это наименьшее количество коммутационных станций, которые нужно вывести из строя, чтобы разорвать сеть, т. е. сделать невозможной связь между любой парой исправных коммутационных станций. А число реберной связности в этом слу- чае— наименьшее количество линий связи, которые нужно разорвать, чтобы достичь этой же цели. Итак, числом реберной (вершинной) связности графа G называется наименьшее коли- чество ребер (вершин), удаление которых нарушает связность графа. Между этими двумя величинами существует тесная связь. Число вершинной связности всегда мень- ше и пи равно числу реберной связности, т. к. удаление одной вершины из каждого ребра разреза нарушает связность графа. Но возможно существование и меньшего подмножества вершин. Верхним пределом, как для реберной, так и для вершинной связности, является минимальная степень вершины, т. к. удаление всех смежных с ней вершин (или ребер, связывающих ее с соседними вершинами) разбивает граф на две части, одна из которых состоит из единственной вершины. Представляет интерес поиск ответов на следующие вопросы. ♦ Является ли граф связным? Самой простой задачей является проверка связности графа. Все компоненты связности можно найти за линейное время посредством простого обхода в глубину или ширину (см. раздел 15.1). Для ориентированных графов следует выяснить, является ли граф сильно связным, т. е. в нем существует ориентированный путь между любой парой вершин. В слабо связном графе возмож- но существование путей к узлам, из которых нет возврата.
522 Часть II. Каталог алгоритмических задач ♦ Содержит ли граф "слабое звено"? Граф G называется двусвязным, если для нару- шения его связности требуется удалить две вершины. Вершина, удаление которой приводит к потере связности графа, называется шарниром. Мост представляет со- бой аналогичное понятие для ребер. Самый простой алгоритм идентификации шарниров (или мостов) — удалять по од- ной вершине (или ребру) и после каждого удаления проверять оставшийся граф на связность с помощью обхода в глубину или ширину. Для обеих задач существуют более сложные алгоритмы на основе обхода в глубину. Реализацию см. в разде- ле 5.9.2. ♦ Как разделить граф на равные части? Нередко требуется найти небольшой разрез, который разделяет граф на приблизительно равные части. Например, мы хотим раз- делить компьютерную программу на две части, которые проще сопровождать. Для этого мы можем создать граф, вершины которого будут соответствовать процеду- рам программы. Соединим ребрами взаимодействующие вершины (процедуры, од- на из которых вызывает другую). Далее нам нужно разделить все процедуры на на- боры приблизительно одинакового размера так, чтобы минимизировать количество пар взаимодействующих процедур, расположенных по разные стороны от линии раздела. Эта задача называется задачей разбиения графа, которая рассматривается в разде- ле 16.6. Хотя эта задача является NP-полной, для ее решения существуют приемле- мые эвристические методы. ♦ Можно ли разбивать граф произвольным образом, или требуется разбить кон- кретную пару вершин? Существует две разновидности обшей задачи связности. В одной требуется найти наименьший разрез для всего графа, а в другой — наи- меньший разрез, чтобы отделить вершину 5 от вершины t. Любой алгоритм поиска связности между вершинами ли Г можно использовать с любой из л(и-1)/2 пар вершин, что даст нам алгоритм для проверки общей связности. Не так очевиден тот факт, что для проверки реберной связности будет достаточно п - 1 проходов, одна- ко, мы знаем, что после удаления разреза вершина и хотя бы одна из остальных п — 1 вершин будут находиться в разных компонентах. Компоненты связности можно найти с помощью методов, применяемых для решения задач о потоках в сети. В задаче о потоках в сети (см. раздел /5.9) взвешенный граф рассматривается как сеть труб, в которой каждое ребро/труба имеет максимальную пропускную способность. Целью задачи является максимизировать поток между двумя данными вершинами графа. Максимальный поток между вершинами v, и v; в графе G в точности равен весу наименьшего набора ребер, которые нужно удалить для разъе- динения этих вершин. Таким образом, число реберной связности можно найти, мини- мизировав поток между вершиной v, и каждой вершиной из остальных п- 1 вершин невзвешенного графа G. Почему? Потому что после удаления минимального реберного разреза вершина v, будет отделена от какой-либо другой вершины. Вершинная связность описывается теоремой Менгера (Monger's theorem), в которой утверждается, что граф является ^-связным тогда и только тогда, когда каждую пару вершин связывают, по меньшей мере, к путей, не имеющих общих вершин. Здесь тоже можно использовать задачу о потоках в сети, т. к. поток объемом к между парой вер-
Гпава 15. Задачи на графах с полиномиальным временем исполнения 523 шин свидетельствует о наличии к путей, не имеющих общих ребер. Чтобы применить теорему Менгера, создаем такой граф G’, в котором любой набор путей, не имеющих общих ребер, соответствует путям, не имеющим общих вершин, в графе G. Для этого каждая вершина v, графа G заменяется такими двумя вершинами v,j и 2. что ребро (v,.i, v,j) g Сдля всех v, е G, а каждое ребро (v„x) е G заменяется ребрами (v,;/,jc*). где j*ke {О, I} в графе G'. Таким образом, каждому из путей, не имеющих общих вер- шин. в графе G соответствуют два пути, не имеющих общих ребер, в графе G'. Реализации. Коллекция MINCUTLIB содержит коды нескольких высокопроизводи- тельных алгоритмов поиска разреза, включая алгоритмы, использующие метод потоков в сети и метод сжатия. Эти реализации были разработаны Чекури (Chekuri) и его кол- легами в ходе многочисленных экспериментов (см. [CGK+97]). Для некоммерческого использования программы можно загрузить с веб-сайта http://www.avglab.com/ andrew/soft.html. Здесь же можно загрузить и полную версию самой работы [CGK.+97], описывающей эти алгоритмы и необходимые для их быстрого исполнения эвристические методы. Большинство библиотек структур данных для представления графов, рассматриваемых в разделе 15.1, содержат процедуры проверки на связность и двусвязность. Библиотека процедур на языке C++ Boost Graph Library (см. [SLL02]) содержит процедуры провер- ки реберной связности. Загрузить библиотеку можно с веб-сайта http://www.boost.org/ libs/graph/doc. Библиотека GOBLIN (http://www.math.uni-augsburg.de/~fremuth/goblin.html) содер- жит обширную коллекцию процедур для всех стандартных задач оптимизации на гра- фах. включая процедуры выяснения реберной и вершинной связности. Библиотека LEDA на языке C++ (см. раздел 19.1.1) предоставляет широкую поддержку для выяснения низкоуровневой связности (как двусвязных, так и трехсвязных компо- нент). реберной связности и минимального разреза. Библиотека Combinatorica (см. [PS03]) предоставляет реализации (на языке пакета Mathematica) процедур выяснения реберной и вершинной связности, а также компо- нент связности, двусвязности и сильной связности с мостами и шарнирами. Подробно- сти см. в разделе 19.1 9. Примечания Хорошие описания использования потоков в сети для выяснения реберной и вершинной связности можно найти в [Eve79a] и [PS03]. Правильность этих алгоритмов основана на теореме Менгера (см. [Меп27]), утверждающей, что связность определяется количеством путей, не имеющих общих ребер, и путей, не имеющих общих вершин и связывающих пару вершин. Теорема о минимальном потоке и максимальном разрезе была доказана Фордом (Ford) и Фалкерсопом (Fulkerson); см. [FF62]. Самые быстрые алгоритмы поиска минимального разреза и реберной связности основаны на методе сжатия графа, а не на методе потоков в сети. Сжатие ребра (х,у) в графе G со- единяет две инцидентные ему вершины в одну, удаляя петли, но оставляя мультиребра. Любая последовательность таких сжатий может увеличить (но не уменьшить) минималь- ный разрез в графе G, и не меняет разрез, если не затрагивается ребро разреза. Каргер (Karger) предоставил изящный рандомизированный алгоритм поиска минимального раз- реза, основанный на наблюдении, что вероятность изменения минимального разреза в те-
524 Часть II. Каталог алгоритмических задач чение любой случайной последовательности удалений является ничтожно малой. Самая быстрая версия алгоритма Каргера имеет время исполнения (wilg’n) (см. [КагОО]). Отлич- ный обзор рандомизированных алгоритмов, включая алгоритм Каргера, представлен в книге [MR95]. Детерминистический алгоритм, использующий метод сжатия для поиска минимального разреза за время О(п(т «log/?)), можно найти в работе [NI92]. В каждой итерации этот алгоритм находит и сжимает ребро, которое доказуемо не входит в минимальный разрез. Результаты экспериментальных сравнений алгоритмов поиска минимальных разрезов представлены в работах [CGK 97J и [NOI94] Методы поиска минимального разреза нашли применение в области распознавания обра- зов, в частности, для сегментации изображений. В своей работе [ВК04] Бойков и Колмо- горов приводят экспериментальную оценку алгоритмов поиска минимального разреза в контексте такого применения. Матулой (Matula) был разработан алгоритм, не использующий методы потоков в сети, для проверки графа на реберную А-связность за время О(кп2). Для определенных небольших значений к существуют более быстрые алгоритмы выяснения А-связности. Все трехсвяз- ные компоненты графа можно сгенерировать за линейное время (см. [НТ73а]). в то время как 4-связность можно проверить за время С>(н2). Родственные задачи. Компоненты связности (см. раздел 15.1), потоки в сети (см. раз- дел 15.9), разбиение графов (см. раздел 16.6) 15.9. Потоки в сети Вход. Ориентированный граф G, каждое ребро которого e = (i,j) имеет пропускную способность сс, и две вершины — исток s и сток /. Задача. Найти максимальный поток, который можно направить из вершины s в вер- шину /, соблюдая ограничивающие условия пропускной способности каждого ребра (рис. 15.10). Обсуждение. Область применения задачи о потоках в сети гораздо шире, чем проклад- ка водопровода. Задача о потоках возникает при поиске наиболее экономически эффективного способа транспортировки продукции между множеством предприятий и множеством магазинов или при распределении ресурсов в сетях связи.
Глава 15. Задачи на графах с полиномиальным временем исполнения 525 Реальная мощь задачи о потоках в сети проявляется в том, что большое количество возникающих на практике задач линейного программирования можно смоделировать в виде задач о потоках в сети и соответствующие алгоритмы позволяют решить эти зада- чи намного быстрее, чем применение методов линейного программирования общего назначения. Несколько рассмотренных в этой книге задач на графах можно решить, используя методы потоков в сети, включая задачи поиска паросочетаний в двудольных графах, кратчайшего пути и компонент связности. Вы должны развить в себе умение распознавать возможность моделирования стоящей перед вами задачи в виде задачи о потоках в сети. Такое умение требует практического опыта и теоретических знаний. Я рекомендую сначала создать для вашей задачи мо- дель из области линейного программирования, а потом сравнить ее с моделями для двух основных классов задач о потоках в сети. ♦ Максимальный поток. Здесь мы хотим найти максимальный объем потока из вер- шины 5 в вершине г, соблюдая ограничивающие условия пропускной способности ребер графа G. Пусть переменная х(/ обозначает объем потока, проходящего из вер- шины i через ориентированное ребро (/,у). Так как объем потока, проходящего через это ребро, ограничен его пропускной способностью с„, то 0<x,;<cZJ для I < /, j < п. Кроме того, про каждую вершину, не являющуюся ни истоком, ни стоком, можно сказать, что в нее входит поток такого же объема, какой выходит из нее, поэтому Ех'/-Ёх! =0для \ <i<n. /=i ,=1 Нам нужно найти такой набор значений, который максимизирует поток, идущий в вершину /. а именно . ♦ Поток .минимальной стоимости. Здесь каждое ребро (z,j) имеет дополнительный параметр, а именно стоимость <7;/ перемещения единицы потока от вершины i к вершине j. Также установлен целевой объем потока/ который мы хотим направить от вершины 5 к вершине t с минимальной общей стоимостью. Следовательно, нам требуется найти такой набор значений, который минимизирует следующую формулу: Е^’х// /=1 соблюдая ограничивающие условия реберной и вершинной пропускной способно- сти для максимального потока, а также дополнительное ограничивающее условие, что E”=ix« =f При этом учитываются следующие особенности: ♦ наличие нескольких истоков и/или стоков. Это обстоятельство не является пробле- мой, т. к. мы можем модифицировать сеть таким образом, чтобы создать вершину- суперисток. которая питает все истоки, и вершину-суперсток, которая поглощает все стоки:
526 Часть II. Каталог алгоритмических задач ♦ пропускная способность каждого ориентированного ребра равна либо 0, либо 1. Для таких сетей существуют более быстрые алгоритмы. Подробности см. в подраз- деле "Примечания"', ♦ одинаковая стоимость всех ребер. В таком случае используйте более простые и быстрые алгоритмы для решения задачи о максимальном потоке, а не задачи о по- токе минимальной стоимости. Задача о максимальном потоке без стоимости ребер возникает во многих приложениях, включая поиск компонент реберной и вершин- ной связности и паросочетаний в двудольных графах; ♦ перемещение по сети ра'знотипных продуктов. В телекоммуникационной сети каж- дое сообщение снабжается информацией о его источнике и месте назначения. Каж- дый узел-адресат должен получать только те сообщения, которые предназначены ему, а не общий объем сообщений от всех отправителей. Эту ситуацию можно смо- делировать в виде задачи о многопродуктовом потоке, где разные запросы опреде- ляют разные продукты, и нам требуется удовлетворить все запросы, не превысив пропускную способность ребра. Если допускаются фрагментированные потоки, задачу о многопродуктовом потоке можно будет решить методами линейного программирования. К сожалению, задача о нефрагментированном многопродуктовом потоке является NP-полной даже для двух продуктов. Алгоритмы решения задач о потоках в сети могут быть сложными и требовать значи- тельных усилий для повышения их производительности. Поэтому мы настоятельно рекомендуем использовать уже имеющиеся реализации алгоритмов, вместо того, что- бы пытаться разработать свою собственную. Есть прекрасные программы, которые описаны в подразделах "Реализации" и "Примечания"Существует два основных клас- са алгоритмов вычисления потоков в сети: ♦ методы увеличивающих путей. Эти алгоритмы последовательно находят путь от истока к стоку с положительной пропускной способностью и добавляют его к об- щему потоку. Можно доказать, что поток в сети является оптимальным тогда и только тогда, когда в ней нет увеличивающего пути. Так как каждое добавление пу- ти увеличивает поток, то в итоге должен быть найден глобальный максимум. Раз- ница между алгоритмами этого типа заключается в том, как они выбирают увеличи- вающий путь. Если не проявлять аккуратность, то каждый новый увеличивающий путь будет увеличивать общий поток лишь ненамного, вследствие чего поиск мак- симального потока может занять много времени; ♦ методы проталкивания предпотока. Эти алгоритмы проталкивают потоки от одной вершины к другой, первоначально игнорируя ограничивающее условие, что для каждой вершины исходящий поток должен быть равным входящему. Алгоритмы проталкивания предпотока работают быстрее, чем методы увеличивающих путей, поскольку возможно одновременное увеличение нескольких путей. Эти алгоритмы очень популярны, и они реализованы в программах, упоминаемых в следующем подразделе. Реализации. Высокопроизводительные программы для вычисления максимальных потоков и потоков минимальной стоимости были разработаны Эндрю Голдбергом
Глава 15. Задачи на графах с полиномиальным временем исполнения 527 (Andrew Goldberg) и его коллегами. Для вычисления максимальных потоков сущест- вуют программы HIPR и PRF (см. [CG94]), причем в большинстве случаев рекоменду- ется использовать программу HIPR. Для вычисления потоков минимальной стоимости лучшей является программа CS (см. Gol97]). Все эти программы написаны на языке С и доступны (для некоммерческого использования) на сайте http://www.avglab.com/ andrew/soft.html. Библиотека GOBLIN (http://www.math.uni-augsburg.de/~fremuth/goblin.html) содер- жит обширную коллекцию процедур на языке C++ для всех стандартных задач опти- мизации на графах, включая несколько процедур поиска максимального потока и по- тока минимальной стоимости. То же самое относится и к библиотеке LEDA, если тре- буется решение, имеющее коммерческую ценность. Подробности см. в разделе 19.1.1. На первом соревновании DIMACS по реализациям алгоритмов для задач о потоках в сети и о паросочетаниях (см. [JM93]) было собрано несколько реализаций и генерато- ров. которые можно загрузить из каталога ftp://dimacs.rutgers.edu/pub/netflow/ maxflow. В число этих программ входят написанные на языке С реализация решения задачи о потоках в сети методом проталкивания предпотока и реализация одиннадцати вариантов решения задачи о потоках в сети, включая старые алгоритмы Диница (Dinic) и Карзанова (Karzanov). Примечания Лучшей книгой, посвященной задачам о потоках в сети и их применению, является [АМО93]. Хорошие описания алгоритмов для вычисления потоков в сети можно найти в [CCPS98], [CLRSOl] и [PS98]. Обсуждение сложности задач о многопродуктовом потоке (см. [Ita78]) можно найти в книге [Eve79a]. Максимальный поток и реберная связность графов взаимообусловлены. Фундаментальная теорема о минимальном потоке и максимальном разрезе была доказана Фордом (Ford) и Фалкерсоном (Fulkerson); см. [FF62], Более простые и эффективные алгоритмы вычисле- ния минимального разреза графов обсуждаются в разделе 15.8. Принято считать, что вычисление потока в сети должно занимать время О(я/л); причем наблюдается постоянное снижение временной сложности этого вычисления. Историю ал- горитмов для решения данной задачи см. в книге [АМО93]. Время исполнения самого бы- строго алгоритма вычисления потока в сети равно O(nin \g,(n2/m)) (см. [GT88]). Экспери- ментальные исследования алгоритмов вычисления потоков минимальной стоимости мож- но найти в [GKK74] и [Gol97]. Потоки информации в сети можно моделировать в виде многопродуктовых потоков, при- нимая во внимание то обстоятельство, что создание дубликатов информации и манипули- ровании ими может устранить надобность в отдельных путях от истоков к стокам, когда одну и ту же информацию нужно доставить в несколько стоков. Эти идеи используются в области сетевого кодирования (см. [YLCZ05]) для достижения теоретических пределов прохождения информационных потоков, установленных теоремой о минимальном потоке и максимальном разрезе. Родственные задачи. Линейное программирование (см. раздел 13.6}, паросочетание (см. раздел 15.6}. связность (см. раздел 15.8}.
528 Часть II. Каталог алгоритмических задач 15.10. Рисование графов Вход. I раф G. Задача. Начертить граф G таким образом, чтобы точно отобразить его структуру (рис. 15.11). ВХОД ВЫХОД Рис. 15.11. Пример графа Обсуждение. Задача рисования графов постоянно возникает в приложениях, но по сво- ей природе она не имеет четкого определения. Что такое хорошо нарисованный граф? Нам требуется найти алгоритм, который визуально представляет структуру графа самым понятным для человека образом. В то же самое время мы хотим, чтобы нарисо- ванный граф удовлетворял определенным эстетическим критериям. К сожалению, это расплывчатые критерии, по которым невозможно разработать опти- мизирующий алгоритм. Более того, очень легко создать сильно отличающиеся друг от друга представления одного и того же графа, каждое из которых будет подходящим для определенного контекста. Например, на рис. 16.10 показаны три разных начерта- ния графа Петерсена (Petersen graph). Какое из них является "правильным"? Однако существуют и более четкие критерии, позволяющие частично оценить качество рисунка графа: ♦ пересечения. Рисунок должен содержать как можно меньше пересечений ребер, чтобы не отвлекать внимание; ♦ площадь. Рисунок должен занимать как можно меньшую площадь, и в то же самое время никакие две вершины не должны быть расположены слишком близко друг к другу; ♦ длина ребер. Рисунок не должен содержать длинные ребра, г. к. они перекрывают другие элементы рисунка;
Глава 15. Задачи на графах с полиномиальным временем исполнения 529 ♦ угловое разрешение. Рисунок не должен содержать очень острые углы между реб- рами, инцидентными одной вершине, г. к. это может вызвать частичное или полное наложение линий друг на друга; ♦ пропорциональность сторон. Соотношение высоты к ширине рисунка должно, по возможности, совпадать с соответствующей характеристикой целевого носителя (как правило, это соотношение сторон экрана компьютерного монитора, рав- ное 4/3). К сожалению, эти критерии противоречат друг другу, и задача выбора наилучшего ри- сунка графа из непустого подмножества таких рисунков, вероятно, будет NP-полной. Здесь необходимо сделать два замечания. Для графов, не имеющих естественной сим- метрии или структуры, существование по-настоящему хорошего рисунка, скорее всего, просто невозможно. Это особенно относится к графам, имеющим более 10-15 вершин. Рисунок большого, плотного графа сможет отобразить не каждый монитор. Полный граф из 100 вершин содержит приблизительно 5 000 ребер. На мониторе с разрешени- ем 1 000х 1 000 пикселов получается по 200 пикселов на ребро. Что можно увидеть в таком случае? Учитывая все сказанное, следует признать, что алгоритмы рисования графов могут быть довольно эффективными. Кроме того, экспериментирование с ними может доста- вить массу удовольствия. Чтобы выбрать подходящий алгоритм для рисования вашего графа, попытайтесь найти ответ на следующие вопросы. ♦ Должны ли все ребра быть прямыми. или допускаются дуги? Алгоритмы, рисую- щие прямые линии, сравнительно просты, но имеют свои ограничения. Для визуа- лизации сложных графов, например, электрических схем, лучше всего подходят прямоугольные ломаные (такие, что все отрезки расположены или горизонтально, или вертикально, т. е. наклонные линии не разрешаются). При этом каждое ребро графа представлено последовательностью вертикальных и горизонтальных отрез- ков, соединенных вершинами или точками изгибов. ♦ Возможен ли естественный, специфический для приложения рисунок? Если ваш граф представляет дорожную сеть, то вы вряд ли найдете лучший способ нарисо- вать его иначе, чем поместив все вершины в те же позиции, что и города на карте. Этот принцип действителен для многих других приложений. ♦ Является ли граф планарным графом или деревом? В таком случае воспользуйтесь одним из алгоритмов, описанных в разделах 15.11 и 15.12, для рисования планар- ных графов или деревьев. ♦ Является iu граф ориентированным? Ориентация ребра оказывает значительное влияние на характеристики рисунка. При рисовании ориентированных бесконтур- ных графов важно, чтобы все ребра были направлены в соответствии с некоторые логическим принципом, например, слева направо или сверху вниз. ♦ Насколько быстрым должен быть ваш алгоритм? Если алгоритм предназначается для интерактивного обновления и вывода графов на экран, то он должен работать очень быстро. В таком случае, ваш выбор ограничен инкрементальными алгорит- мами, которые изменяют положение вершин только в непосредственной близости от редактируемой вершины. Если же вы рисуете красивое изображение для дли-
530 Часть II Каталог алгоритмических задач тельного изучения, то вы можете потратить дополнительное время на его оптими- зацию. ♦ Имеет ли граф оси симметрии? Граф-результат на рис. 15.11 выглядит привлека- тельно, потому что исходный граф имеет горизонтальную ось симметрии. Оси сим- метрии в графе можно определить, найдя его автоморфизмы (изоморфизмы с са- мим собой). Все автоморфизмы графа можно без труда найти с помощью программ поиска изоморфизмов (см. раздел 16.9). Я рекомендую сначала создать черновой рисунок графа: расположить вершины по кру- гу на одинаковом расстоянии друг от друга и соединить их прямыми ребрами. Такие рисунки легко поддаются программированию и быстро выполняются. Их значительное преимущество заключается в том, что никакие два ребра не накладываются друг на друга, т. к. в таких рисунках нет трех вершин, расположенных в одну линию. Этих не- желательных эффектов трудно избежать, когда в рисунке разрешается иметь внутрен- ние вершины. Приятным сюрпризом, связанным с круговыми рисунками графов, мо- жет оказаться иногда проявляющаяся симметрия, обусловленная тем, что вершины вы- водятся в том порядке, в котором они определяются в графе. Рисунок можно значительно улучшить, минимизировав длину ребер или количество их пересечений за счет перестановки вершин методом имитации отжига. Хороший универсальный эвристический алгоритм моделирует граф в виде системы пружин, а при расстановке вершин применяет принцип минимизации потенциальной энергии. Пусть смежные вершины притягивают друг друга с усилием, пропорциональ- ным логарифму расстояния между ними, в то время как несмежные вершины отталки- вают друг друга с усилием, пропорциональным расстоянию между ними. Эти условия заставляют ребра укорачиваться и в то же самое время вынуждают вершины удаляться друг от друга. Поведение такой системы можно аппроксимировать, определив силу, действующую на каждую вершину в определенный момент времени, а потом перемес- тив каждую вершину на небольшое расстояние в соответствующем направлении. По- сле нескольких таких итераций система должна стабилизироваться на приемлемом ри- сунке графа. Рисунок 15.11 демонстрирует эффективность встраивания пружин в кон- кретный небольшой граф. Если вам нужен алгоритм, рисующий граф ломаными линиями, я рекомендую изучить системы, представленные далее, и системы, описанные в книге [JM03], чтобы решить, может ли какой-либо из них подойти для решения вашей задачи. Чтобы разработать свой собственный алгоритм рисования графов, вам придется выполнить значительный объем работы. Задача рисования графов имеет свои "подводные камни", связанные с расстановкой меток у ребер и вершин. Метки нужно помещать близко к ребрам и вершинам, чтобы однозначно идентифицировать их. но. в то же время, они не должны накладываться ни на элементы графа, ни друг на друга. Можно доказать, что оптимизация размещения меток является NP-полной задачей, но она может быть эффективно решена эвристиче- скими методами, используемыми для решения задачи разложения по контейнерам (см. раздел 17.9). Реализации. Одной из популярных и хорошо поддерживаемых программ для рисова- ния графов является программа GraphViz, разработанная Стивеном Нортом (Stephen
Глава 15. Задачи на графах с полиномиальным временем исполнения 531 North) из компании Bell Laboratories. Подробности см. на сайте http://www. graphviz.org. Программа отображает ребра в виде сплайнов и может создавать рисунки довольно больших и сложных графов. Собственно говоря, возможностей этой про- граммы было достаточно для удовлетворения всех моих профессиональных требова- ний по рисованию графов на протяжении нескольких лет. Все библиотеки структур данных для представления графов, упомянутые в разде- ле 12.4, содержат средства визуализации графов. Библиотека Boost Graph Library пре- доставляет интерфейс к программе GraphViz. Особенно подходящими для интерактив- ных приложений являются библиотеки графов на языке Java, в особенности библиоте- ка JGraphT (http://jgrapht. sourceforge.net). Для задачи рисования графов существуют очень хорошие коммерческие программы, включая программное обеспечение компании Toni Sawyer Software (www. tomsawyer.com), семейство продуктов yFiles (www.yworks.com) и пакет iLOG JViews (http://www-01.ibm.com/software/integration/visualization/jvicws/enterprise/). Пакет Pajek (см. [NMB05]) специально предназначен для рисования социальных сетей. Загру- зить пакет можно с веб-сайта http://vlado.fmf.uni-lj.si/pub/networks/pajek. Все эти па- кеты предоставляют бесплатные пробные версии или версии для некоммерческого ис- пользования. Библиотека Combinatorica (см. [PS03]) содержит несколько реализаций (на языке паке- та Mathematica) алгоритмов рисования графов, включая круговые, пружинные и упоря- доченные укладки. Для дополнительной информации по Combinatorica, см. раз- дел 19.1.9. Примечания Существует многочисленное сообщество исследователей в области рисования графов, ко- торое стимулирует проведение ежегодной конференции, посвященной этой теме. Прото- колы конференции публикуются издательством Springer-Verlag в серии Lecture Notes in Computer Science. Ознакомившись с этими протоколами, вы получите хорошее представ- ление о последних достижениях в области рисования алгоритмов и о направлениях, в ко- торых ведутся разработки. Книга [Тат08], является, пожалуй, наиболее полным обзором этой области. Отличными книгами по алгоритмам рисования графов являются |ВЕТТ99] и [KW01]. Материал книги [JM03] организован по системам, а не по алгоритмам, однако содержит технические подробности о методах рисования, применяемых в каждой системе. Эври- стические алгоритмы для маркировки графов описываются в работах [BDY06] и [WW95]. Задача равномерного размещения п точек вдоль окружности является тривиальной. Но за- дача размещения точек на поверхности сферы намного труднее. Для облегчения решения этой задачи были разработаны обширные таблицы для п < 130 (см. [HSS07]). Родственные задачи. Рисование деревьев (см. раздел 15.11). проверка на планарность (см. раздел 15.12). 15.11. Рисование деревьев Вход. Дерево Т, являющееся графом, не содержащим циклов. Задача. Создать рисунок дерева Т (рис. 15.12).
532 Часть II. Каталог алгоритмических задач Обсуждение. Во многих приложениях требуется создавать рисунки деревьев. Диа- граммы деревьев часто применяются для отображения иерархической структуры ката- логов файловой системы и их обхода. Поиск в Google по фразе "tree drawing software” ("программы рисования деревьев") возвращает около 88 тысяч результатов, включая специализированные приложения для визуализации генеалогических деревьев, синтак- сических деревьев (дерево грамматического разбора предложения), а также эволюци- онные филогенетические деревья. Каждое приложение предъявляет свои требования к внешнему виду дерева. Однако основным вопросом в области рисования деревьев является выяснение, какое дерево нужно нарисовать — свободное или корневое. ♦ Корневые деревья определяют иерархически упорядоченную структуру, исходящую из одного узла, называемого корнем. Любой рисунок такого дерева должен отра- жать иерархию его элементов, а также любые специфичные для приложения огра- ничивающие условия на порядок, в котором должны отображаться дочерние узлы. Например, генеалогические деревья имеют корень, а дочерние узлы в них обычно отображаются слева направо, по очередности рождения. ♦ Свободные деревья не содержат никакой информации о структуре, кроме топологии соединений. Например, минимальное остовное дерево графа не имеет корня, поэто- му иерархическое отображение этого дерева не имеет смысла. Рисунки таких де- ревьев могут обладать свойствами рисунка полного дерева, например, дорожной карты, на которой расстояние между населенными пунктами определяет минималь- ное остовное дерево Деревья являются планарными графами, поэтому их можно и нужно рисовать, без пе- ресечения ребер. Для этой цели годится любой алгоритм рисования на плоскости из раздели 15.12, но для создания рисунков деревьев на плоскости можно использовать намного более простые алгоритмы. В частности, эвристические алгоритмы, основан-
Гпава 15. Задачи на графах с полиномиальным временем исполнения 533 ные на "встраивании" пружин (см. раздел 15.10). хорошо подходят для рисования сво- бодных деревьев, хотя в некоторых приложениях они могут оказаться слишком мед- ленными. В большинстве естественных алгоритмов рисования деревьев предполагается, что они работают с корневыми деревьями. Но их можно с таким же успехом применять и для рисования свободных деревьев, выбрав одну из вершин в качестве корневой. Этот псевдокорень можно выбрать произвольно или, что даже лучше, использовать для него центральную вершину дерева. Центральная вершина минимизирует максимальное рас- стояние до других вершин. У деревьев центром всегда является или одна, или две смежных вершины Этот центр дерева можно найти за линейное время, последователь- но удаляя все листья, пока не останется только центральный узел. При рисовании корневых деревьев вы имеете выбор между упорядоченной и радиаль- ной укладками. ♦ Упорядоченная укладка. Корень помещается вверху по центру страницы, после чего страница разбивается сверху вниз на полосы, количество которых равно степени корня. Удаление корня создает несколько поддеревьев, количество которых равно степени корня и каждое из которых расположено в своей собственной полосе. Каж- дое поддерево рисуется рекурсивно, причем его новый корень (вершина, смежная со старым корнем) помещается в центре его полосы на фиксированном расстоянии от верха страницы, а старый корень соединяется линией с новым. На рис. 15.12 изо- бражена упорядоченная укладка сбалансированного двоичного дерева. Такие упорядоченные укладки особенно эффективны для корневых деревьев, пред- ставляющих какую-либо иерархическую структуру, будь то генеалогическое дерево, структура данных или служебная лестница. Расстояние по вертикали показывает удаленность каждого узла от корня. К сожалению, такое последовательное деление на полосы, в конце концов, приводит к появлению очень узких полосок, вследствие чего большинство вершин оказывается в небольшой части страницы. Старайтесь отрегулировать ширину каждой полосы так, чтобы отразить общее количество уз- лов. которые она будет содержать. И не бойтесь занять соседнюю область на полосе после завершения построения более коротких поддеревьев. ♦ Радиальная укладка Свободные деревья лучше рисовать, используя радиальную укладку, в которой корень или центр дерева помещается в центре страницы. Про- странство вокруг этой центральной вершины разделяется на секторы для каждого поддерева. Хотя при этом также может возникнуть проблема "скученности", ради- альные укладки используют место на странице эффективнее, чем упорядоченные, и являются более естественными для свободных деревьев. Ранг вершин в терминах расстояния от центра иллюстрируется концентрическими кругами вершин. Реализации. Одной из наиболее популярных программ для рисования графов является программа GraphViz, разработанная и Стивеном Нортом (Stephen North) из компании Bell Laboratories. Подробности см. на сайте http://www.graphviz.org. Программа пред- ставляет ребра в виде сплайнов и может создавать рисунки довольно больших и слож- ных графов. В течение многих лет эта программа удовлетворяет все мои профессио- нальные потребности в рисовании графов.
534 Часть II. Каталог алгоритмических задач Для рисования графов и деревьев существуют очень хорошие коммерческие про- граммы. включая программное обеспечение компании Tom Sawyer Software (www.tomsawyer.com). семейство продуктов yFiles (www.yworks.com) и пакет iLOG J Views (http://www-01.ibm.com/software/integration/visualization/jviews/enterprise/). Все эти пакеты предоставляют бесплатные пробные версии или версии для некоммер- ческого использования. Библиотека Combinatorica (см. [PS03]) содержит несколько реализаций (на языке паке- та Mathematica) алгоритмов рисования деревьев, включая упорядоченные и радиальные укладки. Дополнительную информацию по Combinatorica см. в разделе 19.1.9. Примечания Все книги и обзоры по рисованию графов содержат описания алгоритмов, специально предназначенных для рисования деревьев. Книга [Тат08] является, пожалуй, наиболее полным обзором положения дел в этой области. Отличными книгами по алгоритмам ри- сования графов являются [ВЕТТ99] и [KW01J. Материал книги [JM03] организован по системам, а не по алгоритмам, однако содержит технические подробности о методах ри- сования, применяемых в каждой системе. Исследования эвристических алгоритмов для укладки деревьев предпринимались многи- ми учеными (см. например, работы [RT81, Мое90]), а работа [BJL06] отражает последние достижения в этой области. При некоторых ограничениях, носящих эстетический харак- тер, задача рисования графов является NP-полной (см. [SR83]). Некоторые алгоритмы укладки деревьев получаются из приложений, не связанных с рисо- ванием графов. Метод Емде Боаса организации двоичного дерева обеспечивает более вы- сокую производительность при работе с внешними устройствами хранения данных, чем обычный двоичный поиск, правда, за счет более сложной реализации Информацию по этой и другим нечувствительным к кэшированию структурам можно найти в [ABF05], Родственные задачи. Рисование графов (см. раздел 15.10), рисование на плоскости (см. раздел 15.12). 15.12. Планарность Вход. Граф G. Задача. Выяснить, является ли граф G планарным, т. е возможно ли нарисовать его на плоскости так, чтобы никакие два ребра не пересекались. Если возможно, создать та- кой рисунок (рис. 15.13). Обсуждение. Рисование на плоскости (или укладка) графа дает ясное представление о его структуре, поскольку не содержит ни одного пересечения ребер, которое можно было бы принять за вершину. Графы, моделирующие дорожные сети, компоновку пе- чатных плат и т. п., являются планарными по своей сути, т. к. они полностью опреде- ляются плоскостными структурами. Планарные графы обладают разнообразными свойствами, которые можно использо- вать, чтобы получить более быстрые алгоритмы для решения многих задач. Самый важный факт, который вы должны знать, заключается в том, что все планарные графы являются разреженными. Для любого нетривиального планарного графа G = (V.E) справедлива формула Эйлера |£|<3|Р] —6. Это означает, что у каждого планарного
Глава 15. Задачи на графах с полиномиальным временем исполнения 535 графа количество ребер линейно зависит от количества вершин, а также, что каждый планарный граф имеет вершину со степенью <5. Любой подграф планарного графа яв- ляется планарным, поэтому должна существовать последовательность вершин низкой степени, которые можно удалить из графа G. сводя его к пустому графу. Чтобы вы получили более полное представление о тонкостях рисования графов на плоскости, я рекомендую вам создать укладку для графа А5 — е, показанного на рис. 15.13. а потом попытаться создать такую укладку, в которой все ребра прямые. Затем добавьте к графу недостающее ребро и попробуйте проделать то же самое для графа А5. Исследование планарности значительно способствовало развитию теории графов. Од- нако следует признать, что на практике необходимость в проверке графов на планар- ность возникает сравнительно редко. Большинство систем рисования графов не стре- мится специально создавать планарные укладки. На сайте этой книги. Algorithm Repository (http://www.cs.sunysb.edu/~algorith). тема выяснения планарности является самой малопосещаемой (см. [Ski99]). Несмотря на это. вам полезно уметь обращаться с планарными графами. Необходимо понимать разницу между задачей проверки графа на планарность (т. е. выяснением, возможен ли рисунок графа на плоскости) и задачей создания планарной укладки (т. е. собственно созданием рисунка), хотя и то и другое можно выполнить за линейное время. Многие эффективные алгоритмы для работы с планарными графами не используют рисование, а применяют описанную ранее последовательность удаления вершин с низкими степенями. Алгоритмы проверки на планарность начинают работу с размещения на плоскости произвольного цикла из графа, после чего изучают дополнительные пути в графе, со- единяющие вершины с этим циклом. При пересечении двух таких путей один из них нужно рисовать вне цикла, а другой в цикле. При попарном пересечении трех путей конфликт нельзя разрешить, и, следовательно, граф не может быть планарным. Алго- ритмы с линейным временем исполнения, выполняющие проверку на планарность.
536 Часть II. Каталог алгоритмических задач — _ а основаны на обходе в глубину, но они достаточно сложны, так что вам лучше поискать существующую реализацию, а не пытаться создать собственную. Алгоритмы выявления пересекающихся путей можно использовать для создания пла- нарной укладки, добавляя пути в рисунок по одному. К сожалению, так как эти алго- ритмы работают инкрементально, ничто не может помешать им поместить слишком много вершин и ребер в сравнительно небольшую область рисунка. Размещение чрез- мерного количества элементов графа в ограниченной области является большой про- блемой, т. к. затрудняет восприятие полученных рисунков графов. Поэтому были раз- работаны более эффективные алгоритмы, создающие решеточные укладки, в которых вершины расположены в решетке размером (2п - 4)х(« - 2). В результате никакие об- ласти рисунка не "захламлены", и ни одно из ребер не является слишком длинным. Тем не менее, полученные рисунки обычно выглядят не так естественно, как хотелось бы. Для непланарных графов часто требуется найти рисунок с минимальным количеством пересечений ребер. К сожалению, задача вычисления количества пересечений ребер графа является NP-полной. Однако существуют эвристические алгоритмы, которые выделяют большой планарный подграф из графа G, выполняют укладку этого подгра- фа, а потом вставляют по одному остальные ребра таким образом, чтобы минимизиро- вать количество пересечений. Этот способ не очень подходит для плотных графов, в которых невозможно избежать большого количества пересечений, но он хорошо pa- I ботает с почти планарными графами, такими как многослойные печатные платы, или ( сети дорог с эстакадами. Большие планарные подграфы могут быть найдены за счет модифицирования алгоритмов проверки на планарность таким образом, чтобы все об- наруженные проблемные ребра удалялись. Реализации. Библиотека LEDA (см. раздел 19.1.1) содержит реализации алгоритмов с линейным временем исполнения как для проверки на планарность, так и для создания решеточных укладок. Программа проверки планарности этих реализаций возвращает препятствующий подграф Куратовского (см. подраздел "Примечания") для любого предположительно непланарного графа, предоставляя убедительное доказательство его непланарности. Среда JGraphEd (http://www.jharris.ca/JGraphEd) для рисования графов, написанная на языке Java, содержит несколько реализаций алгоритмов проверки на планарность и создания планарных укладок, включая как реализацию алгоритма Бута-Люкера (Booth- Lueker) на основе PQ-деревьев, так и реализацию современных алгоритмов создания решеточной укладки. PIGALE (http://pigale.sourceforge.net)— это среда, включающая в себя библиотеку алгоритмов для работы с графами и редактор графов. Она написана на языке C++ и ориентирована, в основном, на планарные графы. Она содержит разнообразные алго- ритмы для создания рисунков графов на плоскости, а также эффективные алгоритмы для проверки на планарность и построения препятствующего подграфа (Ку.з или А"5). если такой существует. Алгоритм GRASP (Greedy Randomized Adaptive Search, "жадный" рандомизированный адаптивный поиск) для поиска наибольшего планарного подграфа реализован Рибейро (Ribeiro) и Ресенде (Resende) (см. [RR99]) языке FORTRAN и числится в коллекции
Глава 15. Задачи на графах с полиномиальным временем исполнения 537 алгоритмов АСМ под номером 797 (см. раздел 19.1.6). Эту реализацию можно загру- зить с веб-сайта http://wwvv.research.att.com/~mgcr/src. Примечания Куратовский в своей работе [КигЗО] впервые охарактеризовал планарные графы, а именно показал, что они не содержат подграфа, гомеоморфного графу , или Л'5. Таким образом, если вы все еще пытаетесь выполнить планарную укладку графа можете прекратить это занятие. Теорема Фари ([F48]) утверждает, что любой планарный граф можно нарисо- вать так, что все ребра будут прямыми. Первый алгоритм с линейным временем исполнения для рисования графов был разрабо- тан Хопкрофтом (Hopcroft) и Тарьяном (Tarjan); см. [НТ74]. Альтернативный алгоритм проверки на планарность на основе PQ-деревьев был разработан Бутом (Booth) и Люке- ром (Lueker); см. [BL76J. Упрощенные алгоритмы проверки на планарность описываются в работах |ВСРВ04]. [ММ96] и [SH99], Эффективные алгоритмы создания решеточных укладок размером 2п>п были впервые представлены в [6FPP90]. Книга [NR04] содержит хороший обзор алгоритмов рисования планарных графов. Внешнепланарным называется граф, который можно нарисовать так, чтобы все его вер- шины принадлежали внешней грани рисунка. Для таких графов характерно отсутствие подграфа, гомеоморфного графу К^. Распознать их и выполнить их укладку можно за ли- нейное время. Родственные задачи. Разбиение графов (см. раздел 16.6), рисование деревьев (ш. раздел 15.11).
ГЛАВА 16 Сложные задачи на графах По поводу алгоритмов для работы с графами существует достаточно циничное мнение, что любая задача, возникающая на практике, слишком сложна. Действительно, ни для одной задачи, представленной в этой главе, не предложено алгоритма с полиномиаль- ным временем исполнения. Все эти задачи являются NP-полными, за исключением за- дачи об изоморфизме графов, для которой вопрос сложности остается открытым. Теория NP-полноты гласит, что если для какой-то одной NP-полной задачи существует алгоритм с полиномиальным временем исполнения, то такой алгоритм должен сущест- вовать для всех NP-полных задач. "Перевернув" это утверждение, мы получим, что сведение задачи к заведомо NP-полной является достаточным доказательством отсут- ствия эффективного алгоритма решения исходной задачи. Тем не менее, не стоит терять надежду на решение вашей задачи, даже если она рас- сматривается в этой главе. Для каждой задачи мы предлагаем наиболее эффективный подход к ее решению, будь то комбинаторный поиск, эвристический метод, аппрокси- мирующий алгоритм или алгоритмы для ограниченных экземпляров. Для сложных за- дач требуется иная методика, чем для задач, решаемых за полиномиальное время, но при правильном подходе с ними можно успешно справиться. Следующие книги будут весьма полезны вам при решении NP-полных задач: ♦ [GJ79]— классический справочник по теории NP-полноты. Он содержит краткий каталог свыше 400 NP-полных задач с соответствующими ссылками и коммента- риями. Обращайтесь к этому каталогу, как только у вас возникнет сомнение в суще- ствовании эффективного алгоритма для решения вашей задачи. К этой книге из мо- ей библиотеки я обращаюсь чаще, чем к остальным; ♦ [ACG OS] — эта исключительно полезная книга посвящена аппроксимирующим алгоритмам. Ее справочный раздел находится по адресу wwvv.nada.kth.se/ ~viggo/problemlist. Именно с него вы должны начинать поиски хорошего эвристи- ческого алгоритма для решения любой задачи; ♦ [Vaz04]— подробное изложение теории аппроксимирующих алгоритмов, написан- ное авторитетным исследователем этой области; ♦ [Нос96]— первый обзор аппроксимирующих алгоритмов для решения NP-полных задач. Стремительное развитие данной области привело к тому, что эта хорошая книга немного устарела; ♦ [Gon07]— этот справочник по аппроксимирующим и метаэвристическим алгорит- мам содержит свежие обзоры разных методов решения сложных задач, как при- кладных, так и теоретических.
Глава 16 Сложные задачи на графах 539 16.1. Задача о клике Вход. Граф G - (Г. Е). Задача. Haiini наибольшее подмножество вершин .*> е Г, такое что для всех (х. у) е 5 справедливо (г,у) е £(рис. 16.1). ВХОД Рис. 16.1. Клика Обсуждение. Почти каждый из нас может вспомнить, что в его школе существовала "клика"— группа друзей, которые постоянно держались вместе и оказывали влияние навею социальную жизнь школы. Рассмотрим граф, представляющий школьный соци- ум. Вершины этого графа соответствуют ученикам, а ребра соединяют друзей. Тогда школьная клика будет представлена кликой из теории графов, т. е. полным подграфом в графе дружеских отношений. Задача выявления "кластеров", или родственных объектов, часто сводится к поиску больших клик графа. Интересным примером использования клик является программа, разработанная Налоговой службой США для борьбы с организованным мошенничест- вом в налоговой сфере. В Налоговую службу поступает большое количество ложных деклараций, составленных в расчете на незаконное получение компенсации. Однако создание большого количества действительно разных деклараций требует больших усилий. Программа Налоговой службы создает графы, в которых вершины представ- ляют поданные налоговые декларации, и соединяет ребрами заявки, которые выглядят подозрительно похожими. Любая большая клика в графе помогает выявить мошенни- чество. Так как любые две соединенные ребром вершины представляют клику, трудность за- ключается не в выявлении хоть какой-нибудь клики, а в поиске большой клики. А это уже сложно, поскольку задача поиска максимальной клики является NP-полной. Си- туацию усугубляет тот факт, что даже аппроксимация клики с точностью до п !~~е явля- ется сложной задачей. Теоретически, задача поиска клики является самой сложной в этой книге. Итак, на что мы можем надеяться?
540 Часть II. Каталог алгоритмических задач Попытайтесь найти ответы наследующие вопросы. ♦ Будет ли достаточно максимальной клики? Максимальной называется клика, кото- рую нельзя увеличить добавлением вершины. Отсюда не следует, что максимальная клика обязательно должна быть сравнима по величине с наибольшей возможной кликой, но это вполне вероятно. Чтобы найти максимальную (и не исключено, что большую) клику, отсортируем вершины по степени в порядке убывания, занесем первую вершину в клику, а потом проверим каждую из оставшихся вершин на смежность со всеми вершинами, добавленными в клику на данном этапе. Если про- веряемая вершина удовлетворит этому условию, добавим ее в клику, а если нет, то перейдем к следующей вершине. При использовании битового вектора для пометки вершин, находящихся в клике, этот алгоритм может исполняться за время О(п з т). Альтернативный подход заключается во внесении элемента случайности в упорядо- чение вершин и принятии самой большой из максимальных клик, найденных после некоторого количества попыток. ♦ Не лучше ли искать большой плотный подграф? Настаивать на выявленнии клик при поиске кластеров рискованно, т. к. нехватка одного ребра может исключить вершину из рассмотрения. Поэтому имеет смысл искать большие плотные подгра- фы, т. е. подмножества вершин, соединенных большим количеством ребер. По оп- ределению, клики являются наиболее плотными возможными подграфами. Наибольшее множество вершин, у которых порожденный подграф имеет мини- мальную вершинную степень >к, можно найти с помощью простого алгоритма с линейным временем исполнения. Начнем с удаления всех вершин со степенью меньшей, чем к. Это может понизить степень других вершин до значения, меньшего к. если они были смежными для удаленных вершин с низкой степенью. Повторяя этот процесс до тех пор, пока степень всех оставшихся вершин не будет превышать к, мы получим наибольший подграф с высокой степенью. Можно создать реализа- цию этого алгоритма со временем исполнения О(п + т), используя списки смежно- сти и очередь с приоритетами, рассматриваемую в разделе 12.2. Продолжая удалять вершины с самой низкой степенью, мы, в конечном счете, получим клику или набор клик, но они могут состоять всего лишь из двух вершин. ♦ Как поступать с планарным графом? Планарные графы не могут содержать клики, имеющие больше четырех вершин, в противном случае они не могут быть планар- ными. Так как каждое ребро определяет клику из двух вершин, то единственными представляющими интерес кликами в планарном графе являются клики из трех и четырех вершин. Эффективные алгоритмы поиска таких небольших клик рассмат- ривают вершины в порядке возрастания их степени. Любой планарный граф должен содержать вершину со степенью, не превышающей 5 (см. раздел 15.12), поэтому чтобы найти клику, содержащую эту вершину, нужно выполнить полную проверку лишь области постоянного размера. Эта вершина удаляется, после чего остается меньший планарный граф, содержащий другую вершину низкой степени. Этот про- цесс проверки и удаления продолжается до тех пор. пока граф не станет пустым. Если вам действительно нужно найти самую большую клику графа, то единственным реальным решением будет исчерпывающий перебор с возвратом. Поиск выполняется во всех ^-подмножествах вершин, и подмножество удаляется, как только в нем обна-
Глава 16 Сложные задачи на графах 541 руживается вершина, не смежная со всеми остальными вершинами. Очевидной верх- ней границей размера максимальной клики графа G является наибольшая степень вер- шины. увеличенная на I. Более точную верхнюю границу можно получить, отсортиро- вав вершины в порядке убывания их степени. Пусть j будет наибольшим индексом, таким, что степень /-й вершины равна, по крайней мере, j — I. Самая большая клика графа содержит не больше, чем j вершин, т. к. клика размером j не может содержать вершину со степенью меньшей, чем (/- I). Чтобы ускорить поиск, из графа G следует удалить все такие вершины. Эвристические алгоритмы поиска больших клик на основе рандомизированных мето- дов, таких как метод имитации отжига, должны работать достаточно хорошо. Реализации. Набор процедур Cliquer на языке С, разработанный Патриком Остергар- дом (Patrie Ostergard), предназначен для поиска клик в произвольно взвешенных гра- фах. Процедуры основаны на алгоритме метода ветвей и границ (branch-and-bound algorithm). Загрузить код можно с веб-страницы http://users.tkk.fi/~pat/cliquer.html. Второе соревнование по реализациям DIMACS было посвящено поиску клик и не- зависимых множеств. Наборы данных и код можно загрузить по адресу ftp://dimacs.rutgers.edu. Исходные коды находятся в каталоге pub/challenge/ graph/solvers. а тестовые данные — в каталоге pub/dsj. Решатель dfmax.c реализует про- стой алгоритм метода ветвей и границ, похожий на алгоритм, описанный в работе [СР90]. А решатель dmclique.c использует "полужадный" подход для поиска больших независимых множеств, описанный в работе [JAMS91], В книге [KS99] рассматриваются программы на языке С на основе метода ветвей и границ для поиска максимальной клики, используя разные нижние границы. Загрузить программы можно с веб-страницы http://www.math.mtu.edu/~kreher/cages/Src.html. Библиотека GOBLIN (http://www.math.uni-augsburg.de/~fremuth/gohlin.html) содер- жит реализации алгоритмов метода ветвей и границ для поиска больших клик Утвер- ждается, что эти реализации могут обрабатывать большие графы, содержащие 150— 200 вершин. Примечания Самый полный обзор задачи поиска максимальных клик приводится в [ВВРР99]. Особый интерес представляет работа общества исследования операций по алгоритмам метода вет- вей и границ для эффективного поиска клик. Более свежие результаты можно найти в [JS01], Доказательство NP-полноты задачи о клике было предложено Карпом (Karp); см [Каг72]. Выполнив необходимое сведение (см. раздел 9.3.3), он установил, что задачи о клике, вершинном покрытии и независимом множестве тесно связаны, поэтому эвристические алгоритмы, которые решают одну из этих задач, также должны дать удовлетворительные решения для двух других. В задаче о подграфе с наибольшей плотностью требуется найти такое подмножество вер- шин, чтобы порождаемый ими подграф имел максимально возможную среднюю степень вершин. Клика из к вершин, очевидно, является самым плотным подграфом этого разме- ра, но неполные подграфы большего размера могут иметь более высокую степень вершин. Эта задача является NP-полной, но простые эвристические методы на основе последова- тельного удаления вершин с самой низкой степенью выдают удовлетворительные прибли-
542 Часть II. Каталог алгоритмических задач зительные решения (см. [AITTOO]). Интересное применение задачи о самом плотном под- графе, а именно обнаружение ссылочного спама в Интернете, описывается в работе [GKT05], В работе [Has82] было доказано, что клику нельзя аппроксимировать с точностью до л'12'' за исключением того случая, когда Р = \'Р (и с точностью до и1-* при более слабых пред- положениях). Родственные задачи. Независимое множество (см. раздел 16.2), вершинное покрытие (см раздсч 16.3). 16.2. Независимое множество Вход. Граф G = (V, Е). Задача. Найти такое наибольшее подмножество £ множества вершин V, в котором для каждого ребра (х, у) е Е верно, что либох g Е, либо у g Е (рис. 16.2). ВХОД ВЫХОД Рис. 16.2. Независимое множество Обсуждение. Необходимость в поиске большого независимого множества вершин возникает в задаче выбора места для точек обслуживания. Важно не поместить два сервис-центра некоего предприятия слишком близко друг к другу, чтобы избежать конкуренции между ними. Для решения этой задачи мы можем создать граф, в котором вершины представляют возможные расположения точек обслуживания, а потом соеди- нить ребрами пары точек, расположенных так близко, что они мешаю г друг другу. Максимальное независимое множество определяет наибольшее количество сервис- центров. которые мы можем расположить без внутренней конкуренции. В независимых множествах (которые также называются устойчивыми множествами) отсутствуют конфликты между элементами, вследствие чего к ним часто прибегают в задачах теории кодирования и календарного планирования. Для использования незави- симого множества в теории кодирования определяем граф, вершины которого пред- ставляют набор возможных кодовых слов, и соединяем ребрами любые две вершины, кодовые слова которых похожи друг на друга настолько, что их можно перепутать при
Глава 16. Сложные задачи на графах 543 шумовых помехах. Максимальное независимое множество этого графа определит код самой большой емкости для данного канала связи. Задача о независимом множестве тесно связана с двумя другими NP-полными зада- чами: ♦ с задачей о клике. Дополнив независимое множество, мы получим клику. Дополне- нием графа G = (И, Е) является такой граф С = (И, Е'), для которого (z,j) е Е' тогда и только, когда (z',j) й Е'. Иными словами, мы убираем имеющиеся ребра и ставим ребра туда, где их не было. Максимальное независимое множество графа G пред- ставляет собой максимальную клику графа G', следовательно, эти две задачи алго- ритмически идентичны. Таким образом, алгоритмы и реализации из раздела 16.1 можно использовать и для решения задачи о независимом множестве; ♦ с задачей вершинной раскраски. Задача вершинной раскраски графа G = (И, £) за- ключается в разбиении множества вершин V на подмножества таким образом, что- бы смежные вершины были разных цветов. Каждый цвет определяет независимое множество. Многие применения задачи о независимом множестве в задачах кален- дарного планирования в действительности являются задачами раскраски, поскольку все работы, в конце концов, должны быть завершены. Один из эвристических подходов к поиску большого независимого множества со- стоит в применении любого алгоритма вершинной раскраски и использовании са- мого большого цветового класса. Отсюда следует, что все графы с небольшим ко- личеством цветов (например, планарные или двудольные графы) имеют большие независимые множества. Самый простой эвристический алгоритм поиска независимого множества можно опи- сать так. Находим вершину с самой низкой степенью, добавляем ее в независимое множество, а потом удаляем эту и все смежные с ней вершины. Повторяя этот процесс, пока граф не окажется пустым, мы получим максимальное независимое множество, т. е. множество, которое нельзя увеличить простым добавлением вершин. Применение рандомизации или исчерпывающего перебора может дать независимые множества чуть большего размера. Задача о независимом множестве в определенной степени является двойственной зада- че паросочетания в графах. В первой задаче требуется найти большой набор вершин, не имеющих общих ребер, а во второй — большой набор ребер без общих вершин. Это наводит на мысль попробовать переформулировать NP-полную задачу о независимом множестве как задачу паросочетания. которая поддается эффективному решению. Максимальное независимое множество для дерева можно найти за линейное время следующим способом: удаляются листья, удаленные листья добавляются к независи- мому множеству, удаляются все смежные узлы, а затем процедура повторяется для по- лученных деревьев до тех пор, пока дерево не станет пустым. Реализации. Любую программу для вычисления максимальной клики в графе можно применить для поиска максимальных независимых множеств, просто дополнив граф. Поэтому все реализации, упомянутые в разделе 16.1. подходят и для решения задачи о независимом множестве.
544 Часть II. Каталог алгоритмических задач Библиотека GOBLIN (http://www.math.uni-augsburg.de/~frernuth/goblin.html) содер- жит реализацию алгоритма метода ветвей и границ для поиска независимых множеств, которые в инструкции к программе называются устойчивыми множествами. Алгоритм 787 на языке FORTRAN из коллекции алгоритмов АСМ, реализованный Ресенде (Resende), представляет собой эвристическую процедуру GRASP для поиска независимого множества (см. [RFS98]). Эту реализацию можно загрузить с веб-сайта http://www.research.att.com/~mgcr/src. Примечания Доказательство NP-полноты задачи о независимом множестве было предоставлено Кар- пом (Karp); см. [Каг72]. Эта задача остается NP-полной для планарных кубических графов (см. [GJ79]). Для задачи о независимом множестве на двудольных графах существует эф- фективное решение (см. [Law76]). Сама задача не является тривиальной, поскольку боль- шая "часть" двудольного графа не обязательно представляет собой максимальное незави- симое множество. Родственные задачи. Задача о клике (см. раздел 16 I), вершинная раскраска (см. раз- дел 16."), вершинное покрытие {см.раздел 16.3). 16.3. Вершинное покрытие Вход. Граф G = (V, Е). Задача. Найти наименьшее подмножество Sc V, для которого каждое ребро (х.у) е Е содержит, по крайней мере, одну вершину из этого множества (рис. 16.3). ВХОД ВЫХОД Рис. 16.3. Вершинное покрытие Обсуждение. Задача о вершинном покрытии является частным случаем более общей задачи о покрытии множества, которая имеет на входе произвольную коллекцию под- множеств S = (Sj. .... S„) универсального множества L/= {1, и заключается в по- иске наименьшего набора подмножеств из 5, объединение которых будет равно U. За- дача о вершинном покрытии возникает во многих приложениях, связанных с покупкой товаров, которые продаются в комплектах. Задача о покрытии множества обсуждается в разделе 18.1.
Глава 16. Сложные задачи на графах 545 Преобразуем задачу о вершинном покрытии в задачу о покрытии множества. Пусть универсальное множество V представляет множество Е ребер графа G, а множество S,— набор ребер, инцидентных вершине i. Набор вершин определяет вершинное по- крытие графа G тогда и только тогда, когда соответствующие подмножества опреде- ляют покрытие множества в данном экземпляре. Но так как каждое ребро может нахо- диться только в двух разных подмножествах, то экземпляры задачи о вершинном по- крытии проще, чем общая задача покрытия множества. Задача о вершинном покрытии является сравнительно простой среди NP-полных задач и поддается решению более эффективно, чем общая задача о покрытии множества. Задачи о вершинном покрытии и независимом множестве тесно связаны друг с другом. Так как каждое ребро в множестве Е по определению является инцидентным какой- либо вершине в любом покрытии 5. то невозможно существование ребра, обе вершины которого находились бы в множестве (VS). Таким образом, множество (И-5) должно быть независимым. Так как минимизация множества 5 равносильна максимизации множества (И-5), то эти две задачи эквивалентны. Это означает, что любая программа для решения задачи о независимом множестве может быть использована и для реше- ния задачи о вершинном покрытии. Наличие двух способов подхода к решению задачи очень удобно, т. к. в некоторых контекстах один из них может оказаться проще. Самый простой эвристический алгоритм для решения задачи о вершинном покрытии начинает работу с выбора вершины наивысшей степени, добавляет ее к покрытию, удаляет все смежные ребра, а потом повторяет процесс до тех пор. пока граф не станет пустым. На правильно построенных структурах данных этот алгоритм может быть вы- полнен за линейное время, и в большинстве случаев он выдает "довольно хорошее" покрытие. Но для некоторых входных экземпляров это покрытие может быть в Ign раз хуже, чем оптимальное. К счастью, всегда можно найти вершинное покрытие, размер которого, самое большее, в два раза больше оптимального. Для этого находим максимальное паросочетание гра- фа Л/, т. е. множество ребер, в котором ни одна пара ребер не имеет общей вершины и которое нельзя расширить, добавив к нему дополнительные ребра. Такое максимальное паросочетание можно создать инкрементальным образом: выбираем в графе произ- вольное ребро, удаляем все ребра, имеющие общую вершину с этим ребром, и повто- ряем процесс до тех пор. пока в графе не останется больше ребер. Взяв обе вершины каждого ребра в максимальном паросочетаний. мы получим вершинное покрытие. По- чему? Поскольку любое вершинное покрытие должно содержать хотя бы одну из двух вершин каждого ребра паросочетания только для того, чтобы покрыть ребра макси- мального паросочетания М, это покрытие больше минимального максимум в два раза. Этот эвристический алгоритм можно настроить так, что он будет работать быстрее, если не в теории, то на практике. Мы можем выбирать ребра паросочетания. чтобы в конечном счете удалить как можно больше других ребер, тем самым уменьшив размер максимального паросочетания и, следовательно, количество пар вершин в вершинном покрытии. Кроме того, некоторые из вершин из паросочетания М в действительности могут оказаться ненужными, поскольку все инцидентные им ребра будут "охвачены" другими выбранными вершинами. Такие вершины можно идентифицировать и уда- лить, выполнив второй проход по полученному покрытию. 18 Зак. 3741
546 Часть II. Каталог алгоритмических задач В задаче о вершинном покрытии требуется охватить все ребра, используя как можно меньшее количество вершин. Есть две другие задачи, в которых ставятся аналогичные цели: ♦ задача о доминирующем множестве. В этой задаче требуется найти такое наи- меньшее множество вершин £>, для которого каждая вершина в подмножестве (V-D) соединена ребром, по крайней мере, с одной вершиной в доминирующем множестве D. Каждое вершинное покрытие нетривиального связного графа также является и доминирующим множеством, но доминирующие множества могут быть намного меньшими, чем вершинные покрытия. Любая отдельная вершина пред- ставляет минимальное доминирующее множество полного графа К„. в то время как для вершинного покрытия требуется п— 1 вершина. Задача о доминирующем мно- жестве обычно возникает в коммуникационных сетях, т. к. оно соответствует мно- жеству узлов, которых достаточно для связи со всеми другими узлами. Задачи о доминирующем множестве можно с легкостью представить в виде экземп- ляров задачи о покрытии множества (см. раздел 18.1). Каждая вершина v, определя- ет подмножество вершин, в которое входит она сама и все смежные с нею вершины. "Жадный" эвристический алгоритм находит аппроксимацию оптимального домини- рующего множества для этого экземпляра за время O(lg«); ♦ задача о реберном покрытии. В этой задаче требуется найти наименьшее множест- во ребер такое, что каждая вершина графа инцидентна одному из ребер этого мно- жества. Задачу эту можно эффективно решить, найдя паросочетание максимальной мощности (см. раздел 15.6), а потом выбрав произвольные ребра, чтобы учесть и вершины, не вошедшие в паросочетание. Реализации. Любую программу поиска максимальной клики графа можно использо- вать для решения задачи о вершинном покрытии. Для этого нужно дополнить входной граф и выбрать вершины, которые не принадлежат клике. Поэтому мы отсылаем чита- теля к программам поиска клики, упомянутым в разделе 16.1. Очень эффективным инструментом поиска вершинного покрытия является программа COVER (см. [RHG07]) на основе стохастического алгоритма локального поиска. Про- грамму COVER можно загрузить с веб-сайта http://www.nicta.com.au/people/richters/. Библиотека графов jGraphT на языке Java (http://jgrapht.sourceforge.net) содержит реализации "жадного" и 2-аппроксимирующего эвристических алгоритмов для поиска вершинного покрытия. Примечания Впервые NP-полнота задачи о вершинном покрытии была доказана Карпом (Karp); см. [Каг72]. Несколько эвристических методов, включая рандомизированное округление, составляют основу для 2-аппроксимирующих алгоритмов поиска вершинного покрытия. Хорошие описания этих 2-аппроксимирующих алгоритмов содержатся в работах [CLRS01], [Нос96], [Pas97] и [Vaz04]. Пример "жадного" алгоритма, который может ра- ботать хуже оптимального в Ign раз, был впервые приведен в [Joh74] и изложен в книге [PS98], Экспериментальные исследования эвристических методов для поиска вершинного покрытия описаны в [GMPV06], [GW97] и [RHG07],
Глава 76. Сложные задачи на графах 547 Одна из важных нерешенных задач аппроксимирующих алгоритмов касается существова- ния аппроксимирующего решения для задачи о вершинном покрытии, имеющего коэф- фициент лучший, чем 2. Хастад (Hastad) в своей работе [Has97] доказал, что для этой за- дачи не существует аппроксимирующего алгоритма, выдающего решение, имеющее ко- эффициент лучший, чем 1,1666. Основным справочником по доминирующим множествам является монография [HHS98]. Эвристические методы решения задачи о связном доминирующем множестве рассматри- ваются в работе [GK98]. Невозможно получить аппроксимирующее решение задачи о до- минирующем множестве, имеющее коэффициент лучший, чем Q(lgn) (см. [ACG 03]). Родственные задачи. Независимое множество (см. раздел 16 2), вершинное покрытие (см. раздел 16.3). 16.4. Задача коммивояжера Вход. Взвешенный граф G. Задача. Найти цикл минимальной стоимости, проходящий через каждую вершину графа G ровно один раз (рис. 16.4). Обсуждение. Задача коммивояжера— наиболее известная изо всех NP-полных задач. В качестве причин такой известности можно назвать прикладную ценность задачи и легкость ее популярного изложения. Представьте себе коммивояжера, планирующего поездку на автомобиле в несколько населенных пунктов. Он стремится разработать наиболее короткий маршрут, позволяющий посетить эти населенные пункты и возвра- титься в исходную точку, т. е. минимизировать общую протяженность поездки? Задача коммивояжера возникает во многих транспортных приложениях. Другим важ- ным применением этой задачи является оптимизация перемещения инструментов на производстве. Рассмотрим, например, робот-манипулятор для пайки соединений на печатных платах. Наиболее эффективным маршрутом для такого манипулятора будет самый короткий маршрут, посещающий каждую точку пайки. При решении задачи коммивояжера возникает несколько вопросов. ♦ Является ли граф взвешенным? Если граф маршрута является невзвешенным, или вес каждого ребра имеет одно из двух возможных значений, то задача сводится
548 Часть II. Каталог алгоритмических задач к поиску гамильтонова цикла. Задача поиска гамильтонова цикла обсуждается в разделе 16.5. ♦ Удовлетворяет ли вход аксиоме треугольника? Наше интуитивное понятие о рас- стоянии основано на аксиоме треугольника. Эта аксиома утверждает, что d(j,j) <d(i, k) + d(k,j) для всех вершин i,j, k <t V. Геометрические расстояния удов- летворяют аксиоме треугольника, т. к. кратчайший путь между двумя точками про- ходит по прямой линии между ними. Однако цены на коммерческие авиарейсы не удовлетворяют этой аксиоме, что и является причиной трудностей с выбором само- го дешевого авиарейса между двумя точками. Эвристические методы решения зада- чи коммивояжера работают намного лучше с графами разумной степени сложности, удовлетворяющими аксиоме треугольника. ♦ Входные данные представляют собой набор из п точек или взвешенный граф? С геометрическими экземплярами часто легче работать, чем с графами. Так как па- ра точек определяет полный граф, то поиск подходящего маршрута не вызывает проблем. Вычисляя расстояния между точками лишь по мере надобности, можно сэкономить память, избавив себя от необходимости хранить матрицу расстояний размером п * п. Геометрические экземпляры по своей сути удовлетворяют аксиоме треугольника, поэтому на них можно достичь производительности, гарантирован- ной некоторыми эвристическими методами. Кроме прочего, можно воспользоваться такими геометрическими структурами данных, как kd-деревья, чтобы быстро найти непосещенные точки, расположенные близко друг к другу. ♦ Разрешено ли многократное посещение одной и той же вершины? Для многих при- ложений запрет на повторное посещение одной и той же вершины не имеет значе- ния. Например, при авиаперелетах самый дешевый маршрут облета всех пунктов может проходить через один центральный аэропорт несколько раз. Обратите вни- мание, что этот вопрос не возникает, когда входной экземпляр удовлетворяет ак- сиоме треугольника. Задача коммивояжера, в которой допускается многократное посещение вершин, легко решается с помощью любой программы, предназначенной для решения об- щей версии этой задачи, на вход которой подается обновленная матрица стоимостей D, у которой ячейка D(i,j) содержит кратчайшее расстояние от точки i к точке j. Эту матрицу можно создать, решив задачу поиска кратчайшего пути между всеми пара- ми точек (см. раздел 15.4), и она удовлетворяет аксиоме треугольника. ♦ Метрика расстояний симметрична? Метрика расстояний асимметрична, когда су- ществуют такие точки (х, у), для которых d(x, у) Ф d(y. х). На практике асимметрич- ную задачу коммивояжера решить намного труднее, чем симметричную. Старайтесь избегать таких необычных метрик расстояний. Имейте в виду, что существует спо- соб преобразования асимметричных экземпляров задачи коммивояжера в симмет- ричные. содержащие вдвое больше вершин (см. [GP07]). Вы можете воспользовать- ся им при решении вашей задачи, т. к. программы решения симметричных экземп- ляров работают намного лучше. ♦ Насколько важно найти оптимальный маршрут? В большинстве случаев доста- точно решения, полученного эвристическим методом. Когда на первый план выхо- дит оптимальность, можно применить любой из двух подходов. Методы секущих
Глава 16. Сложные задачи на графах 549 плоскостей моделируют задачу в виде задачи целочисленного программирования, а потом решают ее соответствующими методами с ослабленными ограничивающими условиями. Если полученное оптимальное решение не является целочисленным, то добавляются дополнительные ограничивающие условия, чтобы обеспечить цело- численность решения. Алгоритмы метода ветвей и границ выполняют комбина- торный поиск, тщательно соблюдая при этом верхнюю и нижнюю границы стоимо- сти маршрута. При умелом обращении такие алгоритмы могут решать задачи с ты- сячами вершин. Непрофессионалу для этого придется использовать самую лучшую существующую программу. Почти любая версия задачи коммивояжера является NP-полной, поэтому правильным подходом к ее решению будет использование эвристического метода. Эти методы обычно выдают решение, отличающее от оптимального на несколько процентов, и это- го обычно достаточно для практических целей. К сожалению, для задачи коммивояже- ра было предложено несколько десятков решений, и выбрать наиболее подходящее не так-то просто. Результаты экспериментов, излагаемые в литературе, несколько проти- воречивы. Но я рекомендую вам выбрать один из следующих эвристических подходов: ♦ использование минимальных остовных деревьев. Этот эвристический метод начина- ет с построения минимального остовного дерева, а потом выполняет обход в глуби- ну полученного дерева. При этом каждое из п — I ребер проходится ровно два раза: один раз при открытии вершин во время движения вниз и второй раз на обратном пути вверх. После этого определяется маршрут за счет упорядочивания вершин по времени их открытия. Если граф удовлетворяет аксиоме треугольника, то получив- шийся маршрут будет, самое большее, вдвое длиннее оптимального. Но на практике результат обычно еще лучше и превышает оптимальный на 15-20%. Кроме этого, время исполнения этого алгоритма ограничено временем исполнения для построе- ния минимального остовного дерева, что в случае точек на плоскости равно O(7zlg/?) (см. раздел 15.3): ♦ методы инкрементальной вставки. Эвристические алгоритмы этого класса начи- нают работу с одной вершины, а потом вставляют новые точки в этот частичный маршрут по одной, пока маршрут не будет завершен. Из алгоритмов этого класса лучше всего, по-видимому, работает версия, основанная на вставке самой дальней точки: изо всех оставшихся точек в частичный маршрут Т добавляется такая точ- ка v, для которой: |/-| max min(t7(v, v,) + d(y, v,..)) vel' j=1 Условие "min" гарантирует, что мы вставляем вершину в позицию, которая добав- ляет к маршруту наименьшее расстояние, a "max" — что самая "худшая" из таких вершин выбирается первой. Этот подход работает потому, что сначала создается черновой маршрут и лишь потом уточняются его детали. Полученные таким обра- зом маршруты обычно лишь на 5-10% длиннее оптимальных; ♦ использование к-оптимальных маршрутов. Значительно более мощными являются эвристические методы Кернигана-Лина (Kerniglian-Lin heuristics). В этих методах сначала выбирается произвольный маршрут, в который потом вносятся локальные изменения с целью его улучшения. В частности, из маршрута удаляются подмноже-
550 Часть II. Каталог алгоритмических задач ства, состоящие из к ребер, а оставшиеся после этого к цепочек снова соединяются, в расчете на получение более короткого маршрута. Маршрут называется /г-опти- мальным, когда из него больше нельзя удалить подмножество из к ребер и по- другому соединить оставшиеся цепочки. Один из эффективных способов улучше- ния результатов любого другого эвристического алгоритма задачи коммивояжера — выполнить над ним 2-оптимальное улучшение маршрута. В результате многочис- ленных экспериментов было установлено, что 3-оптимальные маршруты обычно отличаются от оптимальных лишь на несколько процентов. А для к > 3 время вы- числения возрастает значительно быстрее, чем качество решения. Метод имитации отжига предоставляет альтернативный механизм обращения ребер для оптимизации решений, полученных эвристическими методами. Реализации. Разработанная Эпплгейтом (Applegate). Биксби (Bixby). Чваталом (Chvatal) и Куком (Cook) (см. [АВСС07]) программа Concorde на языке ANSI С предна- значена для решения симметричных задач коммивояжера и подобных задач оптимиза- ции сетей. Эта программа установила несколько мировых рекордов и получила опти- мальные решения для 106 из 110 экземпляров из библиотеки экземпляров задачи ком- мивояжера (TSPLIB, http://comopt.ifi.uni-heidelberg.de/software/TSPLIB95/), самый большой из которых содержит 15 112 узлов. Программу Concorde можно загрузить для академических исследований с веб-сайта http://www.tsp.gatech.edu/concorde. Это. бесспорно, самая лучшая реализация решения задачи коммивояжера. На веб-сайте раз- работчиков (http://www.tsp.gatech.edu) можно найти очень интересный материал по истории и практическим приложениям задачи коммивояжера. Работа [LP07] Лоди (Lodi) и Пуннена (Punnen) является отличным обзором сущест- вующего программного обеспечения для решения задачи коммивояжера. Текущие ссылки на все упоминаемые в этом обзоре программы поддерживаются на веб-сайте http://www.or.deis.unibo.it/research_pages/tspsoft.html. Библиотека TSPLIB содержит коллекцию стандартных сложных экземпляров задачи коммивояжера, которые возникают на практике. Лучшую версию библиотеки TSPLIB можно загрузить с веб-сайта http://comopt.ifi.uni-heidelberg.de/software/TSPLIB95/. хотя библиотека NetLib также содержит эти экземпляры. Программа tsp_solve на языке C++, разработанная Чадом Гурвицем (Chad Hurwitz) и Робертом Крейгом (Robert Craig), предоставляет как эвристические, так и оптимальные решения. Программа может обрабатывать экземпляры, содержащие до 100 узлов. За- грузить ее можно по адресу http://www.cs.sunysb.edu/~algorith или запросить у Чада Гурвица по адресу churritz@cts.com. Библиотека GOBLIN (http://www.math.uni- augsburg.de/~fremuth/goblin.html) содержит реализации алгоритмов метода ветвей и границ для решения как симметричных, так и асимметричных версий задачи комми- вояжера, а также реализации различных эвристических алгоритмов. Алгоритм 608 на языке FORTRAN (см. [Wes83]) из коллекции алгоритмов АСМ реали- зует эвристический метод для решения квадратичной задачи о назначениях, которая является более общей задачей, включающей в себя задачу коммивояжера, как частный случай. Алгоритм 750 на языке FORTRAN (см. [CDT95]) из этой же коллекции выдает точное решение для асимметричных экземпляров задачи коммивояжера. Подробности см. в разделе 19.1.5.
Глава 16. Сложные задачи на графах 551 Примечания В книге [АВСС07] Эпплгейта (Applegate), Биксби (Bixby), Чватала (Chvatal) и Кука (Cook) задокументированы методы, использованные авторами в их программах, решающих зада- чи коммивояжера за рекордное время, а также изложены теоретические основы задачи и ее история. Самым лучшим справочником по многочисленным версиям задачи комми- вояжера является книга Гутина (Gutin) и Пуннена (Punnen) [GP07], заменившая старый и всеми любимый справочник [LLKS85]. Экспериментальные результаты применения эвристических методов для решения боль- ших экземпляров задачи коммивояжера можно найти в [Ben92a], [GBDS80] и [Rei94]. Эти методы обычно позволяют получить решение, отличающееся от оптимального лишь на несколько процентов. Эвристический алгоритм Кристофидеса (Christofides) (см. [Chr76]) представляет собой усовершенствованный вариант эвристического метода для построения минимального ос- товного дерева. На евклидовых графах он гарантирует маршрут, превышающий опти- мальный не более чем в полтора раза. Время исполнения этого алгоритма равно O(nJ), а узким местом является поиск совершенного паросочетания с минимальным весом (см. раздел 15.6). Эвристический алгоритм построения минимального остовного дерева был впервые изложен в работе [RSL77]. Аппроксимирующие методы для евклидовой задачи коммивояжера были разработаны Аророй (Arora) и Митчеллом (Mitchell) (см. [Аго98] и [Mit99]). Они выдают приблизи- тельный результат за полиномиальное время с точностью до 1 + е от оптимального для любого е > 0. Эти методы представляют большой теоретический интерес, хотя их практи- ческая ценность пока неизвестна. История поиска оптимальных решений задачи коммивояжера свидетельствует о постоян- ном прогрессе в этой области. В 1954 г. Данциг (Dantzig), Фалкерсон (Fulkerson) и Джон- сон (Johnson) решили экземпляр задачи коммивояжера для 42 городов Соединенных Шта- тов (см. [DFJ54]). В 1980 г. Падберг (Padberg) и Хонг (Hong) нашли решение экземпляра, состоящего из 318 вершин (см. [РН80]). А недавно Эпплгейт и др. (см. [АВСС07]) решили экземпляр задачи, размер которого в двадцать раз больше. Конечно, нельзя сбрасывать со счетов развитие аппаратного обеспечения, но значительная часть успехов объясняется по- явлением более удачных алгоритмов. Такой темп роста производительности свидетельст- вует о том, что при острой необходимости получение точных решений для NP-полных за- дач вполне возможно. К счастью или к сожалению, необходимость в них редко бывает острой. Для больших экземпляров размер — не единственный критерий успешности решения. Можно с легкостью создать громадный граф, состоящий из одного дешевого цикла, для которого получить оптимальное решение не составит труда. Для множества точек, обра- зующих выпуклый многоугольник на плоскости, минимальный маршрут коммивояжера определяется их выпуклой оболочкой (см. раздел 17.2), которую можно вычислить за время O(nlgn). Известны также другие простые частные случаи этой задачи. Родственные задачи. Гамильтонов цикл (см. раздел 16.5), минимальное остовное де- рево {см. раздел 15.3), выпуклая оболочка (см. раздел 17.2). 16.5. Гамильтонов цикл Вход. Граф G = (V, Е). Задача. Найти маршрут, состоящий из ребер графа, такой что каждая вершина посеща- ется ровно один раз (рис. 16.5).
552 Часть II. Каталог алгоритмических задач ВХОД выход Рис. 16.5. Гамильтонов цикл Обсуждение. Задача поиска гамильтонова цикла или пути в графе G является частным случаем задачи коммивояжера для графа С с тем условием, что длина каждого ребра графа G равна 1 в графе G'. Расстоянию между вершинами, не соединенными ребрами, присваивается большее значение, например, 2. Такой взвешенный граф содержит мар- шрут коммивояжера стоимостью п в графе С тогда и только тогда, когда граф G явля- ется гамильтоновым. С задачей поиска гамильтонова пути тесно связана задача поиска самого длинного пу- ти или цикла в графах, необходимость в котором часто возникает в задачах распозна- вания образов. В таких приложениях вершины графа соответствуют допустимым сим- волам, а ребра соединяют пары символов, которые могут быть соседними. Самый длинный путь в этом графе будет наилучшим кандидатом для правильной интерпре- тации. Обе задачи (поиска самого длинного пути и цикла) являются NP-полными, даже на не- взвешенных графах. Тем не менее, существует несколько подходов к их решению, и для выбора наилучшего вы должны найти ответы на следующие вопросы. ♦ Назначаются ли крупные штрафы за посещение вершин более одного раза? Если мы переформулируем задачу поиска гамильтонова цикла вместо того, чтобы пы- таться минимизировать общее количество вершин, посещенных при полном мар- шруте, то получим задачу оптимизации, что позволит использовать для ее решения эвристические и аппроксимирующие алгоритмы. Построив остовное дерево графа и выполнив по нему обход в глубину, как показано в разделе 16.4. мы получим мар- шрут, содержащий, самое большее, 2и вершин. Использование рандомизации или метода имитации отжига может значительно уменьшить этот число. ♦ Является ли бесконтурным орграфом граф, в котором осуществляется поиск са- мого длинного пути? Задачу поиска самого длинного пути в бесконтурном орграфе можно решить за линейное время с помощью методов динамического программи-
Глава 16. Сложные задачи на графах 553 рования. К счастью, для этого можно использовать алгоритм поиска кратчайшего пути в бесконтурном орграфе (представленный в разделе 15.4), просто заменив min на max. Бесконтурные орграфы являются наиболее интересными экземплярами за- дачи поиска самого длинного пути, для которых существуют эффективные алго- ритмы. ♦ Является чи граф плотным? Достаточно плотные графы всегда содержат гамильто- новы циклы. Более того, циклы на таких графах можно строить достаточно эффективно. В частности, любой граф, все вершины которого имеют степень, большую или равную и/2, должен быть гамильтоновым. Существуют и другие усло- вия достаточности для наличия гамильтоновых циклов; подробности см. в подраз- деле "Примечания". ♦ Нужно ли посетить все вершины или требуется пройти по всем ребрам? Убеди- тесь в том, что перед вами действительно стоит задача посещения вершин, а не ре- бер. При должной изобретательности иногда удается переформулировать задачу поиска гамильтонова цикла в терминах задачи поиска эйлерова цикла, в которой требуется пройти по каждому ребру графа. Пожалуй, самым известным таким эк- земпляром является задача создания последовательностей де Брейна, рассматри- ваемая в разделе 15.7. Польза от такой переформулировки состоит в том, что для решения задачи поиска эйлерова цикла и многих родственных ей задач существуют быстрые алгоритмы, в то время как задача поиска гамильтонова цикла является NP- полной. Если вам действительно нужно знать, является ли ваш граф гамильтоновым, то единст- венным решением этой задачи является поиск с возвратом и разрежением поискового пространства. Обязательно проверьте граф на двусвязность (см. раздел 15.8). Если граф не двусвязный, то это означает, что в нем есть шарнир, удаление которого разъединит граф, и поэтому граф не может быть гамильтоновым. Реализации. Описанная ранее конструкция (в которой вес ребер равен 1, а расстояние между несмежными вершинами равно 2) сводит задачу поиска гамильтонова цикла к симметричной задаче коммивояжера, удовлетворяющей аксиоме треугольника. Поэто- му для решения задачи поиска гамильтонова цикла можно использовать программы дла решения задачи коммивояжера, рассмотренные в разделе 16 4. Лучшей из них яв- ляется программа Concorde, написанная на языке ANSI С и предназначенная для реше- ния симметрических задач коммивояжера и подобных задач оптимизации сетей. Про- грамму Concorde можно загрузить для академических исследований с веб-сайта http://www.tsp.gatech.edu/concorde. Это, бесспорно, самая лучшая реализация реше- ния задачи коммивояжера. Эффективная программа для решения задач поиска гамильтонова цикла была разрабо- тана на основе диссертации Вандергринда (Vandegriend); см. [Van98]. Код программы и текст работы можно загрузить с веб-сайта http://webdocs.cs.ualberta.ca/~joe/ Theses/vandegricnd.html. Лоди (Lodi) и Пуннен (Punnen) написали прекрасный обзор существующего программ- ного обеспечения для решения задачи коммивояжера, включая частный случай гамиль- тонова цикла (см. [LP07]). Действующие ссылки на все упоминаемые в этом обзоре программы поддерживаются на веб-сайте http://www.or.deis.unibo.it/research_ pages/tspsoft.html.
554 Часть II. Каталог алгоритмических задач Программа для ранжирования футбольных команд (речь идет об американском футбо- ле— прим. перев.) из базы графов Standford GraphBase (см. раздел 19.1.8) использует многоуровневый "жадный" алгоритм для решения асимметричной задачи поиска само- го длинного пути. Целью задачи является получение цепочки результатов футбольных матчей, чтобы установить превосходство одной футбольной команды над другой. Ведь если команда университета Вирджинии победила команду университета Иллинойса с разрывом в 30 очков, а команда университета Иллинойса победила команду универси- тета Стоуни Брук с разрывом в 14 очков, тогда, по идее, можно считать, что если бы команда университета Вирджинии играла с командой университета Стоуни Брук, то она победила бы ее с разрывом в 44 очка, не так ли? Мы хотим найти самый длинный простой путь в графе, где вес ребра (х,у) обозначает превосходство в счете выиграв- шей команды х над проигравшей у. Эффективная программа перечисления всех гамильтоновых циклов графа методом по- иска с возвратом представлена в книге [NW78], Подробности см. в разделе 19.1.10. Ал- горитм 595 (см. [Маг83]) на языке FORTRAN из коллекции алгоритмов АСМ пред- ставляет собой аналогичную программу, которую можно использовать для получения точного или приблизительного решения, управляя количеством возвратов. Подроб- ности см. в разделе 19.1.5. Примечания По-видимому, задача поиска гамильтонова цикла впервые возникла в процессе изучения Эйлером задачи о ходе шахматного коня, хотя они были популяризированы в 1839 г. в го- ловоломке "Вокруг света”, изобретенной Гамильтоном. Обширный справочный материал по задаче коммивояжера, включающий в себя обсуждение задачи поиска гамильтонова цикла, представлен в [АВСС07], [GP07] и [LLKS85]. В большинстве учебников по теории графов обсуждаются условия достаточности для на- личия гамильтоновых циклов. Моим любимым является учебник [WesOO], В последнее время большое внимание привлекают к себе методы оптимизации лабора- торных биологических процессов. При первом применении этих методов "биовычисле- ний" Аделман (Adleman) (см. [Adl94J) решил задачу поиска гамильтонова пути на экземп- ляре с семью вершинами. К сожалению, этот подход требует экспоненциального количе- ства молекул, и, зная число Авогадро, можно утверждать, что такие эксперименты невозможны для графов с количеством вершин свыше п = 70. Родственные задачи. Эйлеров цикл (см. раздел 15.7), задача коммивояжера (см. раз- дел 16.4). 16.6. Разбиение графов Вход. Граф G = (Г, Е) (взвешенный) и целые числа к и т. Задача. Разбить множество вершин графа на т приблизительно равных подмножеств таким образом, чтобы общая стоимость ребер во всех подмножествах не превышала к (рис. 16.6). Обсуждение. Задача разбиения графа возникает во многих алгоритмах типа "разделяй и властвуй", эффективность которых основана на разбиении задачи на несколько меньших подзадач, с последующей "сборкой" решения из решений этих подзадач.
Глава 16 Сложные задачи на графах 555 ВХОД ВЫХОД Рис. 16.6. Разбиение графа Стремление к минимизации количества ребер, удаленных при разбиении графа, обыч- но упрощает задачу слияния решений подзадач в одно общее решение. Задача разбиения графов также возникает, когда нужно собрать вершины в кластеры логических компонентов. Если ребра соединяют сходные объекты, то оставшиеся по- сле разбиения кластеры должны отражать логически связанные группы. Большие гра- фы часто разбиваются на части приемлемого размера, чтобы облегчить обработку дан- ных или чтобы получить менее загроможденный рисунок. Наконец, разбиение графов является важным шагом во многих параллельных алгорит- мах. В качестве примера можно назвать метод конечных элементов, который использу- ется для расчета физических явлений (таких как механическое напряжение или тепло- передача) в геометрических моделях. Параллелизация этих вычислений требует раз- биения моделей на равные части с небольшой общей границей. Это задача разбиения графа, т. к. топология геометрической модели обычно представляется с помощью графа. В зависимости от требуемой целевой функции, могут возникать разные виды задачи разбиения графа. А именно: ♦ минимальный разрез. Наименьшее множество ребер, удаление которых разделит граф на части, можно эффективно найти, используя метод потока в сети или рандо- мизированные алгоритмы. Дополнительную информацию по алгоритмам связности см. в разделе 15.8. Но наименьший разрез может отделить только одну вершину, вследствие чего получившееся разбиение может оказаться очень несбалансирован- ным; ♦ разбиение графа. При более практичном критерии разбиения мы ищем разрез не- большого размера, который делит граф на приблизительно равные части. К сожале- нию, эта задача является NP-полной. Однако на практике она хорошо поддается решению эвристическими методами, рассматриваемыми далее. Некоторые специализированные графы обязательно имеют небольшие множества вершин-разделителей, которые разбивают граф на сбалансированные части. Любое
556 Часть II. Каталог алгоритмических задач дерево всегда имеет одну вершину, удаление которой разбивает дерево на части так, что ни одна из них не содержит больше, чем п/2 первоначальных п вершин. Эти компоненты не обязательно должны быть связными; рассмотрим, хотя бы, разде- ляющую вершину звездообразного дерева. Эту разделяющую вершину можно найти за линейное время посредством поиска в глубину. Каждый планарный граф имеет множество из (9(д/п) вершин, удаление которых не оставляет компонент, содержа- щих более 2п/3 вершин. Наличие разделителей обеспечивает удобный способ раз- ложения геометрических моделей, которые, как правило, часто представляются планарными графами; ♦ максимальный разрез. Для графов, представляющих электронные схемы, макси- мальный разрез графа (рис. 16.7) соответствует максимальному потоку данных, ко- торый может протекать в схеме. Таким образом, канал связи с самой высокой про- пускной способностью должен покрывать разбиение множества вершин, опреде- ляемое максимальным разрезом. Задача поиска максимального разреза графа является NP-полной (см. [Каг72]), но на практике хорошо поддается решению эври- стическими методами, аналогичными методам разбиения графа. Рис, 16.7. Максимальный разрез графа Основной подход к решению задач разбиения графа и поиска максимального разреза состоит в создании первоначального разбиения множества вершин (либо произволь- ным образом, либо в соответствии с некоторой стратегией, специфичной для приложе- ния) с последующим перебором всех вершин и принятием решения по поводу каждой из них, не увеличится ли размер разреза, если ее переместить в другую часть графа. Решение о перемещении вершины v можно принять за время, пропорциональное ее степени, выяснив, какая часть разбиения содержит больше ее соседей. Конечно, наибо- лее подходящая часть для расположения вершины v может измениться после переме- щения ее соседей, поэтому, скорее всего, понадобится несколько итераций, прежде чем процесс достигнет локального оптимума. Но даже в этом случае локальный оптимум может быть сколь угодно далек от глобального максимального разреза. Существует много вариантов этой базовой процедуры, отличающихся от описанного порядком рассмотрения вершин или тем, что в перемещении участвуют целые класте-
Глава 16 Сложные задачи на графах 557 ры вершин. Почти наверняка даст хороший результат использование какого-либо вида рандомизации, особенно метода имитации отжига. Если вершины нужно разбить на более чем две части, то процедуру разбиения следует применять рекурсивно. В спектральных методах разбиения используются сложные приемы линейной алгебры. В частности, для разбиения матрицы на две части методом спектральной бисекции ис- пользуется второй снизу собственный вектор матрицы Лапласа для графа. Спектраль- ные методы обычно позволяют эффективно находить общую область для разбиения, но результаты можно улучшить, обработав полученный результат методом локальной оп- тимизации. Реализации. Программа Chaco широко применяется для разбиения графов в приложе- ниях параллельных вычислений. В ней используется несколько разных алгоритмов разбиения, включая метод Кернигана-Лина и спектральный метод. Программу Chaco можно загрузить с веб-сайта http://wvvw.cs.sandia.gov/~bahendr/chaco.html. Хорошей репутацией пользуется пакет METIS, который можно загрузить с веб-сайта http://glaros.dtc.umn.edu/gkhome/views/inetis. Эта программа успешно применялась для разбиения графов, имеющих более миллиона вершин. Одна из доступных версий программы предназначена для выполнения на многопроцессорных компьютерах, а вторая для разбиения гипер1рафов. Также заслуживают внимания программы Scotch (http://www.labri.fr/perso/pelegrin/scotch) и JOSTLE (http://staffweb.cms.gre.ac.uk/ ~wc()6/jostle/). Примечания Основные эвристические методы для локального улучшения разбиения графа были пред- ставлены в [KL70] и [FM82]. Спектральные методы разбиения графов рассмотрены в [Chu97] и [PSL90]. Эмпирические результаты по эвристическим методам для разбиения графов изложены в [BG95] и [LR93]. Теорема о планарном разделителе и эффективный алгоритм поиска такого разделителя были представлены Липтоном и Тарьяном (см [LT79] и [LT80]). Опыт реализации алго- ритмов поиска разделителей планарных графов изложен в работах [ADGM04J и [HPS 05], Можно ожидать, что любое случайное разбиение множества вершин приведез к удалению половины ребер графа, т. к. вероятность того, что две вершины, имеющие общее ребро, окажутся в разных частях разбиения, равна 1/2. Гомане (Goemans) и Вильямсон (Williamson) в [GW95] представили аппроксимирующий алгоритм, основанный на методе полуопределенного программирования, выдающий решение задачи максимального разре- за с точностью до 0,878. Более точный анализ этого алгоритма был выполнен Карловым (Karloff), см. [Каг96]. Родственные задачи. Реберная и вершинная связности (см. раздел 15 Л). поток в сети (см. раздел 15.9). 16.7. Вершинная раскраска Вход. Граф G = (Г, Е). Задача. Раскрасить множество вершин Г, используя для этого минимальное количест- во цветов, таким образом, чтобы вершины i и j были разного цвета для всех ребер (ij) е Е (рис. 16.8).
558 Часть II. Каталог алгоритмических задач ВХОД выход Рис. 16.8. Вершинная раскраска Обсуждение. Задача вершинной раскраски возникает во многих приложениях кален- дарного планирования и кластеризации. Классическим приложением раскраски графа является распределение регистров при оптимизации программы в процессе компили- рования. Для каждой переменной в данном фрагменте программы существуют интер- валы времени, в течение которых ее значение не должно меняться, в частности, между ее инициализацией и следующим обращением к ней. Следовательно, нельзя помещать в один и тот же регистр две переменные с пересекающимися интервалами времени. Для оптимального распределения регистров создаем граф, в котором вершины соот- ветствуют переменным, и соединяем ребрами каждые две вершины, у которых пере- менные имеют пересекающиеся интервалы времени. Поскольку никакие две перемен- ные с вершинами одного цвета не конфликтуют между собой, им можно выделить один и тот же регистр. Никаких конфликтов не будет, если каждая вершина окрашена в свой цвет. Так как количество регистров процессора ограничено, то мы стремимся выполнить раскраск). используя как можно меньшее количество цветов. Минимальное количество цветов, достаточное для раскраски вершин графа, называется его хроматическим числом. На практике встречается несколько частных случаев задачи раскраски, представляю- щих интерес. Для их распознавания постарайтесь найти ответы на следующие вопросы. ♦ Можно ли раскрасить граф, используя только два цвета? Важным частным случа- ем является проверка графа на двудольность, означающую, что его можно раскра- сить. используя только два разных цвета. Двудольные графы естественно возникают в таких приложениях, как назначение заданий работникам. Быстрые и простые ал- горитмы существуют для задач паросочетания (см. раздел 15.6), если на них нало- жено такое ограничение, как двудольность графа. Проверка графа на двудольность не составляет труда. Окрашиваем первую вершину в синий цвет, а потом выполняем обход графа в глубину. При открытии каждой но- вой, еще не окрашенной вершины, окрашиваем ее в цвет, противоположный цвету смежной с ней вершины, т. к. окраска в тот же самый цвет вызвала бы конфликт. Граф не может быть двудольным, если он содержит ребро (х,у), обе вершины кото- рого окрашены в один цвет. В противном случае конечная раскраска будет двуцвет-
Глава 16. Сложные задачи на графах 559 ной, и раскрашивание займет время О(п + т). Реализацию этого алгоритма см. в разделе 5. Т.2. ♦ Является ти граф планарным? Все ли вершины графа имеют низкую степень? Зна- менитая теорема о четырех красках утверждает, что вершинную раскраску любого планарного графа можно выполнить, используя, самое большее, четыре разных цве- та. Для четырехцветной раскраски планарных графов существуют эффективные ал- горитмы, но выяснение, можно ли данный планарный граф раскрасить тремя цвета- ми, является NP-полной задачей. Для шестицветной раскраски планарного графа существует очень простой алго- ритм. Любой планарный граф содержит вершину, степень которой не больше пяти. Удаляем эту вершину и раскрашиваем граф рекурсивно. Удаленная вершина имеет не более пяти соседей, следовательно, ее всегда можно раскрасить одним из шести цветов, отличающимся от цвета соседних вершин. Метод работает, т. к. после уда- ления вершины из планарного графа получается планарный граф, и у него тоже должна быть вершина низкой степени, которую можно будет удалить. Эту же идею можно использовать для раскраски любого графа максимальной степени Л, задейст- вовав не больше А + 1 цветов, за время О(иА). ♦ Можно ли привести задачу к задаче реберной раскраски? Некоторые задачи вер- шинной раскраски можно сформулировать в виде задачи реберной раскраски, в ко- торой требуется раскрасить ребра графа таким образом, чтобы ни одна пара ребер с общей вершиной не была окрашена одинаковым цветом. Практическая выгода от такой переформулировки состоит в том, что для решения задачи реберной раскрас- ки существует эффективный алгоритм, который возвращает почти оптимальную раскраску. Алгоритмы реберной раскраски рассматриваются в разделе 16.8. Задача вычисления хроматического числа графа является NP-полной, поэтому, если требуется точное решение, нужно будет использовать метод перебора с возвратом, ко- торый может оказаться неожиданно эффективным при раскраске некоторых графов. Но задача поиска приблизительного решения, достаточно близкого к оптимальному, остается сложной, поэтому не следует ожидать никаких гарантий. Наилучшими эвристическими алгоритмами для вершинной раскраски являются ин- крементальные. Так же, как и в ранее упомянутом алгоритме для планарных графов, раскраска вершин выполняется последовательно, при этом цвета для раскраски теку- щей вершины выбираются в зависимости от уже использованных цветов для раскраски смежных вершин. Эти алгоритмы различаются способами выбора следующей вершины и выбора цвета для ее раскраски. Практика подсказывает, что вершины следует встав- лять в порядке невозрастания их степеней. Дело в том, что для вершин с высокой сте- пенью существует больше ограничений по выбору цвета, и вставлять их надо как мож- но раньше, чтобы не потребовался дополнительный цвет. Эвристический алгоритм Брелаза (Brelaz) (см. [Вге79]) динамически выбирает неокрашенную вершину, у кото- рой смежные вершины окрашены в наибольшее количество разных цветов, и окраши- вает ее неиспользованным цветом с наименьшим номером. Полученные инкрементальными методами результаты можно улучшить, используя об- мен цветов. Для этого в раскрашенном графе меняем местами два цвета, т. е„ напри- мер, красные вершины раскрашиваем синим цветом, а синие — красным, сохраняя при
560 Часть II. Каталог алгоритмических задач этом правильную вершинную раскраску. Теперь возьмем раскрашенный должным об- разом граф и удалим из него все вершины, кроме красных и синих. Мы можем пере- красть одну или несколько из получившихся компонент связности, сохраняя при этом правильную раскраску. После такого перекрашивания может оказаться, что какая-либо вершина, ранее смежная как с красными, так и с синими вершинами, является смежной только с синими вершинами, что позволит окрасить ее красным цветом. Обмен цветов позволяет получить более качественную раскраску за счет увеличения времени обработки и сложности реализации. Алгоритмы, основанные на методе ими- тации отжига и использующие обмен цветов для перехода из одного состояния в дру- гое, будут, скорее всего, еше более эффективными. Реализации. Для раскраски графов есть два полезных интернет-ресурса. Страница раскраски графов (http://web.cs.ualberta.ca/~joe/Coloring) предоставляет обширную библиографию и программное обеспечение для генерирования и решения сложных экземпляров задачи раскраски графов. Страница Майкла Трика (Michael Trick) http://mat.gsia.cmu.edu/COLOR/color.html содержит хороший обзор приложений для раскраски графов, аннотированную библиографию, а также коллекцию из более чем 70 экземпляров задачи раскраски графов, возникающих в таких приложениях, как рас- пределение регистров и проверка печатных плат. Оба веб-сайта также содержат реали- зацию на языке С алгоритма раскрашивания DSATUR. На втором соревновании по реализации алгоритмов D1MACS в октябре 1993 г. (см. [JT96J) оценивались программы решения тесно связанных задач поиска клик и раскраски вершин графов. Программы и данные можно загрузить по адресу ftp://dimacs.rutgers.edu. Исходные коды находятся в каталоге pub/challenge/graph, а тестовые данные— в каталоге pub/djs, включая простой "полуисчерпывающий жад- ный" подход, используемый в алгоритме раскраски графов XRLF (см. [JAMS91]). Программа GraphCol (http://code.google.eom/p/graphcol/). написанная на языке С, реа- лизует алгоритм раскраски графов, основанный на методах поиска с запретами и ими- тации отжига. Библиотека Boost Graph Library (см. [SLL02]; http://www.boost.org/libs/graph/doc) со- держит реализацию на языке C++ "жадного" инкрементального эвристического алго- ритма для раскраски вершин графов. Библиотека GOBLIN (http://www.math.uni- augsburg.de/~fremuth/goblin.html) содержит реализацию алгоритма метода ветвей и границ для вершинной раскраски. В книге [SDK83] представлены реализации на языке Pascal алгоритмов раскраски гра- фов, основанных на методе перебора с возвратом, а также нескольких эвристических методов, включая метод инкрементального упорядочения по принципу "наибольшего в начало" и "наименьшего в конец" и метод обмена цветов. Подробности см. в раз- деле 19.1.10. В книге [NW78] приводятся эффективные реализации на языке FORTRAN алгоритмов вычисления хроматических многочленов и раскраски графов, основанных на методе перебора с возвратом. Подробности см. в разделе 19.1.10. Библиотека Combinatorica содержит реализации (на языке пакета Mathematica) алго- ритмов для проверки двудольности графов, раскраски с помощью эвристических мето-
Глава 16. Сложные задачи на графах 561 дов, вычисления хроматических многочленов и вершинной раскраски методом перебо- ра с возвратом. Подробности см. в разделе 19.1.9. Примечания Отличным источником информации по эвристическим методам, включающим результаты экспериментов, является уже не новая книга [SDK83]. Классические эвристические мето- ды для вершинной раскраски приводятся в работах [Bre79], [MMI72] и [Tur88]; последние результаты можно найти в [GH06] и [HDD03]. В своей работе [Wil84J Вильф (Wilf) доказал, что алгоритм для проверки произвольного графа на то, что его хроматическое число равно к, работающий методом перебора с воз- вратом, выполняется за постоянное время, зависящее от к, но не зависящее от п. Впрочем, это не представляет большого интереса, поскольку в действительности очень мало графов являются А-раскрашиваемыми. Известно несколько эффективных (но, тем не менее, имеющих экспоненциальное время исполнения) алгоритмов для вершинной раскраски. Обзор таких алгоритмов представлен в [Woe03]. Работа [PasO3J содержит информацию о доказуемо хороших аппроксимирующих алго- ритмах вершинной раскраски. С одной стороны, сложность задачи получения аппрокси- мирующего решения с полиномиальным коэффициентом была доказана в работе [BGS95]. Но с другой стороны, существуют эвристические методы, гарантирующие нетривиальные результаты в зависимости от значений различных параметров. Одним из таких методов является алгоритм Вигдерсона (Wigderson) (см. [Wig83]), предоставляющий аппроксими- рующее решение, имеющее коэффициент ,;bl/lzl<,) 11, где/(G) является хроматическим числом графа G. Теорема Брука утверждает, что для хроматического числа справедливо /(G)<A(G)+ I, где A(G)— максимальная степень вершин графа G. Равенство имеет место только для циклов нечетной длины (с хроматическим числом 3) и полных графов. Самой знаменитой задачей в истории теории графов является задача четырехцветной рас- краски. Она была впервые поставлена в 1852 г. и окончательно решена в 1976 г. Аппелем (Appel) и Хакеном (Haken), причем для доказательства правильности решения был ис- пользован компьютер. Любой планарный граф можно раскрасить пятью цветами, приме- няя один из вариантов эвристического метода обмена цветов. Несмотря на существование решения задачи четырехцветной раскраски, задача проверки достаточности трех цветов для раскраски конкретного планарного графа является NP-полной. История задачи четы- рехцветной раскраски и доказательство изложены в книге [SK86J. Эффективный алгоритм для четырехцветной раскраски графа представлен в работе [RSST96], Родственные задачи. Независимое множество (см. раздел 16.2), реберная раскраска (см. раздел 16.8). 16.8. Реберная раскраска Вход. Граф G = (V, Е). Задача. Найти наименьший набор цветов, требуемый для раскраски ребер графа G та- ким образом, чтобы ни одна пара ребер, имеющих общую вершину, не была окрашена одним цветом (рис. 16.9). Обсуждение. Задача реберной раскраски графов возникает в приложениях календарно- го планирования, обычно, когда требуется минимизировать количество взаимно не
562 Часть II. Каталог алгоритмических задач конфликтующих маршрутов, необходимых для выполнения данного набора заданий. Рассмотрим для примера ситуацию, когда необходимо составить расписание несколь- ких деловых встреч продолжительностью в один час, в каждой из которых участвуют два сотрудника. Чтобы избежать конфликтов, все встречи можно было бы запланиро- вать на разное время, но разумнее назначить неконфликтующие встречи на одно и то же время. Для решения этой задачи создадим граф, в котором вершины представляют людей, а ребра соединяют тех. которые должны встретиться. Реберная раскраска этого графа определяет искомое расписание. Разные цвета представляют разные временные периоды, при этом все встречи одного цвета происходят одновременно. ВХОД ВЫХОД Рис, 16.9. Реберная раскраска Национальная футбольная лига решает такую задачу реберной раскраски каждый сезон, чтобы составить расписание игр входящих в нее команд. Противники каждой команды определяются по результатам игр предыдущего сезона. Составление расписа- ния игр является задачей реберной раскраски, усложненной дополнительными ограни- чивающими условиями, такими как необходимость в ответных матчах и традиционное требование встречи сильных соперников в понедельник вечером. Минимальное количество цветов, требуемое для реберной раскраски графа, называется реберно-хроматическим числом (edge-chromatic number) или хроматическим индексом (chromatic index). Обратите внимание, что ребра цикла четной длины можно раскрасить двумя цветами, в то время как реберно-хроматическое число циклов нечетной длины равно трем. С задачей реберной раскраски связана интересная теорема. Согласно теореме Визинга. для любого графа с максимальной степенью вершин Л можно выполнить реберную раскраску, используя, самое большее, А + 1 цвет. Чтобы понять это утверждение, обра- тите внимание на то, что любая реберная раскраска должна содержать, как минимум. А цветов, т. к. все ребра, инцидентные одной и той же вершине, должны быть раскра- шены разными цветами. Доказательство теоремы Визинга является конструктивным, т. е. его можно предста- вить в виде алгоритма поиска реберной раскраски из А + 1 цветов с временем исполне-
Глава 16. Сложные задачи на графах 563 ния О(ишА). Так как задача выяснения, можно ли использовать для раскраски на один цвет меньше, является NP-полной, то ее решение вряд ли стоит требуемых усилий. Любую задачу реберной раскраски графа G можно преобразовать в задачу вершинной раскраски реберного графа £(G), имеющего вершину графа L(G) для каждого ребра в графе G, и ребро графа /.(G) тогда и только тогда, когда два ребра графа G инцидентны одной и той же вершине. Реберный граф можно создать за линейное время и можно раскрасить, используя любую программу вершинной раскраски. Реализации. Реализация на языке C++ теоремы Визинга была осуществлена Яном Донгом (Yan Dong) в качестве курсового проекта, когда он учился у меня в универси- тете Стоуни Брук. Загрузить эту программу можно с веб-страницы http://www. cs.sunysb.edu/~algorith. Библиотека GOBLIN (http://www.math.uni-augsburg.de/~fremuth/goblin.html) содер- жит алгоритм метода ветвей и границ для реберной раскраски. Список программ для вершинной раскраски, которые можно применить к реберному графу, построенному на основе вашего целевого графа, вы найдете в разделе 16.7. Биб- лиотека Combinatorica (см. [PS03]) содержит реализации (на языке пакета Mathematica) алгоритмов реберной раскраски, работающих по тому же принципу, т. е. преобразую- щих целевой граф в реберный и применяющих к нему процедуры вершинной раскрас- ки. Дополнительную информацию по Combinatorica см. в разделе 19.1 9. Примечания Обзоры теоретических исследований задачи реберной раскраски представлены в [FW77] и [GT94J. Доказательство, что реберную раскраску любого графа можно выполнить, ис- пользуя для этого, самое большее, А + 1 цвет, было независимо представлено Визингом (Vizing) (см. [Viz64]) и Гуптой (Gupta) (см. [Gup66]). Простое конструктивное доказатель- ство этого результата приводят Миера (Misra) и Приз (Gries) в своей работе [MG92] Не- смотря на свою специфичность, задача вычисления реберно-хроматического числа явля- ется NP-полной (см. [Hoi81]). Реберную раскраску двудольных графов можно выполнить за полиномиальное время (см. [Sch98]). Представляя реберные графы в своей работе [Whi32], Уитни (Whitney) показал, что за ис- ключением графов К, и К', 3, любые два связных графа, реберные графы которых изо- морфны, являются изоморфными. Интересным упражнением будет поиск доказательства того, что реберный граф эйлерова графа является как эйлеровым, так и гамильтоновым, в то время как реберный граф гамильтонова графа всегда будет только гамильтоновым. Родственные задачи. Вершинная раскраска (см. раздел 16.7), календарное планирова- ние (см. раздел 14.9). 16.9. Изоморфизм графов Вход. Графы G и Н. Задача. Найти отображение f (или все возможные отображения) множества вершин графа G в множество вершин графа //такое, что граф G и граф Н идентичны, т. е. (а,у) является ребром графа G тогда и только тогда, когда (Дх), Ду)) является ребром гра- фа И (рис. 16.10).
564 Часть II. Каталог алгоритмических задач ВХОД ВЫХОД Рис. 16.10. Изоморфизм графов Обсуждение. Задача выявления изоморфизма графов заключается в проверке графов на идентичность. Допустим, что требуется выполнить определенные операции над каждым графом из некоторой коллекции. Если мы сможем выяснить, что какие-то гра- фы идентичны друг другу, мы будем игнорировать копии, чтобы не делать двойную ра- боту. Некоторые задачи распознавания образов можно с легкостью сформулировать в виде задачи выявления изоморфизма графов или подграфов. Например, структура химиче- ских соединений естественным образом описывается помеченными графами, в кото- рых каждая вершина представляет отдельный атом. Поиск в базе данных всех молекул, содержащих определенную функциональную группу, является задачей выявления изо- морфизма подграфов. Уточним, что мы понимаем под идентичностью графов. Два помеченных графа G = (Ук, Ек) и Н = ( ЕЛ. Е),) являются идентичными, если (х, у) е тогда и только тогда, когда (х,у) е £),. Выявление изоморфизма заключается в поиске отображения вершин графа G на множество вершин графа Н, при котором графы идентичны. Это отображе- ние и называется изоморфизмом. Другим важным применением изоморфизма графов является выявление симметрии. Отображение графа на самого себя называется автоморфизмом, а коллекция автомор- физмов (группа автоморфизмов) содержит много информации о симметричности гра- фа. Например, полный граф Кп содержит и! автоморфизмов (годится любое отображе- ние), в то время как произвольный граф, скорее всего, будет иметь малое количество автоморфизмов, возможно, только один, поскольку граф G идентичен себе самому. На практике возникает несколько видов задачи изоморфизма графов. Чтобы их распо- знать, постарайтесь ответить на следующие вопросы. ♦ Содержится ли граф G в графе Н? Вместо проверки на идентичность мы нередко должны выяснить, является ли данный граф G под графом графа Н. Такие задачи, как задача о клике, независимом множестве и гамильтоновом цикле, являются важ- ными частными случаями задачи выявления изоморфизма подграфов.
Глава 16. Сложные задачи на графах 565 Выражение "граф G содержится в графе Н" имеет два разных значения в теории графов. В задаче выявления изоморфизма подграфа требуется выяснить, содержит ли граф Н подмножество ребер и вершин, которое является изоморфным графу G. А в задаче выявления изоморфизма порожденного подграфа требуется выяснить, содержит ли граф Н подмножество ребер и вершин, после удаления которого оста- нется подграф, изоморфный графу G. В задаче выявления изоморфизма порожден- ного подграфа требуется, чтобы все ребра графа G присутствовали в графе Н. и что- бы в графе Н не было никаких "не-ребер" графа G. Клика является экземпляром обоих вариантов задачи выявления изоморфизма подграфов, в то время как гамиль- тонов цикл является примером лишь простого изоморфизма подграфов. Следует учитывать это различие при работе над вашим приложением. Задачи выяв- ления изоморфизма подграфов обычно сложнее, чем задачи выявления изоморфиз- ма графов, а задачи выявления изоморфизма порожденных подграфов еще сложнее. Единственным разумным подходом к решению таких задач является метод перебо- ра с возвратом. ♦ Помечены графы или нет? Во многих приложениях вершины и/или ребра графов помечаются каким-либо атрибутом, который нужно учитывать при выявлении изо- морфизма. Например, при сравнении двудольных графов, содержащих вершины двух типов, скажем, "рабочий" и "задание", любой изоморфизм, уравнивающий за- дание с рабочим, будет лишен смысла. Метки и связанные с ними ограничивающие условия можно включить в любой ал- горитм метода перебора с возвратом. Кроме этого, такие ограничивающие условия значительно ускоряют поиск, создавая намного больше возможностей для разреже- ния пространства поиска при каждом случае несовпадения меток двух вершин. ♦ Являются ли деревьями графы, которые проверяются на изоморфизм? Для про- верки на изоморфизм некоторых частных случаев графов, таких как деревья и пла- нарные графы, существуют более быстрые алгоритмы. Возможно, самым важным случаем задачи выявления изоморфизма является выявление изоморфизма деревьев. Эта задача возникает при сопоставлении языковых структур и синтаксическом ана- лизе. Для описания структуры текста часто используется дерево синтаксического разбора. Два таких дерева будут изоморфными, если представляемые ими тексты имеют одинаковую структуру. Эффективные алгоритмы выявления изоморфизма деревьев начинают работу с ли- стьев обоих деревьев и продвигаются к центру. Каждой вершине дерева присваива- ется метка, представляющая набор вершин во втором дереве, который, возможно, изоморфен поддереву с корнем в этой вершине, с учетом ограничений, накладывае- мых метками и степенями вершин. Например, первоначально все листья дерева Т\ потенциально эквивалентны всем листьям дерева Т2. Теперь, продвигаясь внутрь, мы можем разбить смежные с листьями вершины дерева Т\ на классы, в зависимо- сти от количества смежных с ними листьев и других узлов. Отслеживая метки под- деревьев, мы можем убедиться, что имеем одинаковое распределение помеченных поддеревьев для Т\ и Г2. Любое несовпадение означает, что Т\фТт. в то время как по завершении процесса все вершины оказываются разбиты на классы эквивалентно- сти. определяющие все изоморфизмы.
566 Часть II. Каталог алгоритмических задач ♦ Сколько имеется графов? Во многих приложениях обработки данных требуется выполнить поиск всех экземпляров графа с определенной структурой в большой ба- зе данных. Уже упоминавшаяся задача отображения химических структур относит- ся к этому типу задач. Такие базы данных обычно содержат большое количество сравнительно небольших графов. В связи с этим возникает необходимость индекси- рования базы данных графов по небольшим подструктурам (от пяти до десяти вер- шин каждая) и выполнения дорогостоящих проверок графов на изоморфность толь- ко с теми, которые содержат такие же подструктуры, что и граф запроса. Для решения задачи выявления изоморфизма графов не предложено ни одного алго- ритма с полиномиальным временем исполнения и в то же время не выяснено, является ли эта задача NP-полной. Вместе с задачей разложения на множители целых чисел (см. раздел 13.8), это одна из нескольких важных алгоритмических задач, вычислительная сложность которых до сих пор неизвестна, даже приблизительно. Принято считать, что если P^NP. то задача выявления изоморфизма занимает промежуточное положение между Р- и NP-полной. Но хотя не известно алгоритмов для решения наихудших случаев задачи за полиноми- альное время, задача выявления изоморфизма на практике обычно не очень сложна. Основной алгоритм выполняет перебор с возвратом всех /?! возможных замен меток вершин графа h на метки вершин графа g, а потом проверяет графы на идентичность. Конечно же. пространство поиска можно разредить, убрав все перестановки с одинако- вым префиксом при первом же несовпадении ребер, обе вершины которых имеют этот префикс. Но настоящим ключом к эффективной проверке на изоморфизм будет предварительное разбиение множества вершин на "классы эквивалентности" таким образом, чтобы было невозможно перепутать две вершины из разных классов. Все вершины в каждом классе эквивалентности должны иметь одинаковое значение какого-либо инварианта, не зави- сящего от меток. Возможно использование одного из следующих инвариантов: ♦ степень вершины. Самый простой способ разбиения множества вершин на части основан на их степени, т. е. на количестве ребер, инцидентных каждой вершине. Две вершины с разными степенями не могут быть одинаковыми. Такое простое раз- биение может быть очень полезным, но не в случае с регулярными графами (т е. графами, у которых все вершины имеют одинаковую степень); ♦ матрица кратчайших путей. Для каждой вершины v матрица кратчайших путей между всеми парами (см. раздел 15.4) определяет мультимножество из п— 1 рас- стояний, представляющих расстояния между вершиной v и каждой из остальных вершин. Любые две одинаковые вершины должны определять одинаковые муль- тимножества расстояний, поэтому мы можем разбить вершины на классы эквива- лентности, определяющие одинаковые мультимножества расстояний; ♦ количество путей длиной к. Возведение в к-ю степень матрицы смежности графа G дает матрицу, в которой ячейка Gk[i, /'] содержит количество путей от вершины i к вершине j. Для каждой вершины и для каждого значения /тэта матрица определяет количество путей, которое можно использовать для разбиения вершин на классы, как и расстояния в предыдущем случае. Можно перепробовать все 1 <к<п или да-
Глава 16 Сложные задачи на графах 567 же лежащие за пределами этого диапазона, и использовать любое отклонение как критерий для разбиения. Используя эти инварианты, часто удается разбить граф на много небольших классов эквивалентности. После такого разбиения вершин не составит труда довести работу до конца, используя метод перебора с возвратом. Каждой вершине в качестве метки при- сваивается имя ее класса эквивалентности, и задача решается как задача паросочетания в помеченном графе. Выявить изоморфизм в высокосимметричных графах сложнее, чем в случайных, по причине снижения эффективности таких эвристических методов разбиения вершин на множество классов эквивалентности. Реализации. Наиболее известной программой для проверки графов на изоморфизм является программа nauty, представляющая собой набор очень эффективных процедур на языке С для поиска группы автоморфизма графа с вершинной раскраской. Про- грамма также может выдавать канонически помеченный граф, изоморфный данному, чтобы содействовать проверке на изоморфизм. На основе nauty была разработана пер- вая программа для генерирования всех 11-вершинных графов, не имеющих изоморф- ных графов, которая также может тестировать большинство графов с меньше чем 100 вершинами быстрее, чем за одну секунду. Программа nauty перенесена на разные операционные системы и компиляторы языка С. Загрузить ее можно с веб-сайта http://cs.anu.edu.au/~bdm/nauty/. Теоретические основы программы nauty изложены в работе [МсК81]. Библиотека средств для сравнения графов VFLib содержит реализации нескольких раз- ных алгоритмов для проверки на изоморфизм как графов, так и подграфов. Библиотека была очень тщательно протестирована (см. [FSVO1]). Загрузить ее можно с веб-сайта http://amalfi.dis.unina.it/graph. Пакет программного обеспечения GraphGrep (см. [GS02]) предназначен для поиска за- прашиваемого графа в большой базе данных графов. Загрузить пакет можно с веб- сайта http://www.cs.nyu.edu/shasha/papers/graphgrep. Реализации алгоритмов на языке C++ для проверки изоморфизма графов и подграфов как для деревьев, так и для графов из книги [Val02] предоставляются для общего поль- зования. Эти реализации основаны на кодах библиотеки LEDA (см. раздел /9././); за- грузить их можно с веб-сайта http://www.lsi.upc.edu/~valiente/algorithm. В книге [KS99], в дополнение к более общим операциям теории графов, представлены программы для выявления изоморфизма графов. Эти программы, написанные на язы- ке С. можно загрузить с веб-сайта http://www.math.mtu.edu/~kreher/cages/Src.htnd. Примечания Выявление изоморфизма графов — важная задача теории сложности. Среди монографий, посвященных выявлению изоморфизма, можно выделить работу [Hof82] и книгу [KST93], Книга [Val02] посвящена, преимущественно, алгоритмам выявления изоморфизма деревь- ев и подграфов. Подход к проверке на изоморфизм, применяемый в книге [KS99], основан на теории групп. Обзор систем и алгоритмов анализа данных, представленных в виде гра- фов, приводится в книге [СН06]. Результаты сравнения производительности разных алго- ритмов выявления изоморфизма графов и подграфов содержатся в работе [FSV01], Существуют алгоритмы с полиномиальным временем исполнения для выявления поли- морфизма планарных графов (см. [HW74]) и для графов, у которых максимальная степень
568 Часть II. Каталог алгоритмических задач вершин ограничена константой (см. [Luk80]). Эвристический метод на основе кратчай- шего пути между всеми парами был представлен в работе [SD76], хотя существуют не- изоморфные графы, имеющие точно такое же множество расстояний (см. [ВН90]). Алго- ритм с линейным временем исполнения для выявления изоморфизмов в помеченных и не- помеченных деревьях представлен в книге [AHU74]. Задача называется изоморфно-полной, если она доказуемо такая же сложная, как и задача изоморфизма. Задача проверки на изоморфизм двудольных графов является изоморфно- полной, т. к. любой граф можно преобразовать в двудольный, заменив каждое из его ре- бер двумя ребрами, соединенными с новой вершиной. Очевидно, что исходные графы яв- ляются изоморфными тогда и только тогда, когда изоморфны графы, полученные в ре- зультате такого преобразования. Родственные задачи. Задача кратчайшего пути (см. раздел 15 4), поиск подстрок (см.3<)с<7 18.3). 16.10. Дерево Штейнера Вход. Граф) G = (Е, Е), подмножество вершин Те Г Задача. Найти наименьшее дерево, соединяющее все вершины Г (рис. I6.l I). ВХОД ВЫХОД Рис. 16.11. Дерево Ш гейнера Обсуждение. Задача построения деревьев Штейнера часто возникает при разработке коммуникационных сетей, т. к. минимальное дерево Штейнера описывает, как соеди- нить данное множество станций, используя кабель наименьшей протяженности. Ана- логичные задачи возникают при разработке водопроводных или вентиляционных и отопительных сетей и при разработке сверхбольших интегральных микросхем (СБИС). В области разработки СБИС типичной задачей построения дерева Штейнера является соединение набора площадок, например, с проводом заземления, при соблюдении ограничивающих условий, таких как стоимость материала, время распространения сигнала или емкостное сопротивление. Задача построения дерева Штейнера отличается от задачи построения минимального остовного дерева (см. раздел 15.3) тем, что в ней разрешается создавать или выбирать
Глава 16. Сложные задачи на графах 569 промежуточные соединительные узлы, чтобы уменьшить стоимость дерева. Прежде чем приступать к построению дерева Штейнера, ответьте на несколько вопросов. ♦ Сколько точек требуется соединить? Дерево Штейнера для двух вершин— это просто кратчайший путь между ними (см. раздел 15.4). А дерево Штейнера для всех вершин, когда 5= I', просто определяет минимальное остовное дерево графа. Не- смотря на наличие таких простых частных случаев, общая задача построения дерева Штейнера является NP-полной и в широком диапазоне ограничивающих условий. ♦ Являются ли входные данные набором точек win графом расстояний? Геометриче- ские версии задачи построения дерева Штейнера принимают в качестве входа набор точек, как правило, расположенных на плоскости, и заключаются в поиске наи- меньшего дерева, соединяющего эти точки. Однако возникает затруднение сле- дующего вида. Набор допустимых промежуточных точек не задается как часть вхо- да, а должен быть получен из исходного набора точек. Эти промежуточные точки должны удовлетворять определенным геометрическим условиям, благодаря кото- рым набор точек-кандидатов становится конечным. Например, в минимальном де- реве Штейнера степень каждой точки Штейнера равна 3, а угол между любыми двумя ребрами такой точки должен быть ровно 120°. ♦ Можно ли воспользоваться какими-нибудь ограничениями, наложенными на ребра? Многие задачи монтажа электропроводки соответствуют геометрическим версиям задачи, в которых все ребра должны располагаться либо горизонтально, либо вер- тикально. Задача такого типа называется линейной задачей Штейнера. В задаче по- строения линейного дерева Штейнера на углы между ребрами и степени вершин на- кладываются иные ограничения, чем в задаче построения евклидова дерева. В част- ности. все углы должны быть кратны 90°, а максимальная степень любой вершины не должна превышать 4. ♦ Действительно ли нужно оптимальное дерево? В некоторых приложениях задачи построения дерева Штейнера (например, при проектировании печатных плат или коммуникационных сетей) большой объем вычислений для построения оптималь- ного дерева Штейнера вполне оправдан. В таких случаях применяются исчерпы- вающие методы поиска, такие как перебор с возвратом или метод ветвей и границ. Существует много возможностей для прореживания пространства поиска с исполь- зованием различных геометрических ограничивающих условий. Тем не менее, задача построения дерева Штейнера остается сложной. Поэтому, прежде чем пытаться разрабатывать свою собственную реализацию, попробуйте применить уже существующие, описанные далее в подразделе "Реализации". ♦ Как восстановить точки Штейнера о которых не было известно вначале? В при- ложениях из области классификации и эволюции возникает очень специфичный тип дерева Штейнера. Филогенетическое дерево иллюстрирует относительную схожесть разных объектов или биологических видов. Каждый объект обычно представляется листом дерева, а промежуточные вершины соответствуют точкам разветвления ме- жду классами объектов. Например, в эволюционном дереве видов животного мира листья могут представлять человека, собаку, змею, а внутренние узлы соответству- ют таксонам (животные, млекопитающие, пресмыкающиеся). Дерево с корнем
570 Часть II. Каталог алгоритмических задач животные, в котором человек и собака находятся в таксоне мяекопитающие, озна- чает, что человек имеет более близкое родство с собаками, чем со змеями. Было разработано много разных алгоритмов для создания филогенетических де- ревьев, которые отличаются друг от друга моделируемыми данными и критериями оптимизации. Разные комбинации реконструирующего алгоритма и метрики выда- ют разные ответы, поэтому определение "правильного" метода в каждом конкрет- ном случае весьма субъективно. Разумным решением будет применение одного из упомянутых далее стандартных пакетов реализаций и сравнительный анализ ре- зультатов обработки данных всеми входящими в него реализациями. К счастью, существует эффективный эвристический алгоритм для построения деревьев Штейнера, хорошо работающий на всех версиях задачи. Создаем граф, моделирующий вход задачи, и устанавливаем вес ребра (/,j) равным расстоянию от точки I до точки j. Находим минимальное остовное дерево для этого графа. Оно гарантированно является хорошей аппроксимацией как для евклидовых деревьев, так и для линейных деревьев Штейнера. Самым худшим случаем для получения приблизительного минимального остовного дерева для евклидова дерева Штейнера является входной экземпляр из трех точек, об- разующих равносторонний треугольник. В этом случае минимальное остовное дерево содержит две стороны треугольника (с общей длиной ребер равной 2), в то время как минимальное дерево Штейнера соединяет все три точки с использованием еще одной, внутренней, и общая длина его ребер равна д/з Это соотношение у/з /2~0,866 все- гда достижимо, и на практике минимальное остовное дерево обычно имеет размер, на несколько процентов отличающийся от размера оптимального дерева Штейнера. От- ношение размера линейного дерева Штейнера к размеру минимального остовного де- рева всегда равно 2/3 = 0,667. Такое минимальное остовное дерево всегда можно улучшить, вставив в него точку Штейнера в любом месте, где угол между инцидентными какой-либо вершине ребрами минимального остовного дерева меньше 120°. Вставив эту точку и откорректировав ребра, можно приблизить решение к оптимальному еще на несколько процентов. По- добная оптимизация возможна и для линейных остовных деревьев. Заметим, что нас интересует только поддерево, соединяющее листья. Если к входу за- дачи добавить вершины, не являющиеся конечными, то, возможно, потребуется откор- ректировать минимальное остовное дерево. В таком случае нужно будет оставить только те ребра, которые лежат на (уникальном) пути между какой-либо парой конеч- ных узлов. Полный набор таких ребер можно найти за время О(п), выполнив обход в ширину всего дерева, начиная с любого листа. Альтернативный эвристический алгоритм для графов основан на методе кратчайших путей. Начинаем с дерева, состоящего из кратчайшего пути между двумя конечными узлами. Для каждого оставшегося конечного узла t выбираем кратчайший путь к вер- шине г внутри дерева и добавляем этот путь к дереву. Качество и временная сложность этого эвристического алгоритма зависят от порядка вставки конечных узлов и от спо- соба вычисления кратчайших путей, но вероятность получить простое и эффективное решение достаточно высока.
Глава 16. Сложные задачи на графах 571 Реализации. Варм (Warme), Винтер (Winter) и Закариасен (Zachariasen) разработали пакет GeoSteiner, содержащий программы построения как евклидова, так и линейного дерева Штейнера на плоскости (см. [WWZ00]). Пакет можно также использовать для решения родственной задачи построения минимального остовного дерева в гипергра- фах. Утверждается, что с его помощью были получены оптимальные решения для эк- земпляров, состоящих из I0 ООО точек. Это, пожалуй, самый лучший пакет для реше- ния геометрических экземпляров задачи построения дерева Штейнера. Загрузить пакет можно с веб-сайта http://www.diku.dk/hjemmesider/ansatte/martinz/geosteiner/. Пакет FLUTE предназначен для быстрого построения линейных деревьев Штейнера. Программа предоставляет управляемый пользователем параметр для контроля соотно- шения между качеством решения и временем исполнения. Библиотека GOBLIN (http://www.math.uni-augsburg.de/~fremuth/goblin.html) содер- жит реализации как эвристических, так и поисковых алгоритмов для построения де- ревьев Штейнера в графах. Пакеты PHYLIP (http://evolution.genetics.washington.edu/phylip.html) и PAUP (http:// paup.csit.fsu.edu/) широко применяются для построения филогенетических деревьев. Оба пакета содержат реализации свыше 20 алгоритмов для создания филогенетических деревьев. Хотя многие из них предназначены для работы с моделями молекулярных последовательностей, некоторые методы принимают в качестве входа произвольные матрицы расстояний. Примечания В числе последних монографий, посвященных деревьям Штейнера, можно назвать [HRW92] и [PS02]. Книга [DSROO] содержит коллекцию сравнительно свежих обзоров по всем аспектам построения деревьев Штейнера. Более старый обзор задачи можно найти в [Kuh75J. Результаты исследований эвристических методов работы с деревьями Штейнера изложены в [SFG82] и [Vos92], Задача построения евклидовых деревьев Штейнера была впервые поставлена Ферма, ко- торый изучал вопрос поиска на плоскости точки р такой, что сумма расстояний до трех данных точек минимальна. Эта задача была решена Торричелли до 1640 г. По-видимому, над общей задачей для п точек работал также Штейнер, и поэтому ему по ошибке припи- сывается авторство решения данной задачи. Подробное и интересное изложение исгорни задачи представлено в работе [HRW92]. Гилберт (Gilbert) и Поллак (Pollak) (см. [GP68]) были первыми, кто выдвинул предполо- жение, что отношение размера минимального дерева Штейнера к размеру минимального остовного дерева всегда > %/з / 2 ® 0,866 . После двадцати лет активных исследований, от- ношение Гильберта-Поллака было, наконец, доказано Ду (Du) и Хвангом (Hwang); см [DH92], Евклидово минимальное остовное дерево для п точек можно создать за время O(/7lgw) (см. [PS85]). Аппроксимирующая схема с полиномиальным временем исполнения для деревьев Штей- нера в A-мерном евклидовом пространстве была представлена в работе [Аго98] А в работе [RZ05] приведен метод получения аппроксимирующего решения задачи построения де- ревьев Штейнера для графов, имеющего коэффициент 1,55. Доказательство сложности задачи построения дерева Штейнера для графов (см. [Каг72]1 содержится в [Eve79a]. Описания точных алгоритмов построения деревьев Штейнера для графов можно найти в [Law76]. Сложность задачи построения дерева Штейнера для
572 Часть II. Каталог алгоритмических задач евклидовых и манхэттеновских метрик была доказана в работах [GGJ77] и [GJ77]. Неиз- вестно, является ли задача построения евклидова дерева Штейнера NP-полной, в силу проблем с представлением расстояний. Можно провести аналогию между минимальными деревьями Штейнера и структурами с минимальной энергией в некоторых физических системах. Предположение, что такие системы (например, мыльные пленки на проволочных рамках) "решают" задачу построе- ния дерева Штейнера, обсуждается в [Mie58]. Родственные задачи. Минимальное остовное дерево (см. раздел 15.3), поиск кратчай- шего пути (см. раздел 15 4). 16.11. Разрывающее множество ребер или вершин Вход. Ориентированный граф G = (К, Е). Задача. Найти наименьший набор ребер Е’ или вершин V, удаление которого сделает граф ациклическим (рис. 16.12). ВХОД ВЫХОД Рис. 16.12. Разрывающее множество ребер Обсуждение. Необходимость в поиске разрывающего множества возникает из-за того, что многие задачи легче поддаются решению на бесконтурных орграфах, чем на ори- ентированных графах общего вида. Рассмотрим, например, задачу планирования рас- писания работ с ограничивающими условиями очередности, т. е. задача Л должна вы- полняться перед задачей В. Когда все ограничивающие условия не противоречат друг другу, то получается бесконтурный орграф, и для соблюдения условий вершины можно упорядочить посредством топологической сортировки (см. раздел 15.2). Но как соста- вить расписание, когда существуют циклические ограничивающие условия, например, задание А нужно выполнить перед заданием В, которое нужно выполнить перед зада- нием я , которое нужно выполнить перед заданием А? Найдя разрывающее множество, мы определяем наименьшее количество ограничи- вающих условий, которые нужно отбросить, чтобы получить допустимое расписание.
Глава 16. Сложные задачи на графах 573 В задаче разрывающего множества ребер (или дуг) мы отбрасываем отдельные огра- ничивающие условия очередности. А в задаче разрывающего множества вершин мы отбрасываем целые задания вместе со связанными с ними ограничениями. Аналогичным образом устраняется состояние гонок в электронных схемах. Задача раз- рывающего множества также называется задачей поиска максимального аг1иклического подграфа. Еще одно приложение связано с определением рейтинга спортсменов. Допустим, нам требуется определить рейтинг участников шахматных или теннисных соревнований. Для этого мы можем создать ориентированный граф, в котором ребро идет от вершины х к вершине у, если игрок х победил игрока у. По идее, игрок более высокого класса должен победить игрока низшего класса, хотя противоположный (неожиданный) ре- зультат наблюдается довольно часто. Естественным определением рейтинга будет то- пологическое упорядочение, полученное после удаления из графа минимального раз- рывающего множества ребер, представляющих неожиданные результаты матча. Решая задачу разрывающего множества, ответьте на следующие вопросы. ♦ Нужно ли отбросить какие-либо ограничивающие условия? Если граф уже является бесконтурным орграфом, что можно выяснить с помощью топологической сорти- ровки, то никакие изменения не требуются. Один из способов поиска разрывающего множества состоит в модифицировании алгоритма топологической сортировки та- ким образом, чтобы при обнаружении конфликта удалялось проблемное ребро или вершина. Но это разрывающее множество может оказаться намного больше требуе- мого. т. к. на ориентированных графах задачи поиска разрывающего множества ре- бер и разрывающего множества вершин являются NP-полными. ♦ Как найти подходящее разрывающее множество ребер? Эффективный эвристиче- ский алгоритм с линейным временем исполнения создает вершинное упорядочение, а потом удаляет каждую дугу, идущую в неправильном направлении. При любом упорядочении вершин направление, по крайней мере, половины дуг должно быть одинаковым (слева направо или справа налево), поэтому в качестве разрывающего множества следует взять то, которое меньше. Как правильно выбрать начальную вершину? Вполне естественно отсортировать вершины по их реберной разбалансированности, т. е. по разности между степенью захода и степенью исхода. При другом подходе в качестве начальной берется слу- чайная вершина v. Любая вершина х, определяемая входящим ребром (х. v). будет помещена слева от вершины v. Аналогичным образом, любая вершина х, опреде- ляемая исходящим ребром (у.х), будет помещена справа от вершины v. Теперь можно рекурсивно выполнить эту процедуру на левом и правом подмножествах, чтобы завершить упорядочение вершин. ♦ Как найти хорошее подходящее множество вершин? Только что описанная эври- стическая процедура возвращает упорядоченный набор вершин с некоторым коли- чеством обратных ребер. Пам нужно найти небольшое множество вершин, покры- вающее эти обратные ребра. То есть мы имеем задачу о вершинном покрытии, эв- ристические методы решения которой рассматриваются в разделе 16.3. ♦ Как поступить, когда нужно разорвать все циклы в неориентированном графе? Задача поиска разрывающих множеств в неориентированных графах существенно
574 Часть II. Каталог алгоритмических задач отличается от задачи их поиска в орграфах. Деревья являются ациклическими не- ориентированными графами, и каждое дерево из п вершин содержит ровно п-1 ребро. Таким образом, мощность наименьшего разрывающего множества ребер лю- бого неориентированного графа G будет |£| - (и - с), где с — количество компонент связности графа G. Обратные ребра, обнаруженные при обходе в глубину графа G, можно рассматривать как минимальное разрывающее множество ребер. Задача поиска разрывающего множества вершин для неориентированных графов является NP-полной. Эвристическая процедура для ее решения использует обход в глубину для поиска самого короткого цикла графа. Все вершины найденного цикла удаляются из графа, и в оставшемся графе вновь выполняется поиск цикла мини- мальной длины, который снова удаляется Эта процедура поиска и удаления циклов повторяется до тех пор, пока граф не станет ациклическим. Оптимальное разры- вающее множество вершин должно содержать хотя бы одну вершину, принадлежа- щую каждому из этих циклов (у которых нет общих вершин), и поэтому качество полученного приблизительного решения определяется средней длиной удаленных циклов. Иногда имеет смысл улучшить эвристические решения, используя рандомизацию или метод имитации отжига. Для перехода из одного состояния в другое можно изменять порядок вершин, обменивая пары вершин местами или вставляя/удаляя вершины мно- жества, являющегося кандидатом на роль разрывающего. Реализации. Алгоритм 815 из коллекции алгоритмов АСМ (см. раздел 19.1.6) пред- ставляет собой эвристическую процедуру GRASP для поиска разрывающего множест- ва как вершин, так и ребер (см. [FPR011). Эти коды на языке FORTRAN можно также загрузить с веб-сайта http://www.research.att.com/~nigcr/src. Библиотека GOBLIN (http://www.math.uni-augsburg.de/~fremuth/goblin.html) содер- жит реализацию аппроксимирующего алгоритм поиска минимального разрывающего множества дуг. Программа econ order из базы графов Standford GraphBase (см. раздел 19.1.8) выполня- ет перестановку рядов и столбцов матрицы таким образом, чтобы минимизировать суммы чисел под главной диагональю. Использование матрицы смежности в качестве входа и удаление всех ребер под главной диагональю позволяет получить ацикличе- ский граф. Примечания Обзор решений задачи разрывающего множества представлен в [FPR99]. Обсуждение до- казательства сложности задачи поиска минимального разрывающего множества (см. [Каг72]) можно найти в [AHU74] и [Eve79a]. Как задача разрывающего множества вершин, так и задача разрывающего множества ребер остается сложной даже в том слу- чае, когда степень захода и исхода всех вершин не больше двух (см. [GJ79]). В работе [BBF99] представлен аппроксимирующий алгоритм, выдающий решение для за- дачи разрывающего множества вершин, имеющее коэффициент 2. Разрывающее множе- ство ребер в ориентированных графах можно аппроксимировать с коэффициентом O(logHloglogn) (см. [ENSS98]). В работе [CFR06] рассматривается эвристическая про-
Гпава 16 Сложные задачи на графах 575 цедура определения рейтинга участников соревнований. Эксперименты с эвристическими методами описаны в журнале [Кое05]. Интересное применение задачи разрывающего множества дуг в экономике представлено в книге [Knu94] Для каждой пары .1, Д секторов экономики дается информация об объеме потока финансов из сектора 4 в сектор В. Требуется упорядочить секторы таким образом, чтобы выяснить, какие из них являются преимущественно поставщиками для других сек- торов, а какие поставляют товар, в основном, потребителям. Родственные задачи. Уменьшение ширины ленты матрицы (см. раздел 13.2). тополо- гическая сортировка (см.раздел 15.2), календарное планирование (см. раздел 14.9).
ГЛАВА 17 Вычислительная геометрия Вычислительная геометрия — это область дискретной математики, в которой изучают- ся алгоритмы решения геометрических задач. Ее возникновение совпало с возникнове- нием таких прикладных областей, как компьютерная графика и системы автоматизиро- ванного проектирования и производства, а также с более широким применением ком- пьютеров в различных научных отраслях. Все эти факторы стимулируют развитие вычислительной геометрии. В течение последних двадцати лет это развитие происхо- дило бурными темпами, в результате чего на свет появилось множество полезных ал- горитмов, программ, исследовательских работ и учебников. Среди книг по вычисли- тельной геометрии можно выделить следующие: ♦ [dBvKOSOO] — самое лучшее введение в общую теорию вычислительной геометрии и ее основные алгоритмы; ♦ [O'ROl] — прекрасное практическое введение в вычислительную геометрию. В нем особое внимание уделяется тщательной и правильной реализации алгоритмов. Код на языках С и Java можно загрузить с веб-страницы http://maven.smith.edu/ —orourke/books/compgeom.html; ♦ [PS85]— несмотря на солидный возраст, эта книга остается хорошим введением в вычислительную геометрию. В ней подробно описаны алгоритмы построения вы- пуклой оболочки, построения диаграмм Вороного и выявления пересечений; ♦ [GO04] — недавно изданный сборник статей содержит подробный обзор положения дел во всех областях дискретной и вычислительной геометрии. Симпозиум ассоциации АСМ по вычислительной геометрии (ACM Symposium on Computational Geometry) проводится ежегодно в последних числах мая или в первых числах июня. Хотя на нем представляются, в основном, теоретические результаты, на- учное сообщество прилагает усилия по расширению присутствия прикладных и экспе- риментальных работ, используя для этого видеообзоры и стендовые доклады. Существует постоянно расширяющаяся база реализаций алгоритмов вычислительной геометрии. Как правило, в каталоге указываются конкретные реализации, но вам сле- дует обратить особое внимание на обширную библиотеку CGAL (Computational Geometry Algorithms Library, библиотека алгоритмов вычислительной геометрии). Она содержит реализации большого количества алгоритмов на языке C++, разработан- ные как часть крупного общеевропейского проекта. Каждому, кто серьезно интере- суется вычислительной геометрией, следует ознакомиться с этой библиотекой (http://www.cgal.org).
Глава 17. Вычислительная геометрия 577 17.1. Элементарные задачи вычислительной геометрии Вход. Точка р и отрезок / или два отрезка, Ц и Л. Задача. Выяснить, находится ли точка р на отрезке I. а если не находится, то по какую сторону от него она расположена. Выяснить, пересекаются ли отрезки 1\ и 12 (рис. 17.1). ВХОД ВЫХОД Рис. 17.1. Пересечение прямых Обсуждение. Элементарные задачи вычислительной геометрии имеют свои "подвод- ные камни" даже в таких простых случаях, как поиск точки пересечения двух прямых. Эта задача сложнее, чем кажется на первый взгляд. Например, какой результат нужно возвратить, если прямые параллельны? Как поступить, если прямые совпадают, и пере- сечением является не точка, а вся прямая? Если одна из прямых расположена горизон- тально, в процессе решения уравнений придется выполнить деление на нуль. Если прямые почти параллельны, точка пересечения находится так далеко от начала коор- динат, что при вычислении ее местоположения возникнет арифметическое переполне- ние. Ситуация значительно усложняется при поиске пересечения двух отрезков, по- скольку тогда возникает много частных случаев, которые нужно отслеживать. Если вы новичок в области реализации алгоритмов вычислительной геометрии, я ре- комендую вам изучить книгу [O'ROl], в которой содержатся практические советы и законченные реализации основных алгоритмов вычислительной геометрии и структур данных. Мы имеем дело с двумя разными понятиями: геометрической вырожденностью и чис- ленной устойчивостью. Под вырожденностью (degeneracy) понимаются частные слу- чаи, требующие специальных подходов, например, такие, в которых две прямые имеют несколько точек пересечения. Существует три основных подхода к решению проблемы вырожденности: ♦ игнорировать ее. Выдвигаем рабочую гипотезу, что наша программа будет работать правильно только в тех случаях, когда никакие три точки не коллинеарны, никакие три прямые не пересекаются в одной точке, пересечение отрезков не происходит в их конечных точках и т. д. Это, пожалуй, самый распространенный подход, который я могу рекомендовать для краткосрочных проектов, если допустимы частые отказы программы. Недостаток такого подхода обусловлен тем фактом, что самые инте- 19 Зак 3711
578 Часть II. Каталог алгоритмических задач ресные данные снимаются с линий решетки и поэтому неизбежно являются сильно вырожденными; ♦ подослать невырожденность. Случайным образом внесите беспорядок в данные, чтобы сделать их невырожденными. Перемещая каждую точку на небольшое рас- стояние в случайном направлении, можно исключить вырожденность данных, не создавая при этом слишком много новых проблем. Это первое, что вы должны по- пробовать. когда решите, что частота отказов вашей программы превышает допус- тимый уровень. Недостаток этого подхода состоит в том, что случайные изменения могут исказить входные данные неприемлемым для вашего приложения образом. Существуют способы внесения "символических" возмущений в данные для непро- тиворечивого исключения вырожденности, но корректное применение этих спосо- бов требует серьезных исследований; ♦ обрабатывать вырожденность. Программу можно сделать более устойчивой, до- бавив в нее специальный код для обработки каждого частного случая. Такой подход может быть оправдан, если его осуществить в начале разработки программы, но не- допустимо добавлять "заплатки" при каждом сбое программы. Если вы решительно настроены на применение этого подхода, будьте готовы вложить в его реализацию значительные усилия. В геометрических вычислениях часто применяются арифметические операции с пла- вающей точкой, из-за чего могут возникнуть такие проблемы, как переполнение и по- теря точности. Существуют три основных подхода к обеспечению численной устойчи- вости: ♦ использование целочисленных арифметических операций. Принудительное разме- щение всех представляющих интерес точек на целочисленной решетке позволит выполнять точные проверки двух чисел на равенство или двух отрезков на пересе- чение. Нужно отдавать себе отчет в том, что точка пересечения двух прямых не все- гда будет находиться строго на решетке. Тем не менее, это самый простой и лучший способ при условии, что он дает приемлемые результаты; ♦ использование действительных чисел двойной точности. При операциях над чис- лами двойной точности с плавающей точкой вам может повезти настолько, что вы избежите численных ошибок. Однако лучше всего для представления данных ис- пользовать действительные числа одинарной точности, а двойную точность приме- нять для промежуточных вычислений; ♦ использование арифметических операций произвольной точности. Алгоритм, ис- пользующий такой подход, несомненно даст правильные результаты, но будет рабо- тать очень медленно. Однако он завоевывает все большую популярность в научном сообществе. Тщательный анализ может свести к минимуму необходимость в ариф- метических операциях с высокой точностью и, следовательно, потерю производи- тельности. Тем не менее вы должны понимать, что операции с высокой точностью выполняются на несколько порядков медленнее, чем стандартные арифметические операции с плавающей точкой. Разработка устойчивого программного обеспечения для решения задач вычислитель- ной геометрии связана со значительными трудностями. На практике самым лучшим подходом будет создание приложений на основе небольшого набора алгоритмов вы-
Глава 17. Вычислительная геометрия 579 числительной геометрии, выполняющих большой объем вычислений низкого уровня. В число таких алгоритмов входят следующие: ♦ вычисление площади треугольника. Хотя хорошо известно, что площадь А(1) тре- угольника I = (а. Ь, с) равна половине произведения его основания на высоту, вы- числение длины основания и высоты требует кропотливых вычислений с примене- нием тригонометрических функций. Лучше вычислить двойную площадь, используя следующую формулу: 2-Л(/) = Ьх av 1 Ьу 1 = axb - avbx + avcx - ахсу + bxcv - c\br Эта формула обобщает задачу вычисления б?!-кратного объема многогранника в г/-мерном пространстве. Таким образом, в трехмерном пространстве шестикрат- ный (3! = 6) объем четырехгранника t = (a, b, с, d) будет равен: 6Л(/) = О,, az ' br b. 1 > - cv c. 1 dv d. 1 Обратите внимание, что эти значения могут быть отрицательными, поэтому в даль- нейших вычислениях следует использовать их модули. Определители обсуждаются в разделе 13.4. Концептуально самым простым способом вычисления площади многоугольника (или многогранника) будет выполнение его триангуляции с последующим суммиро- ванием площадей всех полученных треугольников. Реализацию изящного алгорит- ма, в котором не используется триангуляция, см. в [O'ROl] и [SR03]; ♦ выяснение местоположения точки. Как расположена точка с относительно прямой /? Чтобы дать корректный ответ на этот вопрос, следует представить прямую / в ви- де направленной линии, проходящей через точку а раньше, чем через точку Л, и вы- яснить, слева или справа от этой линии находится точка с. Решить эту задачу можно, используя знак определителя, соответствующего площа- ди треугольника, вычисленной способом, приведенным выше. Если площадь тре- угольника t(a, b, с)>0, значит, точка с находится слева от ab. Если площадь тре- угольника t(a, b, с) = 0, значит, точка с расположена на ab. Наконец, если площадь треугольника 1(а, Ь, с) < 0, значит, точка с находится справа от ab. Этот подход легко обобщается для трех измерений, где знак вычисленного опреде- лителя, соответствующего площади, показывает местоположение точки d над ори- ентированной плоскостью (а, Ь, с), под плоскостью или на ней; ♦ проверка пересечения прямой и отрезка. Предшествующий способ выяснения рас- положения точки можно использовать и для проверки пересечения прямой с отрез- ком. Такое пересечение имеет место тогда и только тогда, когда одна конечная точ- ка отрезка находится слева от прямой, а вторая — справа. Проверка пересечения
580 Часть II. Каталог алгоритмических задач отрезков является сходной, но более сложной задачей. Ответ на вопрос, пересека- ются ли два отрезка, если они имеют общую концевую точку, зависит от конкретно- го приложения. Это типичная ситуация, в которой возникают проблемы с вырож- денностью: ♦ выяснение, находится ли точка внутри круга. Необходимость в проверке, находит- ся ли точка d внутри круга, определяемого точками а, b и с, возникает во всех алго- ритмах триангуляции Делоне. Соответствующий алгоритм может быть использован в качестве надежного способа сравнения расстояний. Предполагая, что обозначение точек а. Ь. и с идет против часовой стрелки, вычислим определитель incircle-. incircle(a,b,c) = ах ау ах । bf b2+b2 1 Су С< + СГ ' < 1 Значение определителя incircle будет нулевым, если все четыре точки лежат на кру- ге, положительным, если точка d лежит внутри круга, и отрицательным, если точка d находится вне круга. Прежде чем пытаться создать свою собственную реализацию, ознакомьтесь с уже су- ществующими, включая указанные в разделе "Реализации". Реализации. Библиотеки CGAL (www.cgal.org) и LEDA (см. раздел 19.1.1) содержат реализации алгоритмов вычислительной геометрии, написанные на языке C++. С биб- лиотекой LEDA легче работать, но библиотека CGAL полнее и, к тому же, бесплатна. Если вы разрабатываете серьезное геометрическое приложение, вам непременно сле- дует ознакомиться с возможностями этих библиотек, прежде чем создавать свою соб- ственную реализацию. В книге [O'ROl] представлены реализации на языке С большинства алгоритмов, обсу- ждаемых в этом разделе. Подробности см. в разделе 19.1.10. Хотя эти реализации были созданы, в основном, для ознакомительных целей, а не для коммерческого примене- ния, они надежны и годятся для небольших приложений. Библиотека Core (http://cs.nyu.edu/exact/) содержит API-интерфейс, в котором исполь- зуется подход EGC (Exact Geometric Computation, точные геометрические вычисления) к созданию численно устойчивых алгоритмов. Эту библиотеку можно использовать с небольшими изменениями в любой программе на языке C/C++, с легкостью поддер- живая три уровня точности: машинную точность, произвольную точность и гарантиро- ванную точность. Устойчивую реализацию основных алгоритмов вычислительной геометрии на язы- ке C++, представленную в работе [She97], можно загрузить по адресу http:// www.cs.cmu.edu/~quake/robust.html. Примечания Книга [O'ROl] — прекрасное практическое введение в вычислительную геометрию. В ней особое внимание уделяется тщательной и правильной реализации алгоритмов. Библиотека LEDA (см. [MN99]) является еще одним отличным образцом для подражания для тех, кто собирается разрабатывать собственные реализации.
Глава 17 Вычислительная геометрия 581 В работе [Yap04] дан замечательный обзор методов построения устойчивых реализаций. Когда писались эти строки, на ее основе готовилась книга (см. [MY07]). В работе [КМР'04] дано графическое представление проблем, возникающих при использовании арифметических операций над действительными числами в алгоритмах вычислительной геометрии, таких как поиск выпуклой оболочки. Управляемое внесение возмущений в данные (см. [MOS06]) является новым подходом к обеспечению устойчивых вычислений, пользующимся значительным вниманием. В работах [She97] и [FvW93] представлены ре- зультаты подробных исследований стоимости арифметических операций произвольной точности для геометрических вычислений. Если использовать эти операции с должной осторожностью, можно обеспечить приемлемую эффективность вычислений при их пол- ной устойчивости. Родственные задачи. Выявление пересечений (см. раздел 17.8), конфигурации прямых (см. раздел 17 15). 17.2. Выпуклая оболочка Вход. Множество S из п точек в d-мерном пространстве. Задача. Найти наименьший выпуклый многоугольник (или многогранник), содержа- щий все точки множества Л’ (рис. 17.2). ВХОД ВЫХОД Рис. 17.2. Выпуклая оболочка Обсуждение. Задача построения выпуклой оболочки набора точек является самой важ- ной элементарной задачей вычислительной геометрии, точно так же, как задача сорти- ровки является самой важной задачей комбинаторных алгоритмов. Важность этой за- дачи обусловлена тем, что создание выпуклой оболочки позволяет получить приблизи- тельное представление о форме или размере набора данных. Построение выпуклой оболочки является первым шагом предварительной обработки в большинстве алгоритмов вычислительной геометрии. Рассмотрим, например, задачу поиска диаметра набора точек, иначе говоря, поиска пары точек с наибольшим рас-
582 Часть II. Каталог алгоритмических задач стоянием между ними. Диаметр равен расстоянию между двумя точками, расположен- ными на выпуклой оболочке. Алгоритм вычисления диаметра (имеющий время испол- нения O(nlgn)) начинает работу с создания выпуклой оболочки, после чего для каждой вершины оболочки он находит на оболочке вершину, наиболее удаленную от данной. Так называемый "метод штангенциркуля" можно использовать для быстрого переме- щения от одного конца диаметра оболочки к другому, двигаясь вдоль оболочки по ча- совой стрелке. Существует почти такое же количество алгоритмов для построения выпуклой оболоч- ки, как и алгоритмов сортировки. При выборе наиболее подходящего из них ответьте на следующие вопросы. ♦ С каким количеством измерений вы имеете дело? С выпуклыми оболочками в двух- и даже трехмерном пространстве работать довольно легко. Но некоторые предпо- ложения, справедливые для небольшого количества измерений, становятся недейст- вительными при увеличении количества измерений. Например, в двумерном про- странстве любой многоугольник с п вершинами имеет ровно и ребер. Но с увеличе- нием количества измерений всего лишь до трех, зависимость между количеством вершин и граней становятся более сложной. Например, куб имеет восемь вершин и шесть граней, а восьмигранник — шесть вершин и восемь граней. Поэтому при вы- боре структур данных, представляющих оболочки, принципиально важно, требуется ли нам всего лишь найти точки оболочки или мы должны построить многогранник. Следует иметь в виду такие особенности многомерных пространств, если при реше- нии своей задачи вы будете вынуждены работать с ними. Существуют простые алгоритмы для построения выпуклой оболочки с временем исполнения O(nlogn) для двумерных и трехмерных случаев. С увеличением размер- ности задача становится более сложной. Для создания выпуклых оболочек в много- мерных пространствах используется базовый алгоритм, называющийся алгоритмом заворачивания подарка (gift-wrapping algorithm). Обратите внимание, что трехмер- ный выпуклый многогранник состоит из двумерных граней, соединенных одномер- ными ребрами. Каждое ребро соединяет ровно две грани. Заворачивание подарка начинается с определения начальной грани, связанной с самой нижней вершиной, после чего от этой грани выполняется поиск в ширину, чтобы найти следующие грани. Каждое ребро е, принадлежащее грани, должно быть общим с какой-то дру- гой гранью. Перебрав все п точек, мы можем выяснить, какая точка определяет дру- гую грань ребра е. Образно говоря, мы "оборачиваем" точки по одной грани за раз, сгибая оберточную бумагу по ребрам, пока она не закроет первую точку. Для обеспечения эффективности алгоритма необходимо, чтобы каждое ребро ис- следовалось только один раз. Время исполнения корректно реализованного алго- ритма для d измерений равно O(n0j_| + Фи^Фа-г), где Ф,^ — количество граней, а Фд-2— количество ребер в выпуклой оболочке. Но если выпуклая оболочка очень сложна, то время исполнения может ухудшиться до О(и^/2^+1). Я рекомендую ис- пользовать готовые реализации, а не пытаться разработать свою собственную. ♦ Данные представлены в виде вершин или в виде полупространств? Задача поиска пересечения набора п полупространств в d-мерном пространстве (каждое из кото- рых содержит начало координат) двойственна задаче поиска выпуклых оболочек из
Гпава 17. Вычислительная геометрия 583 п точек в (/-мерном пространстве. Таким образом, для решения обеих задач доста- точно одного алгоритма. Требуемое преобразование рассматривается в разде- ле 17.15. Когда внутренняя точка не задана, задача поиска пересечения полуплоско- стей перестает быть двойственной задаче поиска выпуклой оболочки, поскольку со- ответствующий экземпляр не всегда возможен (пересечение полуплоскостей может оказаться пустым). ♦ Сколько точек может содержать оболочка? Если набор точек сгенерирован псев- дослучайным образом, большинство точек будет, скорее всего, находиться внутри оболочки. Практическую эффективность программ построения выпуклых оболочек на плоскости можно повысить, используя то обстоятельство, что самая левая, самая правая, самая верхняя и самая нижняя точки должны находиться на выпуклой обо- лочке. Таким образом, мы имеем набор из трех или четырех разных точек, опреде- ляющих треугольник или четырехугольник. Любая точка внутри этой области не может находиться на выпуклой оболочке и. следовательно, ее можно отбросить путем линейного перебора точек. В идеале, после этого останется небольшое коли- чество точек, которые можно просмотреть с помощью алгоритма построения вы- пуклой оболочки. Этот прием можно применять в многомерных пространствах, но с увеличением ко- личества измерений его эффективность снижается. ♦ Как выяснить форму данного набора точек? Хотя выпуклая оболочка дает прибли- зительное представление о форме, многие подробности теряются. Например, вы- пуклая оболочка для набора точек в форме буквы G (см. рис. 17.2) неотличима от выпуклой оболочки для буквы О. Более общими структурами, которые можно па- раметризировать, чтобы сохранить "невыпуклость" множества точек, являются аль- фа-очертания (alpha shapes). Ссылки на справочный материал и реализации соот- ветствующих алгоритмов приводятся далее. Основным алгоритмом построения выпуклой оболочки на плоскости является скани- рование по Грэхему. Алгоритм Грэхема начинает работу с точки р, заведомо находя- щейся на выпуклой оболочке (например, с точки, имеющей самое меньшее значение координаты х), и сортирует остальные точки в порядке возрастания полярного угла, измеряемого против часовой стрелки, относительно этой точки. Начиная с выпуклой оболочки, состоящей из точки р и точки v, имеющей наименьший полярный угол, ал- горитм выполняет перебор всех точек вокруг точки v против часовой стрелки с добав- лением новых точек в оболочку. Если угол между очередной точкой и последним реб- ром оболочки меньше, чем 180°, то эта точка добавляется в оболочку. А если этот угол больше 180°, то цепочка вершин, начинающаяся с последнего ребра оболочки, подле- жит удалению для сохранения выпуклости. Общее время исполнения алгоритма равно O(«lg«), т. к. узким местом является сортировка точек вокруг v. Базовую процедуру алгоритма Грэхема можно использовать для создания простого (не имеющего самопересечений) многоугольника, проходящего через все заданные точки. Для этого сортируем точки вокруг вершины v, но вместо проверки углов соединяем точки в том порядке, в каком они отсортированы. В результате получим многоуголь- ник без самопересечений, но, нередко, с большим количеством некрасивых выступаю- щих частей.
584 Часть II. Каталог алгоритмических задач Алгоритм заворачивания подарка значительно упрощается в двумерном пространстве, когда каждая "грань" становится ребром, а каждое "ребро" — вершиной многоуголь- ника, и "поиск в ширину" выполняется вдоль оболочки, по или против часовой стрел- ки. Время исполнения алгоритма заворачивания подарка для двух измерений равно O(nh), где h— количество вершин в выпуклой оболочке. Я рекомендую использовать алгоритм Грэхема за исключением случаев, когда вы точно знаете заранее, что обо- лочка не содержит слишком много вершин. Реализации. Библиотека CGAL (www.cgal.org) включает в себя реализации на языке C++ разнообразных алгоритмов построения выпуклой оболочки для произвольного количества измерений. Альтернативные реализации на языке C++ алгоритмов по- строения выпуклых оболочек содержатся в библиотеке LEDA (см. раздел 19.1.1). Популярной реализацией алгоритма построения выпуклых оболочек в пространствах с небольшим количеством измерений является программа Qhull (cm.[BDH97]). оптими- зированная для работы в диапазоне от двух до восьми измерений. Программа написана на языке С и может также создавать триангуляции Делоне, диаграммы Вороного, а также пересечения полупространств. Программа Qhull широко используется в научных приложениях и доступна на веб-сайте http://www.qhull.org/. В книге [O'ROl] дана устойчивая реализация алгоритма Грэхема для двух измерений и реализация с временем исполнения О(п~) инкрементального алгоритма построения вы- пуклых оболочек в трех измерениях. Программы написаны на языках С и Java. Под- робности см. в разделе 19.1.10. Замечательные программы для построения альфа-очертаний, основанные на работе [ЕМ94], доступны на веб-сайте http://biogeometry.duke.edu/software/alphashapes/ Реализацию Hull для построения выпуклых оболочек в многомерных пространствах также можно использовать для построения альфа-очертаний. Программа доступна на веб-сайте http://www.netlib.org/voronoi/hull.html. Для перечисления вершин пересекающихся полупространств с большим количеством измерений требуются другие реализации. Программа Ihs (http://cgm.cs.mcgill.ca/ ~avis/C/lrs.html) является арифметически устойчивой реализацией на языке ANSI С алгоритма поиска в обратном направлении Ависа-Фукуды для решения задач перечис- ления вершин и построения выпуклых оболочек. Так как обход многогранников вы- полняется неявным образом и они явно не сохраняются в памяти, то иногда можно ре- шить даже задачи с очень большим размером вывода. Примечания Роль плоских выпуклых оболочек в вычислительной геометрии сравнима с ролью сорти- ровки в комбинаторике. Подобно сортировке, задача построения выпуклой оболочки яв- ляется фундаментальной, и разнообразные подходы ведут к созданию интересных или оп- тимальных алгоритмов. Алгоритмы Quickhull и mergehull являются примерами алгорит- мов построения выпуклой оболочки, на создание которых разработчиков вдохновили алгоритмы сортировки (см. [PS85]). Простая конструкция из точек на параболе, рассмот- ренная в разделе 9.2 7, сводит задачу построения выпуклой оболочки к задаче сортировки, так что теоретическая нижняя граница сортировки позволяет сделать вывод, что для по- строения плоской выпуклой оболочки требуется время Q(nlgn). Более сильная нижняя граница установлена в работе [Уао81].
Гпава 17. Вычислительная геометрия 585 Подробное обсуждение алгоритмов Грэхема (см. [Gra72]) и Джарвиса (см. [Jar73]) содер- жится в книгах [dBvKOSOO], [CLRS01], [O'ROl] и [PS85]. Алгоритм для построения пло- ской выпуклой оболочки (см. [K.S86]) выдает оптимальное решение за время O(?7lg/7), где h— количество вершин оболочки, которое обеспечивает наилучшую производительность как алгоритма Грэхема, так и алгоритма заворачивания подарка и является (теоретически) оптимальным вариантом. Вычисление планарных выпуклых оболочек можно эффективно выполнять "на месте", т. е. без выделения дополнительной памяти (см. [В!К+04]). В работе [Sei04] представлен самый свежий обзор алгоритмов построения выпуклой оболочки и их вариантов, в особенности для многомерных пространств. Альфа-очертания, обсуждаемые в работе [EK.S83], дают хорошее представление о форме набора точек. Обобщение оболочек этого типа для трехмерного пространства и соответ- ствующая реализация приведены в работе [ЕМ94]. Алгоритмы поиска в обратном направлении для построения выпуклых оболочек эффек- тивно работают и в многомерных пространствах (см. [AF96]). При удачном построении отображения в пространство с большим количеством измерений (см. [ES86]) задачу соз- дания диаграммы Вороного в <7-мерном пространстве можно свести к задаче создания вы- пуклой оболочки в (d + 1 )-мерном пространстве. Подробности см. в разделе П 4 Алгоритмы, предназначенные для работы с выпуклыми оболочками, используют специ- альные структуры данных, которые позволяют выполнять вставку и удаление произволь- ных точек и при этом непрерывно отображают текущую выпуклую оболочку. Первая из таких динамических структур данных (см. [OvL81]) поддерживала вставки и удаление за время O(lg">7) Позже стоимость этих операций была понижена до почти логарифмическо- го амортизированного времени (см. [ChaOl ]). Родственные задачи. Сортировка (см. раздел 14.1), диаграммы Вороного (см. раз- дел 17 4). 17.3. Триангуляция Вход. Набор точек или многогранник. Задача. Разбить внутреннюю часть набора точек или многогранника на треугольники (рис. 17.3). ВХОД ВЫХОД Рис. 17.3. Триангуляция
586 Часть II. Каталог алгоритмических задач Обсуждение. Обычно первым шагом в обработке сложных геометрических объектов является разбиение их на простые геометрические фигуры. Это обстоятельство делает триангуляцию фундаментальной задачей вычислительной геометрии. Самым простым двумерным геометрическим объектом является треугольник, а самым простым трех- мерным— четырехгранник. В число классических применений триангуляции входят анализ методом конечных элементов и компьютерная графика. Особенно интересным применением триангуляции является интерполирование по- верхности или интерполирование функций. Допустим, что имеются значения высоты горы в некотором множестве точек. Как установить приблизительную высоту горы в заданной точке q? Мы можем проецировать точки, в которых замерена высота, на плоскость, а потом выполнить триангуляцию. Плоскость будет разбита на треугольни- ки, позволяющие приблизительно вычислить высоту точки q путем интерполирования высот вершин треугольника, содержащего эту точку. Кроме того, триангуляция и соот- ветствующие значения высот определяют поверхность горы, пригодную для создания компьютерного изображения. Триангуляция на плоскости выполняется путем соединения точек непересекающимися отрезками до тех пор, пока добавление новых отрезков не становится невозможным. Вы должны ответить на следующие вопросы. ♦ Триангулируется набор точек или многоугольник? Обычно приходится триангули- ровать набор точек, как в рассмотренной ранее задаче интерполирования поверхно- сти. Для решения этой задачи нужно сначала создать выпуклую оболочку для дан- ного набора точек, а потом разбить внутренюю область полученной оболочки на треугольники. Самый простой алгоритм обработки набора точек, имеющий время исполнения O(//lg/z), сначала сортирует точки по х-координате. Потом отсортированные точки выбираются слева направо, согласно алгоритму построения выпуклой оболочки, описанному в разделе 4.1. При этом выполняется триангуляция путем добавления ребра к каждой новой отрезанной от оболочки точке. ♦ Имеет ли значение внешний вид треугольников триангуляции? Обычно вход можно разбить на треугольники многими разными способами. Рассмотрим набор из п то- чек в выпуклой оболочке на плоскости. Самый простой способ выполнить триангу- ляцию этих точек — добавить к выпуклой оболочке диагонали, идущие от первой точки до всех остальных. Но в таком случае обычно появляются "узкие" треуголь- ники. Во многих приложениях необходимо избегать "узких" треугольников или. что тоже самое, минимизировать количество очень острых углов триангуляции. Триангуля- ция набора точек методом Делоне минимизирует максимальный угол по всем воз- можным вариантам триангуляции. Правда, это не совсем то, что нам требуется, но. в принципе, весьма близко. Вообще, триангуляция методом Делоне имеет достаточ- но много других интересных свойств (включая то. что она является двойственной задаче построения диаграмм Вороного), чтобы я рекомендовал предпочитать ее другим методам триангуляции. Кроме того, используя описанные далее реализации, такую триангуляцию можно выполнить за время OQzIgw).
Глава 17. Вычислительная геометрия 587 ♦ Как можно улучшить результат триангуляции? Каждое внутреннее ребро любой триангуляции принадлежит двум треугольникам. Четыре вершины, определяющие эти два треугольника, образуют выпуклый или невыпуклый четырехугольник. Дос- тоинство выпуклого варианта заключается в том, что при замене внутреннего ребра на ребро, связывающее две другие вершины, получается другая триангуляция. Иначе говоря, мы имеем локальную операцию замены ребер, посредством которой можно модифицировать и, не исключено, улучшать данную триангуляцию. В самом деле, триангуляцию Делоне можно получить из любой триангуляции, удаляя "узкие" треугольники до тех пор, пока не будут исчерпаны все возможности для локальных улучшений. ♦ Какова размерность задачи? Трехмерные задачи обычно гораздо сложнее, чем двумерные. Обобщение процедуры триангуляции до трех измерений заключается в разбиении пространства на четырехвершинные четырехгранники путем добавления непересекающихся граней. Трудность заключается в том, что некоторые много- гранники нельзя разбить на четырехгранники, не добавляя дополнительные верши- ны. Кроме этого, задача выяснения, существует ли такое разбиение на четырехгран- ники, является NP-полной, поэтому вы можете смело добавлять новые вершины, чтобы упростить задачу. ♦ Каковы ограничивающие условия на входные данные? При триангуляции много- угольника или многогранника имеется естественное ограничение, запрещающее до- бавлять ребра, пересекающие какие-либо внешние грани. Вообще говоря, может существовать набор препятствий, которые нельзя пересекать добавляемыми ребра- ми. Самой лучшей триангуляцией при таких условиях будет, скорее всего, так назы- ваемая триангуляция Делоне с ограничениями. ♦ Можно ли добавлять дополнительные точки или перемещать существующие? Ко- гда внешний вид треугольников не имеет значения, может оказаться выгодным до- бавление небольшого количества промежуточных точек в набор данных с целью облегчить создание триангуляции, удовлетворяющей заданному условию, напри- мер, требованию отсутствия очень острых углов. Как упоминалось ранее, триангу- ляцию некоторых многогранников невозможно выполнить без добавления допол- нительных точек в исходные данные. Чтобы выполнить триангуляцию выпуклого многоугольника за линейное время, просто выберем случайную начальную вершину v и соединим ее ребрами со всеми остальны- ми вершинами многоугольника. Так как многоугольник выпуклый, то мы можем быть уверены в том, что добавляемые ребра не будут пересекать стороны многоугольника и что все они будут находиться внутри многоугольника. Самый простой алгоритм для выполнения общей триангуляции многоугольника проверяет каждое из О(п2) возмож- ных ребер на пересечение с граничными или ранее вставленными ребрами и вставляет его только в случае отсутствия такого пересечения. Существуют пригодные для прак- тического применения алгоритмы с временем исполнения O(«lg«) и представляющие теоретический интерес алгоритмы с линейным временем исполнения. Подробности см. в подразделах "Реализации" и "Примечания". Реализации. Пакет Triangle, разработанный Джонатаном Шевчуком (Jonathan Shew- chuk), содержит набор процедур на языке С для генерирования триангуляций Делоне,
588 Часть II. Каталог алгоритмических задач триангуляций Делоне с ограничениями (т. е. триангуляций, в которых некоторые ребра вставляются в принудительном порядке), а также триангуляций Делоне, обеспечиваю- щих качественный результат (т. е. таких, в которых очень острые углы исключаются с помощью вставки промежуточных точек). Этот высокопроизводительный и устойчи- вый пакет широко используется для анализа методом конечных элементов. Если бы мне требовалась реализация алгоритма двумерной триангуляции, то я начал бы с про- грамм из пакета Triangle. Загрузить пакет можно с веб-страницы http://www. cs.cmu.edu/~quake/triangle.html. Программа Sweep2, написанная на языке С, прекрасно подходит для создания двумер- ных диаграмм Вороного и триангуляций Делоне. С ней особенно удобно работать, ко- гда вам требуется лишь триангуляция Делоне точек на плоскости. Программа основана на алгоритме заметающей прямой (sweepline alogirthm) для создания диаграмм Вороно- го (см. [For87]). Создание сеток для методов конечных элементов является обширной областью для исследований. Веб-страница Стива Оуэна (Steve Owen) Meshing Research Corner, (http://www.andrew.cmu.edu/user/sowen/mesli.html) содержи! исчерпывающий обзор литературы в этой области, а также ссылки на многочисленные реализации. Особенно рекомендую программу QMG. доступную по адресу http://www.cs.cornell.edu/ Info/PeopIe/vavasis/qmg-home.html. Как библиотека CGAL (www.cgal.org), так и библиотека LEDA (см. раздел 19.1.1) со- держат реализации на языке C++ самых разных алгоритмов триангуляции в двух- и трехмерных пространствах, включая триангуляции Делоне с ограничениями. В многомерных пространствах триангуляции Делоне представляют собой частный случай выпуклых оболочек. Популярной реализацией алгоритма построения выпуклых оболочек в пространствах с небольшим количеством измерений является программа Qhull (см. [BDH97]), оптимизированная для работы в диапазоне от двух до восьми из- мерений. Программа написана на языке С и может также создавать триангуляции Делоне, диаграммы Вороного, а также пересечения полупространств. Программа Qhull широко используется в научных приложениях и доступна на веб-сайте http://www.qhull.org/. В качестве альтернативы доступна программа Hull, предназна- ченная для построения выпуклых оболочек в многомерных пространствах (http://www.netlib.org/voronoi/hull.htnil). Примечания После длительных исследований Шазель (Chazelle) создал линейный алгоритм триангуля- ции простого многоугольника (см. [Cha91]). Но реализация этого алгоритма является ис- ключительно трудной задачей, так что он больше подходит в качестве доказательства су- шествования триангуляционного разбиения. Первый алгоритм триангуляции многоуголь- ников с временем исполнения O(nlgw) был представлен в работе [GJPT78], После создания этого алгоритма (но раньше появления алгоритма Шазеля) Тарьян и ван Вик разработали еще один алгоритм с таким же временем исполнения (см. [TW88]). Обзор по- следних результатов в области триангуляции наборов точек и многоугольников представ- лен в [Вег04а]. Для исследователей, интересующихся генерированием сеток и решеток, проводится еже- годная конференция International Meshing Roundtable. Отличными обзорами методов генерирования сеток являются работы [Вег02] и [EdeO6J.
Глава 17. Вычислительная геометрия 589 Алгоритмы с линейным временем исполнения для триангуляции монотонных много- угольников давно известны (см. [GJPT78]) и составляют базу для алгоритмов триангуля- ции простых многоугольников. Монотонным называется многоугольник, для которого существует такое направление, что любая прямая, идущая в этом направлении, пересекает многоугольник, самое большее, в двух точках. Активно исследуемый класс оптимальных триангуляций включает в себя решения, стре- мящиеся минимизировать суммарную длину использованных ребер. В работе [MR06] бы- ло доказано, что такая задача является NP-полной. После этого внимание исследователей переключилось на доказуемо хорошие аппроксимирующие алгоритмы. Триангуляцию с минимальным весом выпуклого многоугольника можно выполнить за время О(п\ ис- пользуя динамическое программирование (см. раздел 8.6.1). Родственные задачи. Диаграммы Вороного (см. раздел 17.4), разбиение многоуголь- ников (см.раздеч 17.11). 17.4. Диаграммы Вороного Вход. Множество Sточекр\, ...,рп. Задача. Разбить пространство на области вокруг каждой точки таким образом, чтобы все точки в области вокруг точки р, были ближе к этой точке, чем к любой другой точ- ке множества S'(рис. 17.4). ВХОД ВЫХОД Рис. 17.4. Диаграмма Вороного Обсуждение. Диаграмма Вороного представляет области влияния вокруг точек из дан- ного набора. Если эти точки соответствуют расположению ресторанов быстрого пита- ния, то диаграмма Вороного разбивает пространство на ячейки, окружающие каждый ресторан. Для человека, живущего в конкретной ячейке, соответствующий ресторан является ближайшим местом, где можно заказать горячий бутерброд.
590 Часть II Каталог алгоритмических задач Диаграммы Вороного имеют множество различных применений: ♦ поиск ближайшей точки. Поиск ближайшего соседа точки q из фиксированного множества S точек сводится к задаче поиска ячейки в диаграмме Вороного, которая содержит данную точку. Подробности см. в разделе 17.5; ♦ размещение точек обслуживания. Предположим, что владелец сети ресторанов бы- строго питания хочет открыть еще один ресторан. Чтобы свести к минимуму конку- ренцию с уже имеющимися в этой местности ресторанами, следует разместить но- вый ресторан как можно дальше от ближайшего ресторана. Искомое место всегда будет находиться в вершине диаграммы Вороного, и его можно найти за линейное время, выполнив поиск по всем вершинам; ♦ наибольший пустой круг. Допустим, что требуется найти большой неосвоенный зе- мельный участок, чтобы построить фабрику. Условие, определяющее выбор места для ресторана быстрого питания, применимо и при указании месторасположения нежелательных объектов. Иначе говоря, они должны располагаться как можно дальше от любого интересующего нас места. Вершина Вороного определяет центр наибольшего пустого круга вокруг точек; ♦ разработка маршрута. Если объекты множества S' являются препятствиями, кото- рых следует избегать, то ребра диаграммы Вороного определяют возможные мар- шруты с максимальным расстоянием до препятствий. Таким образом, самый "безо- пасный" путь среди препятствий будет проходить по ребрам диаграммы Вороного; ♦ улучшение триангуляции. Выполняя триангуляцию набора точек, мы обычно хотим получить треугольники без очень острых углов и стараемся избегать "узких" тре- угольников. Триангуляция Делоне максимизирует минимальный угол по всем воз- можным триангуляциям. Кроме того, она легко формулируется, как двойственная задача для диаграммы Вороного. Подробности см. в разделе 17.3. Каждое ребро диаграммы Вороного является отрезком серединного перпендикуляра двух точек в множестве S, т. к. это линия, которая делит плоскость посередине между двумя точками. Самым простым методом создания диаграмм Вороного является ран- домизированное инкрементальное построение. Чтобы добавить новую точку в диа- грамму, мы находим содержащую ее ячейку и добавляем серединные перпендикуляры, которые отделяют эту точку от других, определяющих области, подвергающиеся влия- нию. Если точки вставляются в произвольном порядке, то каждая такая вставка по- влияет, скорее всего, лишь на небольшое количество областей. Однако самым лучшим методом является алгоритм Форчуна, основанный на методе заметающей прямой. Его главное достоинство состоит в том, что для него существуют устойчивые реализации. Алгоритм отображает набор точек на плоскости на набор ко- нусов в трехмерном пространстве таким образом, что диаграмма Вороного определяет- ся отображением конусов обратно на плоскость. Преимущества алгоритма Форчуна; оптимальное время исполнения O(nlogw), легкость реализации и отсутствие необхо- димости сохранять всю диаграмму, если мы можем использовать ее в процессе заме- тания. Между выпуклыми оболочками в (d+ 1)-мерном пространстве и триангуляциями Де- лоне (или. что эквивалентно, диаграммами Вороного) в J-мерном пространстве суще-
Глава 17. Вычислительная геометрия 591 ствует интересная связь. Отобразив каждую точку из (хь х?,.... х(/) на точки (хь Х2,.... xj, х2 ). создав выпуклую оболочку этого (d+ 1)-мерного набора точек, а потом отобразив его обратно в tZ-мерное пространство, мы получим триангуляцию Делоне. Дополнительная информация содержится в подразделе "Примечания", однако изло- женный способ создания диаграмм Вороного в многомерных пространствах является самым лучшим. Программы построения выпуклых оболочек в многомерном простран- стве рассматриваются в разделе 17.2. На практике возникает несколько важных вариантов стандартной диаграммы Воро- ного: ♦ неевклидовы метрики расстояний. Вспомним, что диаграммы Вороного разбивают пространство на зоны влияния вокруг каждой заданной точки. До сих пор мы пред- полагали, что влияние измеряется евклидовым расстоянием, но в некоторых прило- жениях это не так. Например, если люди добираются до ресторана на машине, то требуемое время зависит не от расстояния, а от того, как проложены дороги. Суще- ствуют эффективные алгоритмы для создания диаграмм Вороного при разных мет- риках, а также при наличии объектов неправильной формы; ♦ диаграммы мощности. Эти структуры разбивают пространство на зоны влияния вокруг точек, которые могут иметь разную мощность. Рассмотрим, например, сеть радиостанций, работающих на одной частоте. Зона влияния каждой станции зависит как от мощности ее передатчика, так и от расположения и мощности соседних стан- ций; ♦ диаграммы k-го порядка и диаграммы для самых дальних точек. Идею разбиения пространства на зоны с каким-то общим свойством можно распространить за пре- делы диаграмм Вороного для ближайших точек. Все точки в одной ячейке диа- граммы Вороного /r-го порядка имеют один и тот же набор ближайших точек из множества S. В диаграммах для самых дальних точек все точки внутри определен- ной области имеют одну и ту же самую дальнюю точку из множества S. Расположе- ние точек (см. раздел 17.7) на этих структурах позволяет быстро находить требуе- мые точки. Реализации. Для создания двумерных диаграмм Вороного и триангуляций Делоне широко используется написанная на языке С программа Sweep2. С ней очень просто работать, если вам требуется только диаграмма Вороного. Программа основана на ал- горитме заметающей прямой для создания диаграмм Вороного (см. [For87]). Загрузить ее можно с веб-страницы http://www.netlib.org/voronoi. Как библиотека CGAL (www.cgal.org). так и библиотека LEDA (см. раздел 19.1.1) со- держат реализации на языке C++ самых разных алгоритмов для создания диаграмм Вороного и триангуляций Делоне в двух- и трехмерных пространствах. Диаграммы Вороного в пространствах с большим количеством измерений и диаграм- мы Вороного для самых дальних точек можно создавать как частный случай выпуклых оболочек в многомерном пространстве. Популярной реализацией алгоритма построе- ния выпуклых оболочек в пространствах с небольшим количеством измерений являет- ся программа Qhull (cm.[BDH97]), оптимизированная для работы в диапазоне от двух
592 Часть II. Каталог алгоритмических задач до восьми измерений. Программа написана на языке С и может создавать триангуля- ции Делоне, диаграммы Вороного, а также пересечения полупространств. Программа Qhull широко используется в научных приложениях и доступна на веб-сайте http://www.qhull.org/. Альтернативным вариантом является программа Hull для создания выпуклых оболочек высокой размерности (http://www.netlib.org/voronoi/ hull.shar). Примечания Современное название диаграммы Вороного получили по имени математика Г. Вороного, который сообщил о них в своем докладе в 1908 г. Однако Дирихле изучал этот вопрос в 1850 г., вследствие чего эти диаграммы иногда называют разбиениями Дирихле. Наиболее полное обсуждение диаграмм Вороного и их приложений содержится в книге [OBSCOO]. Отличные обзоры диаграмм Вороного и их вариантов, таких как диаграммы мощности, даются в работах [Aur91] и [For04]. Первый алгоритм создания диаграмм Вороного за время O(nlgn) основан на методе "разделяй и властвуй". Он приведен в работе [SH75J. Подробное изложение алгоритма Форчуна (см. [For87]) для создания диаграмм Вороного за время O(»lg«), а также обсуждение связи между триангуляциями Делоне и (d + il- мерными выпуклыми оболочками (см. [ES86]) можно найти в [dBvKOSOO] и [O'ROl]. Для создания диаграммы Вороного А-го порядка пространство разбивается на области та- ким образом, что каждая точка данной области находится ближе всего к одному и тому же набору из А точек. Представленный в работе [ES86| алгоритм позволяет создавать диа- граммы Вороного А-го порядка за время О(п'). Выясняя местоположение точек в этой струкзуре, можно найти А ближайших соседей заданной точки за время О(А + lg«). Обсуж- дение диаграмм Вороного А-го порядка содержится в книгах [O'ROl] и [PS85], Задачу наименьшей охватывающей окружности можно решить за время O(nlg«), исполь- зуя диаграммы Вороного (и - 1)-го порядка. Более того, существует алгоритм с линейным временем исполнения, использующий методы линейного программирования в простран- ствах с небольшим количеством измерений. Линейный алгоритм создания диаграммы Во- роного для выпуклого многоугольника приводится в работе [AGSS89]. Родственные задачи. Поиск ближайшего соседа (см. раздел 17.5), выяснение место- положения точки (см. раздел 17.7), поиск области (см. раздел 17.6). 17.5. Поиск ближайшей точки Входа. Множество S из п точек в d-мерном пространстве; точка q. Задача. Найти точку в А, ближайшую к точке q (рис. 17.5). Обсуждение. Во многих геометрических приложениях возникает необходимость бы- стро найти точку, ближайшую к заданной точке. Классическим примером такой ситуа- ции является разработка диспетчерской системы пожаротушения. Получив вызов, дис- петчер должен найти ближайшую к пожару станцию, чтобы минимизировать время прибытия пожарной команды. Аналогичная ситуация возникает в любом приложении, где требуется сопоставление клиентов с поставщиками услуг. Задача поиска ближайшего соседа также имеет важность в области классификации. Допустим, что у нас имеется база данных избирателей, в которой указаны возраст, рост, вес. образование, уровень дохода и т. п. Про каждого избирателя также известно.
Глава 17. Вычислительная геометрия 593 какой из двух политических партий — демократической или республиканской — он симпатизирует. Нам требуется создать классификатор, позволяющий прогнозировать, как данный избиратель будет голосовать. Каждый избиратель в базе данных представ- лен точкой, помеченной названием партии, в cf-мерном пространстве. Простой класси- фикатор можно создать, присваивая новой точке ту же метку, что и у ее ближайшего соседа. ВХОД ВЫХОД Рис. 17.5. Поиск ближайшей точки Такие классификаторы, основанные на методе поиска ближайшего соседа, применяют- ся достаточно широко. При сжатии изображений методом векторного квантования изображение разбивается на участки размером 8Х8 пикселов. Метод использует зара- нее составленную библиотеку из нескольких тысяч элементов размером 8Х8 пикселов и заменяет каждый участок изображения наиболее похожим элементом из библиотеки. Такой элемент представляется точкой в 64-мерном пространстве, ближайшей к точке, которая представляет рассматриваемый участок изображения. Сжатие, сопровождае- мое некоторой потерей качества изображения, достигается заменой 64 пикселов на идентификатор самого похожего библиотечного элемента. Решая задачу поиска бли- жайшего соседа, ответьте на следующие вопросы. ♦ Каков размер пространства поиска? Если набор данных состоит из небольшого количества точек (скажем, п < 100) или если планируется небольшое количество за- просов, то самый простой подход будет самым лучшим. Сравниваем точку q с каж- дой из п точек набора данных. Более сложные методы стоит рассматривать только в том случае, когда требуется быстрое выполнение запросов для большого количества точек. ♦ Какова размерность пространства? По мере возрастания количества измерений поиск ближайшего соседа становится все труднее. Представленная в разделе 12.6 структура данных kd-деревьев очень хорошо подходит для работы в пространствах с небольшим количеством измерений, даже с плоскостями. Тем не менее при коли- честве измерений, превышающем 20, поиск в kd-деревьях вырождается практически до линейного поиска. Поиск в многомерных пространствах становится трудным по
594 Часть II. Каталог алгоритмических задач той причине, что с увеличением размерности шар радиусом г, представляющий все точки на расстоянии <г от центра, имеет все меньший объем по сравнению с объе- мом куба. Таким образом, любая структура данных, основанная на разбиении мно- жества точек на подмножества внутри охватывающих сфер, будет становиться все менее эффективной по мере увеличения количества измерений. Диаграммы Вороного в двумерном пространстве (см раздел 17.4) предоставляют эффективную структуру данных для поиска ближайшего соседа. Диаграмма Воро- ного на плоскости разбивает эту плоскость на ячейки таким образом, что каждая ячейка, содержащая точку р. содержит все множество точек, расположенных ближе к точке р, чем к любой другой точке множества S. Задача поиска ближайшего сосе- да точки q сводится к задаче поиска ячейки диаграммы Вороного, содержащей точ- ку q. и указания точки данных, связанной с этой ячейкой. Хотя диаграммы Вороно- го можно создавать в многомерных пространствах, с увеличением количества изме- рений их размер быстро возрастает до такой степени, что они становятся непригодными для использования. ♦ Действительно ли нужен ближайший сосед? Поиск точного ближайшего соседа в многомерном пространстве представляет трудную задачу, для которой, скорее все- го, нет решения лучше, чем линейный поиск (т. е. поиск методом исчерпывающего перебора). Но для решения этой задачи существуют эвристические алгоритмы, ко- торые могут довольно быстро найти точку, расположенную достаточно близко от заданной точки. Один из подходов заключается в понижении размерности (dimension reduction). Существуют способы отображения любого набора из п точек в ^/-мерном простран- стве на пространство, имеющее d' = O(lgn/e2) измерений, таким образом, что рас- стояние до ближайшего соседа в пространстве с меньшим количеством измерений превышает расстояние до действительно ближайшего соседа примерно в (1 + е) раз. Отображение точек на произвольную гиперплоскость, имеющую d' измерений, в пространстве Е1, скорее всего, принесет желаемые результаты. Альтернативным подходом является внесение элемента случайности при поиске в структуре данных. Структура данных kd-деревьев позволяет выполнять эффектив- ный поиск ячейки, содержащей точку q, т. е. ячейки, граничные точки которой яв- ляются хорошими кандидатами на роль близких соседей. Теперь допустим, что мы ищем точку </', которая является результатом небольшого случайного перемещения точки q. Мы попадем в другую ячейку поблизости, одна из граничных точек кото- рой может оказаться еще более близким соседом точки q. Многократное повторение подобных рандомизированных запросов позволит нам эффективно потратить на улучшение результата столько времени, сколько мы можем себе позволить. ♦ Набор данных является статическим ичи динамическим? Будут ли в вашем прило- жении выполняться операции вставки или удаления точек данных? Если вставка и удаление выполняются нечасто, то после каждой такой операции имеет смысл зано- во строить структуру данных. В противном случае следует использовать версию kd-дерева, поддерживающую операции вставки и удаления. Граф ближайшего соседа для множества £ из п точек связывает каждую вершину со своим ближайшим соседом. Такой граф является подграфом триангуляции Делоне, и
Глава 17. Вычислительная геометрия 595 поэтому его можно вычислить за время O(nlogn). Это довольно хороший результат, т. к. только поиск ближайшей пары точек множества S' занимает время ©(«log»). Задача поиска пары ближайших точек сводится к сортировке в одномерном простран- стве. В отсортированном наборе чисел пара ближайших точек соответствует двум чис- лам, расположенным рядом друг с другом, поэтому нам нужно только найти мини- мальное расстояние между п- I смежными точками. Предельный случай возникает, когда расстояние между ближайшими точками равно нулю, откуда следует, что эле- менты не являются уникальными. Реализации. Библиотека ANN на языке C++ содержит реализации алгоритмов для точного и аппроксимирующего поиска ближайшего соседа в пространствах произволь- ной размерности. Эти реализации дают хорошие результаты для поиска среди сотен тысяч точек в пространствах, имеющих до 20 измерений. Библиотека поддерживает все £р-нормы. включая евклидово и манхэттенское расстояния. Загрузить библиотеку можно по адресу http://www.cs.umd.edu/~mount/ANN/. Если бы мне пришлось решать задачу поиска ближайшего соседа, я начал бы с этой библиотеки. Апплеты Java из коллекции Spatial Index Demos (http://donar.umiacs.umd.edu/ quadtree/) иллюстрируют различные варианты kd-деревьев. Алгоритмы, реализуемые в апплетах, рассматриваются в книге [Sam06]. Программа KDTREE 2 содержит реализа- ции kd-деревьев на C++ и FORTRAN 95 для эффективного поиска ближайшего соседа в многомерных пространствах. Подробности см. по адресу http://arxiv.org/abs/ physies/0408067. Программа Ranger (см. [MS93]) представляет собой инструмент для визуализации и экспериментов с поиском ближайшего соседа и поиском в ортогональных областях в наборах данных с большим количеством измерений с использованием многомерных деревьев поиска. Программа поддерживает четыре разных типа структур данных: примитивные kd-деревья, медианные kd-деревья, неортогональные kd-деревья и VP-деревья. Программу можно загрузить из хранилища алгоритмов по адресу http://www.cs.sunysb.ed u/~a Igorith. Специализированная программа Nearpt3 предназначена для поиска ближайшего соседа в очень больших наборах точек в трехмерном пространстве. Код можно загрузить с веб-страницы http://wrfranklin.org/Research/nearpt3. В разделе 17.4 представлена информация о полной коллекции реализаций алгоритмов построения диаграмм Вороного. В частности, библиотеки CGAL (www.cgal.org) и LEDA (см. раздел 19.1.1) содержат реализации на языке C++ алгоритмов построения диаграмм Вороного, а также алгоритмов выяснения местоположения точек на плоско- сти для их эффективного использования при поиске ближайшего соседа. Примечания В работе [Ind04] представлен отличный обзор алгоритмов, основанных на методе случай- ных отображений для аппроксимирующего поиска ближайшего соседа в многомерных пространствах. Как теоретические, так и экспериментальные результаты (см. [IM04] и [ВМ01] соответственно) указывают на то, что этот метод обладает довольно высокой точ- ностью. Несколько иную теоретическую основу имеет программа ANN для аппроксимирующего поиска ближайшего соседа (см. [АМ93, AMN 98]). На основе набора данных создается
596 Часть II. Каталог алгоритмических задач структура разреженного взвешенного графа, после чего поиск ближайшего соседа осуще- ствляется выполнением "жадного" обхода от произвольной точки до точки запроса. Бли- жайшая точка, найденная в результате нескольких таких попыток, объявляется ближай- шим соседом. Подобные структуры данных оказываются полезными и при решении дру- гих задач в многомерных пространствах. Предметом пятых соревнований DIMACS был поиск ближайшего соседа (см. [GJM02]). Самым лучшим справочником по kd-деревьям и другим пространственным структурам данных является книга [Sam06]. В ней подробно рассмотрены все основные (и многие второстепенные) варианты этих структур. Книга [Sam05] представляет собой краткий об- зор той же темы. Метод случайных смешений точки запроса был представлен в работе [РапОб]. Хорошие описания задачи поиска пары ближайших точек на плоскости (см. [BS76]) мож- но найти в книгах [CLRS01] и [Мап89]. Вместо простого выбора точек из триангуляции Делоне в этих алгоритмах применяется метод "разделяй и властвуй". Родственные задачи. Kd-деревья (см. раздел 12.6), диаграммы Вороного (см. раз- дел 17.4). поиск в области (см. раздел 17.6). 17.6. Поиск в области Вход. Множество 5 из п точек в пространстве и область Q. Задача. Выяснить, какие точки множества 5 находятся в области Q (рис. 17.6). ВЫХОД Рис. 17.6. Поиск в области Обсуждение. Задачи поиска в области возникают в приложениях баз данных и геогра- фических информационных систем. Любой объект базы данных, имеющий d числовых полей (и описывающий, например, человека с его ростом, весом, уровнем дохода и т. д.), можно представить в виде точки в <7-мерном пространстве. Область запроса описывает область в пространстве, а наша задача состоит в поиске всех точек или вы- яснении количества точек в этой области. Например, запрос на поиск всех людей с го-
Глава 17. Вычислительная геометрия 597 довым доходом от 0 до 10 000 долларов, ростом между 185 и 215 см и весом от 25 до 60 кг определяет область, содержащую тощих людей с тощим кошельком. Уровень трудности поиска в области зависит от нескольких факторов. ♦ Велико чи количество выполняемых запросов? Самый простой подход— проверка каждой из п точек на попадание в многоугольник О. Этот метод прекрасно работает при небольшом количестве запросов. Алгоритмы для проверки вхождения точки в данный многоугольник рассматриваются в разделе 17.7, ♦ Какова форма многоугольника? Легче всего выполнять поиск в прямоугольниках, параллельных осям координат, т. к. проверка точки на вхождение в такую область сводится к проверке попадания каждой координаты в требуемый диапазон. Пример поиска в ортогональной области показан на рис. 17.6. При поиске в невыпуклом многоугольнике полезно разбить его на выпуклые облас- ти или, что еще лучше, треугольники и выполнить поиск в каждой из этих областей. Этот метод хорошо работает, потому что проверка точки на вхождение в выпуклый многоугольник является достаточно простой задачей. Алгоритмы разбиения невы- пуклого многоугольника на несколько выпуклых многоугольников рассматривают- ся в разделе ITU. ♦ Какова размерность пространства? Общий подход к поиску в области состоит в создании kd-дерева для входного набора точек (см. раздел 12.6). Затем выполняется обход в глубину этого kd-дерева, причем каждый узел дерева разворачивается толь- ко тогда, когда соответствующий прямоугольник пересекает область запроса. В случае очень больших или смещенных областей запроса может потребоваться об- ход всего дерева, но, вообще говоря, kd-деревья дают эффективное решение. Для двумерного пространства известны алгоритмы с более высокой производительно- стью в наихудшем случае, но на плоскости производительность kd-деревьев являет- ся вполне приемлемой. А в многомерных пространствах поиска они являются един- ственным возможным решением задачи. ♦ Возможны ли операции вставки и удаления? Красивый и практичный подход к ре- шению задачи поиска в области и ряда других геометрических задач поиска осно- ван на триангуляциях Делоне. В триангуляции Делоне ребра соединяют каждую точку р с близлежащими точками. Для поиска в области мы сначала выясняем ме- стоположение точки на плоскости (см. раздел 17.7), чтобы быстро найти треуголь- ник в интересующей нас области. Потом мы выполняем поиск в глубину вокруг ка- кой-либо вершины этого треугольника, разрежая пространство поиска в каждом случае, когда посещаем точку, расположенную слишком далеко, чтобы иметь неот- крытых соседей, представляющих для нас интерес. Эта процедура должна быть эф- фективной. т. к. общее количество посещенных точек приблизительно пропорцио- нально количеству точек в области запроса. Одним из достоинств этого подхода является легкость, с которой можно откоррек- тировать триангуляцию Делоне после вставки или удаления точки, используя для этого операции замены ребер. Подробности см. в разделе 17.3. ♦ Можно ли ограничиться подсчетом количества точек в области win требуется идентифицировать их? Для многих приложений достаточно только подсчитать ко-
598 Часть II Каталог алгоритмических задач пичество искомых точек в области, а не возвращать сами точки. Например, из базы данных, упомянутой в начале раздела, мы можем узнать, каких людей больше, ху- дых и бедных или толстых и богатых. Часто возникает необходимость найти самую плотную или самую разреженную область в пространстве поиска, и эту задачу мож- но решить с помощью подсчета точек в ней. Хорошая структура данных для эффективного поиска ответов на составные запросы поиска в области основана на упорядочении набора точек по признаку доминирова- ния. Говорят, что точках доминирует над точкой у, если точка у располагается сни- зу и слева от точки х. Пусть DOM(p) является функцией, которая подсчитывает количество точек во множестве 5, над которыми доминирует точка р. Количест- во точек т в ортогональном прямоугольнике, определяемом координатами -^min — % — Х’тач иут,п<у <Утах. вычисляется по такой формуле: ^б?Л/(х)пах, УтаХ) — /)ОЛ/(Х|Пах, Утт) DOA/(x'min, Утах) + /9ОЛ/(Хтт, Утт) Четвертый член вносит поправку на точки нижнего левого угла, которые были вы- чтены дважды. Чтобы эффективно отвечать на произвольные запросы по выяснению доминирова- ния, разделяем пространство на п~ прямоугольников, проводя горизонтальную и вертикальную прямые через каждую из п точек. В любом прямоугольнике набор доминируемых точек одинаков для каждой точки, поэтому их можно вычислить за- ранее для нижнего левого угла каждого прямоугольника, сохранить и возвращать в ответ на запрос для любой точки в данном прямоугольнике. Таким образом, поиск в области сводится к двоичному поиску и выполняется за время (?(lgn). К сожале- нию, эта структура данных имеет квадратичную сложность по памяти. Но эту идею можно использовать с kd-деревьями, чтобы создать структуру, расходующую па- мять более эффективно. Реализации. Библиотеки CGAL (www.cgal.org) и LEDA (см. раздел 19.1.1} использу- ют динамическую структуру данных на основе триангуляции Делоне для поддержки круглых, треугольных и ортогональных областей поиска. Обе библиотеки также со- держат реализации древовидных структур, которые поддерживают поиск в ортого- нальных областях за время O(k + lg+?), где п— сложность разбиения, к— количество точек в прямоугольной области. Библиотека ANN на языке C++ содержит реализации алгоритмов для точного и ап- проксимирующего поиска ближайшего соседа в пространствах произвольной размер- ности. Эти реализации дают хорошие результаты для поиска среди сотен тысяч точек в пространствах, имеющих до 20 измерений. Библиотека позволяет выполнять запросы поиска ближайшего соседа постоянного радиуса по всем L/’-HOpMaM расстояний, кото- рые можно использовать для аппроксимации запросов поиска круглых и прямоуголь- ных областей в нормах L1 и L1 соответственно. Библиотеку ANN можно загрузить по адресу http://vvww.cs.umd.edu/~mount/ANN/. Программа Ranger (см. [MS93]) представляет собой инструмент для визуализации и экспериментов с поиском ближайшего соседа и поиском в ортогональных областях в наборах данных с большим количеством измерений с использованием многомерных деревьев поиска. Программа поддерживает четыре разных типа структур данных: примитивные kd-деревья, медианные kd-деревья, неортогональные kd-деревья и
Гпава 17. Вычислительная геометрия 599 VP-деревья. Для каждой из этих структур данных программа поддерживает запросы в пространствах, имеющих до 25 измерений, в любой метрике Минковского. Программу можно загрузить из хранилища алгоритмов по адресу http://www.cs.sunysb.edu/ ~algorith. Примечания Хорошие описания структур данных с временем исполнения O(lgw + к) в наихудшем слу- чае для поиска в ортогональных областях (см. [Wil85]) представлены в книгах [dBvKOSOO] и [PS85], Там же можно найти описания kd-деревьев, применяемых при по- иске точек в прямоугольниках на плоскости. Производительность программ, использую- щих эти деревья, может быть очень низкой. Так. в работе [LW77] описывается двумерный экземпляр запроса, для которого потребовалось время Оу[п , чтобы выяснить, что прямо- угольник пустой. Задача значительно усложняется при поиске в неортогональных областях, т. е. в областях, не являющихся прямоугольниками со сторонами, параллельными осям координат. Поиск пересечений полуплоскостей имеет сложность по времени O(lgn), а по памяти— линей- ную (см. [CGL85]). В случае поиска в простых областях (таких как треугольник) нижние границы препятствуют созданию структур, эффективных для наихудших случаев. Обсуж- дение этой темы и обзор посвященных ей работ представлены в [Aga04], Родственные задачи. Kd-деревья (см. раздел 12.6), выяснение местоположения точки (см. раздел 17.7). 17.7. Местоположение точки Вход. Плоскость, разбитая на многоугольники, и точка q. Задача. Выяснить, какая область плоскости содержит точку q (рис. 17.7). Обсуждение. Выяснение местоположения точки на плоскости является фундаменталь- ной подзадачей вычислительной геометрии, обычно возникающей, как составная часть решения больших геометрических задач. В типичной полицейской диспетчерской
600 Часть II. Каталог алгоритмических задач службе город разбивается на несколько участков (районов). Имея карту участков и точку запроса (место преступления), диспетчер должен найти участок, содержащий данную точку. Это и есть задача выяснения местоположения точки на плоскости. Воз- можны следующие варианты задачи: ♦ Находится ли данная точка внутри многоугольника Р? В самом простом варианте задачи выяснения местоположения точки на плоскости участвуют только две облас- ти, внутри и вне многоугольника Р, и требуется выяснить, в какой из них находится данная точка. Для невыпуклых многоугольников, имеющих много узких выступов, поиск ответа может оказаться очень трудным. Процедура получения решения со- стоит в следующем. Из точки проводим отрезок, заканчивающийся далеко за преде- лами многоугольника, и подсчитываем количество ребер многоугольника, пересе- каемых этим отрезком. Точка будет находиться внутри многоугольника тогда и только тогда, когда это число нечетное. Когда отрезок проходит через вершину, а не ребро, ответ очевиден из контекста, т. к. мы подсчитываем количество пересечений границы многоугольника. Проверка всех п сторон многоугольника на пересечение с отрезком занимает время О(п). Для выпуклых многоугольников существуют более быстрые алгоритмы на основе двоичного поиска со временем исполнения ()(\gn). ♦ Сколько потребуется запросов? Выяснение местоположения точки внутри много- угольника всегда можно производить отдельно для каждой области в данном раз- биении. Однако выполнение большого количества таких запросов на одном и том же разбиении неэкономично. Намного лучшим решением будет создание поверх этого разбиения решеточной или древовидной структуры данных, позволяющей бы- стро попасть в нужную область. Такие структуры подробно рассматриваю гея далее в этом разделе. ♦ Какова сложность областей разбиения? Если области, на которые разбита плос- кость. являются произвольными многоугольниками, требуются довольно сложные проверки на вхождение точки в область. Выполнив предварительную триангуляцию всех многоугольников разбиения, мы сведем проверку на вхождение точки в об- ласть к проверке на вхождение точки в треугольник. Эти проверки можно сделать очень быстрыми и простыми за счет небольших затрат на сохранение имени много- угольника для каждого треугольника. Дополнительная выгода от такой триангуля- ции заключается в том. что чем меньше размер областей разбиения, тем выше про- изводительность решеточных или древовидных суперструктур. Но при триангуля- ции следует соблюдать определенную осторожность, чтобы избежать получения треугольников с очень острыми углами (см. раздел 17.3). ♦ Насколько регулярны размер и форма областей разбиения? Если все треугольники имеют приблизительно одинаковый размер и форму, можно использовать самый простой способ выяснения местоположения точки, при котором на область разбие- ния накладывается решетка размером k*k из горизонтальных и вертикальных пря- мых, идущих через одинаковые интервалы. Для каждой из К прямоугольных ячеек поддерживается список всех областей разбиения, которые хотя бы частично покры- ваются данной ячейкой. Для выяснения местоположения точки в таком сеточном файле выполняется двоичный поиск или поиск в хэш-таблице, чтобы найти ячейку, содержащую точку q, а потом выполняется поиск в каждой области, покрываемой ячейкой, чтобы выяснить, какая из них содержит данную точку.
Глава 17. Вычислительная геометрия 601 Производительность таких сеточных файлов может быть очень хорошей, при усло- вии. что каждая треугольная область перекрывает лишь небольшое количество пря- моугольников (тем самым минимизируя потребности в памяти), а каждый прямо- угольник перекрывает лишь небольшое количество треугольников (тем самым минимизируя время поиска). Производительность этого метода зависит от регуляр- ности областей разбиения. Определенной гибкости можно добиться, располагая горизонтальные линии решетки не на одинаковых расстояниях друг от друга, а в за- висимости от расположения областей разбиения. Рассматриваемый далее метод по- лос является развитием этой идеи и гарантирует эффективную временную слож- ность за счет квадратичной сложности по памяти. ♦ Какова размерность пространства? Для трех и более измерений наиболее подхо- дящим методом выяснения местоположения точки почти наверняка будет исполь- зование kd-дерева. Эти деревья могут оказаться лучшим решением, когда плоскость разбита на области, слишком нерегулярные, чтобы можно было применить сеточ- ные файлы. Kd-деревья (см. раздел 12.6) иерархически разбивают пространство на прямоуголь- ные блоки. В каждом узле дерева текущий прямоугольник разбивается на неболь- шое количество (обычно 2, 4 или 2‘‘ для d измерений) меньших прямоугольников. Прямоугольник, принадлежащий листу, помечается списком областей, содержа- щихся в нем хотя бы частично. Процесс выяснения местоположения точки начина- ется с корня дерева и движется вниз по дереву, пока не будет найден прямоуголь- ник, который содержит точку q. Когда поиск доходит до листа, проверяется каждая содержащаяся в нем подходящая область, чтобы выяснить, какая из них включает в себя точку ц. Так же, как и в случае с сеточными файлами, мы надеемся, что каж- дый лист будет содержать небольшое количество областей, и что каждая область не пересекает слишком много листовых прямоугольников. ♦ Близко ли целевая ячейка? Простым методом выяснения местоположения точки, который хорошо работает в пространствах, имеющих гораздо больше двух измере- ний, является обход. Начинаем поиск с произвольной точки р в произвольной ячей- ке. расположенной (предположительно) недалеко от точки q. Строим луч отр к q и находим сторону (грань) ячейки, пересекаемую этим лучом. В триангуляционных разбиениях такие запросы выполняются за постоянное время. Переходя к следующей ячейке сквозь эту грань, мы еще на шаг приблизимся к цели. Для достаточно регулярных <7-мерных разбиений ожидаемая длина пути будет О[пГ'). но линейной в наихудшем случае. Самым простым алгоритмом, гарантирующим время поиска O(lg„) в наихудшем слу- чае, является метод полос, в котором через каждую вершину проводятся горизонталь- ные прямые, создающие п+ 1 полос между ними. Поскольку полосы определяются горизонтальными прямыми, полосу, содержащую конкретную точку, можно найти, выполнив двоичный поиск по у-координате точки q. Поскольку никакая полоса не со- держит вершин, то область в полосе, содержащую точку q. можно найти с помощью двоичного поиска по ребрам, пересекающим эту полосу. Некоторая трудность заклю- чается в том, что для каждой полосы необходимо сопровождать двоичное дерево поис- ка. что в наихудшем случае дает сложность по памяти, равную О(п). Метод, расхо-
602 Часть II. Каталог алгоритмических задач дующий память более эффективно, основан на создании иерархической триангуляци- онной структуры на областях разбиения. Он тоже обеспечивает сложность по времени, равную O(lgn). и рассматривается в подразделе "Примечания". Эффективные методы вычислительной геометрии для наихудших случаев либо требу- ют большого объема памяти, либо очень сложны для реализации. Но для большинства приложений рекомендуется использовать kd-деревья. Реализации. Как библиотека CGAL (vvww.cgal.org), так и библиотека LEDA (см. раз- дел 19.1.1) содержат реализации на языке C++ самых разных алгоритмов разбиения плоскости. В библиотеке CGAL отдается предпочтение стратегии "переход и обход" (jump-and-walk), хотя также представлена реализация алгоритма поиска с логарифми- ческой сложностью для наихудшего случая. Библиотека LEDA содержит программы выяснения местоположения точки с использованием частично устойчивых (partially persistent) деревьев поиска с ожидаемым временем исполнения O(lg«). Библиотека ANN на языке C++ содержит реализации алгоритмов для точного и ап- проксимирующего поиска ближайшего соседа в пространствах произвольной размер- ности. Программы из этой библиотеки можно использовать для быстрого поиска точки на границе ближайшей ячейки, с которой следует начинать обход. Загрузить библиоте- ку можно по адресу http://www.cs.umd.edu/~mount/ANN/. Пакет Arrange на языке С предназначен для размещения многоугольников на плоско- сти или на сфере. Многоугольники могут быть вырожденными, и в таком случае работа сводится к размещению отрезков. В пакете используется рандомизированный инкре- ментальный алгоритм и поддерживается эффективное выяснение местоположения точки в разбиении. Пакет Arrange был разработан Майклом Голдвассером (Michael Goldwasser) и доступен на веб-сайте http://euler.slu.edu/~goldwasser/publications. В книгах [O'ROl] и [SR03] представлены процедуры проверки расположения точки в простом многоугольнике. Примечания Отличный обзор последних достижений в области решения задачи выяснения местополо- жения точки, как теоретических, так и практических, представлен в [Sno04], Очень под- робные описания детерминистических структур данных, используемых при выяснении местоположения точки на плоскости, содержатся в книгах [dBvKOSOO] и [PS85]. В работе [TV01] задача выяснения местоположения точки на плоскости используется в качестве учебного примера разработки алгоритмов вычислительной геометрии на языке Java. Экспериментальное исследование алгоритмов выяснения местоположения точки на плоскости изложено в работе [ЕКА84]. Лучшим был признан метод группирования, ана- логичный методу сеточного файла. Элегантный метод улучшения треугольников (см. [Kir83]) заключается в создании иерар- хической структуры триангуляционных разбиений поверх имеющегося разбиения таким образом, что каждый треугольник на данном уровне пересекает постоянное количество треугольников следующего уровня. Так как размер каждой триангуляции составляет оп- ределенную долю от размера следующей триангуляции, то общая сложность по памяти вычисляется суммированием геометрической прогрессии и, следовательно, является ли- нейной. Кроме этого, высота иерархической структуры равна O(lgw), что обеспечивает быстрое время исполнения запросов. Альтернативный алгоритм с такими же временными
Гпава 17. Вычислительная геометрия 603 пределами изложен в работе [EGS86]. Описанный ранее метод полос был разработан Добкином (Dobkin) и Липтоном (Lipton) (см. [DL76]) и представлен в книге [PS85]. Опи- сания алгоритмов, проверяющих вхождение точки в простой многоугольник, содержатся в работах [Hai94], [O'ROI ], [PS85] и [SR03]. В последнее время наблюдается интерес к динамическим структурам данных, которые поддерживают быстрое инкрементальное обновление разбиения плоскости (после вставки и удаления ребер и вершин), а также быстрое выяснение местоположения точки. Изучение этой области можно начать с [СТ92], а обновленный справочный материал можно найти в [Sno04]. Родственные задачи. Kd-деревья (см. раздел 12.6), диаграммы Вороного (см. раз- дел 17.4), поиск ближайшего соседа (см. раздел 17.5). 17.8. Выявление пересечений Вход. Множество S отрезков (прямых) /ь .... /„ или пара многоугольников (многогран- ников) Д, и Pi. Задача. Выяснить, какие отрезки пересекаются. Найти пересечение объектов Р\ и А (рис. 17.8). ВХОД ВЫХОД Рис. 17.8. Выявление пересечений Обсуждение. Выявление пересечения является фундаментальной задачей вычисли- тельной геометрии, имеющей много применений. Например, предположим, что мы моделируем здание в виртуальной реальности компьютерной игры. Иллюзия реально- сти исчезнет, как только виртуальный персонаж пройдет сквозь стену. Чтобы обеспе- чить соблюдение физических ограничений, мы должны немедленно обнаруживать пересечение многогранных моделей и информировать о нем игрока или ограничивать его действия. Еще одним применением выявления пересечений является контроль проектирования СБИС. Небольшой дефект конструкции, такой как пересечение двух дорожек, может
604 Часть II. Каталог алгоритмических задач вызвать короткое замыкание микросхемы, но такие ошибки можно с легкостью вы- явить до сдачи проекта в производство, используя программы для выявления пересе- чения отрезков. Вы должны ответить на следующие вопросы, возникающие в задаче выявления пере- сечений. ♦ Нужно вычислить .местоположение пересечения или достаточно лишь выявить сам факт его существования? Задача выявления пересечения решается значитель- но легче, чем вычисление его местоположения, и во многих случаях этого доста- точно. Для приложений виртуальной реальности важным может быть только сам факт столкновения со стеной, а не координаты точки, в которой это произойдет. ♦ Нужно выявить пересечение прямых или отрезков? Разница состоит в том, что лю- бые две непараллельные прямые пересекаются в одной и только одной точке. Все точки пересечения прямых можно вычислить за время О(уГ). сравнив каждую пару прямых. Как показано в разделе 17.15, создание конфигурации прямых предостав- ляет больше информации, чем просто поиск точек пересечения. Поиск всех точек пересечения п отрезков является значительно более трудной зада- чей. Даже проверка двух отрезков на пересечение оказывается не такой уж простой операцией (см. раздеч 1".1). Конечно, можно было бы явно проверить каждую пару отрезков и таким образом найти все пересечения за время (?(«'), но для случаев с небольшим количеством точек пересечения существуют более быстрые алгоритмы. ♦ Каково ожидаемое количество точек пересечения? При контроле проектирования СБИС мы ожидаем, что набор отрезков будет иметь небольшое количество точек пересечения, если, вообще, они найдутся. В этом случае нам нужен алгоритм, чув- ствительный к выводу, т. е. имеющий время исполнения, пропорциональное коли- честву точек пересечения. Такие чувствительные к выводу алгоритмы для выявления пересечений отрезков уже существуют. Время исполнения самого быстрого из них равно O(/zlgw + к), где к— количество пересечений. Эти алгоритмы основаны на методе заметающей пря- мой. ♦ Видна ли точка х из точки у? В некоторых случаях требуется узнать, можно ли в заполненном препятствиями пространстве видеть точку у из точки х. Эту задачу можно сформулировать в виде задачи выявления пересечения отрезков: пересекает ли отрезок между точками х и у какое-либо препятствие? Задачи выяснения види- мости возникают при планировании перемещений роботов (см. раздел 17.14) и при исключении скрытых поверхностей в компьютерной графике. ♦ Являются ли пересекающиеся объекты выпуклыми? Существуют очень хорошие алгоритмы для выявления пересечения многоугольников. В этой ситуации большую важность приобретает вопрос о выпуклости многоугольников. Пересечение выпук- лого п-угольника с выпуклым /л-утольником можно выявить за время О(п + т) с помощью алгоритма, основанного на методе заметающей прямой, описанного да- лее. Это возможно благодаря тому, что результат пересечения двух выпуклых мно- гоугольников— выпуклый многоугольник, содержащий, самое большее, п + т вершин.
Глава 17. Вычислительная геометрия 605 Однако пересечение двух невыпуклых многоугольников лишено этих достоинств. Рассмотрим пересечение двух "расчесок", показанное на рис. 17.8. Как видно из ри- сунка, пересечение невыпуклых многоугольников может быть фрагментированным. Задача поиска пересечения многогранников сложнее задачи поиска пересечения многоугольников, т. к. многогранники могут пересекаться даже тогда, когда их реб- ра не пересекаются. В качестве примера такой ситуации рассмотрим иголку, прон- зающую внутреннюю область грани. Тем не менее, при поиске пересечения как многоугольников, так и многогранников возникаю! одинаковые вопросы. ♦ Выполняется ли многократный поиск пересечений с одними и теми же основными объектами? В примере с прохождением сквозь стену в виртуальном здании непод- вижные объекты не изменяются от эпизода к эпизоду. Движется только виртуаль- ный персонаж, а пересечения происходят редко. В таких случаях типичным подходом будет аппроксимация объектов более просты- ми охватывающими объектами, например, параллелепипедами. Пересечение двух таких охватывающих объектов означает, что содержащиеся в них объекты могут пересекаться, и для выяснения этого вопроса требуется дополнительная обработка. Но проверка на пересечение простых параллелепипедов происходит намного эф- фективнее, чем более сложных объектов, поэтому в случае небольшого количества пересечений этот метод выгоден. Возможны различные варианты этой схемы, но основная идея позволяет заметно повысить производительность в сложной обста- новке. Для эффективного поиска пересечений нескольких отрезков или пересечений и объ- единений двух многоугольников можно использовать алгоритмы, основанные на мето- де заметающей прямой. Эти алгоритмы отслеживают важные события в процессе пе- ремещения вертикальной прямой слева направо по набору входных данных. В самом левом положении прямая не пересекает никаких объектов, но в процессе ее движения вправо происходит следующая последовательность событий: ♦ вставка. Найден левый конец отрезка, и этот отрезок может пересекать какой-либо другой отрезок прямой; ♦ удаление. Найден правый конец отрезка. Это означает, что заметающая прямая пол- ностью прошла данный отрезок, и его можно исключить из дальнейшего рассмот- рения; ♦ пересечение. Если для активных отрезков, пересекаемых заметающей прямой, под- держивается упорядочивание по расположению (сверху вниз), то следующее пере- сечение должно произойти между парой соседних отрезков. После такого пересече- ния относительный порядок этих двух отрезков меняется на обратный. Для слежения за процессом требуются две структуры данных. Под будущие события отводится очередь событий или очередь с приоритетами, упорядоченная по х-ко- ординате всех возможных будущих событий: вставок, удалений и пересечений. Базо- вые реализации очередей с приоритетами рассматриваются в разделе 12.2. Настоящее представлено горизонтом — упорядоченным списком отрезков, пересекающих теку- щую позицию заметающей прямой. Для сопровождения горизонта можно использовать любую словарную структуру данных, например, сбалансированное дерево.
606 Часть II Каталог алгоритмических задач Чтобы адаптировать этот подход для поиска пересечения или объединения много- угольников, изменяется обработка трех основных типов событий. Этот алгоритм мож- но значительно упростить для выявления пересечения пар выпуклых многоугольников, т. к., во-первых, заметающая прямая пересекает, самое большее, четыре ребра много- угольников, что делает горизонт ненужным, и, во-вторых, не требуется сортировать очередь событий, т. к. мы можем начать обработку с самой левой вершины каждого многоугольника и двигаться направо в соответствии с естественным порядком вершин многоугольника. Реализации. Как библиотека LEDA (см. раздел 19.1.1). так и библиотека CGAL (www.cgal.org) содержит реализацию на языке C++ самых разных алгоритмов для вы- явления пересечений отрезков и многоугольников. В частности, обе библиотеки пре- доставляют реализацию алгоритма Бентли-Оттманна (Bentley-Ottmann), основанного на методе заметающей прямой (см. [ВО79]), для поиска всех k точек пересечения между п отрезками прямых на плоскости за время О((н + (c)lg«). Устойчивая программа на языке С для поиска пересечения двух выпуклых много- угольников представлена в книге [O'RO 1 ]. Подробности см. в разделе 19.1.10. Группа GAMMA из университета Северной Каролины разработала несколько эффек- тивных библиотек для обнаружения столкновений, из которых самой последней явля- ется библиотека SW1FT++ (см. [EL01]). Эта библиотека поддерживает выявление пере- сечений, вычисление приблизительных и точных расстояний между объектами, а также выявление контакта между парой объектов в сценах, состоящих из жестких много- гранных моделей. Подробная информация обо всех этих библиотеках, включая условия загрузки и использования, доступна на веб-сайте littp://www.cs.unc.edu/~geom/collide/. Поиск взаимного пересечения набора полупространств является частным случаем по- иска пересечения выпуклых оболочек. Лучшей программой для работы с выпуклыми оболочками в многомерных пространствах является Qhull (см. [BDH97]). Эта программа широко используется в научных приложениях и доступна на веб-сайте http://www.qhull.org/. Примечания Отличный обзор алгоритмов поиска пересечений таких геометрических объектов, как от- резки, многоугольники и многогранники представлен в работе [Мои04]. Книги [dBvKOSOO], [CLRSOl] и [PS85] содержат главы, в которых обсуждается задача поиска пересечений геометрических объектов. Хорошее обсуждение частного случая поиска пе- ресечений и объединений прямоугольников, ориентированных вдоль осей координат (за- дача, которая часто возникает при разработке СБИС), представлено в книге [PS85]. Алгоритм поиска пересечения отрезков с временем исполнения O(n\gn + к) приведен в ра- боте [СЕ92]. Всесторонний обзор более простых, рандомизированных алгоритмов с таки- ми же временными границами представлен в работе [Mul94J. Алгоритмы и программное обеспечение для обнаружения столкновений обсуждаются в работе [LM04], Родственные задачи. Конфигурация прямых (см. раздел 17.15), планирование пере- мещений (см. раздел 17.14).
Гпава 17 Вычислительная геометрия 607 17.9. Разложение по контейнерам Вход. Набор из п объектов размером d\, .... d„. Набор из т контейнеров емкостью С|, сп. Задача. Уложить все объекты в контейнеры, используя как можно меньшее количество контейнеров (рис. 17.9). ВХОД ВЫХОД Рис. 17.9. Разложение по контейнерам Обсуждение. Задача разложения по контейнерам возникает в разных приложениях упаковки и производства. Для примера рассмотрим раскройку листового железа под детали или раскройку ткани для шитья одежды. Для минимизации отходов мы хотим разместить детали или выкройки так, чтобы использовать как можно меньшее количе- ство листов железа или рулонов материи. Задача выяснения, какую деталь разместить на каком листе и в каком месте, является вариантом задачи разложения по контейне- рам, называющейся задачей раскроя (cutting stock problem). После изготовления дета- лей возникает другая задача упаковки— как погрузить ящики с деталями на грузови- ки, чтобы минимизировать необходимое количество грузовиков. Даже самые простые на первый взгляд задачи разложения по контейнерам являются NP-полными (см. обсуждение задачи разбиения множества целых чисел в /зазде- ле 13.10). Поэтому нам приходится использовать эвристические подходы, а не алго- ритмы, выдающие оптимальное решение в наихудшем случае. К счастью, для боль- шинства задач разложения по контейнерам обычно хорошо подходят сравнительно простые эвристические методы. Более того, многие приложения имеют специфические ограничивающие условия, которые не позволяют использовать сложные алгоритмы решения задачи разложения по контейнерам. Выбор эвристического метода для реше- ния конкретной задачи будет зависеть от следующих факторов: ♦ формы и размера объектов. Характер задачи разложения по контейнерам в боль- шой степени зависит от формы пакуемых объектов. Собрать пазл (картинку-
608 Часть II. Каталог алгоритмических задач головоломку) гораздо сложнее, чем уложить квадратики на прямоугольном поле. В задаче одномерной задаче разложения по контейнерам размер каждого объекта указывается в виде целого числа. Эта задача эквивалентна задаче упаковки ящиков одинаковой ширины в контейнер такой же ширины и является частным случаем за- дачи о рюкзаке (см. раздел 13.10). Когда все объекты имеют одинаковый размер и форму, последовательное заполне- ние каждого ряда дает приемлемую, но не обязательно оптимальную упаковку. По- пробуйте заполнить квадрат размером 3*3 прямоугольниками размером 2х 1. Укла- дывая прямоугольники, ориентированные только в одном направлении, можно раз- местить только три из них, а если их ориентировать в двух направлениях, то поместится четыре прямоугольника; ♦ ограничения на ориентацию и размещение объектов. На практике у многих упако- вочных коробок обозначен верх (что предписывает определенную ориентацию ко- робки) или присутствует надпись "не складывать в штабель" (означающая, что ко- робка может находиться только на самом верху штабеля). Такие условия ограничи- вают нашу свободу действий при упаковке и увеличивают количество грузовиков, необходимых для транспортировки товаров. Большинство транспортных компаний решают эту задачу, не обращая внимания на подобные маркировки. Несомненно, любая задача становится намного легче, если не беспокоиться о последствиях; ♦ статичность или динамичность задачи. Известен ли нам весь набор объектов для упаковки в начале работы (статическая задача) или же мы получаем их по одному и должны укладывать их по мере поступления (динамическая задача)? Упаковку можно выполнить гораздо лучше, если есть возможность планирования. Например, имеет смысл расположить объекты в таком порядке, который позволит Осуществить более эффективную упаковку (т. е. отсортировать их по убыванию размера). В стандартных эвристических методах статического разложения по контейнерам объ- екты упорядочиваются по размеру или форме, а потом укладываются в контейнеры. Типичные правила выбора контейнера: ♦ выбираем первый или самый левый контейнер, в который может поместиться объ- ект; ♦ выбираем контейнер с наибольшим свободным объемом; ♦ выбираем контейнер с наименьшим свободным объемом, достаточным для разме- щения объекта; ♦ выбираем случайный контейнер. Аналитические и экспериментальные результаты показывают, что самым лучшим эв- ристическим методом разложения по контейнерам является "первый подходящий кон- тейнер в порядке убывания размера объектов". Сортируем объекты по убыванию раз- мера. Вкладываем объекты по одному в контейнер, в котором имеется достаточно мес- та для очередного объекта. Если в контейнере больше нет места, то переходим к следующему контейнеру. В случае одномерного разложения по контейнерам при таком подходе потраченное количество контейнеров никогда не превысит действительно не- обходимое больше чем на 22%, причем обычно этот показатель намного лучше. Такой метод интуитивно привлекателен, т. к. мы укладываем большие объекты в первую оче- редь и надеемся, что удастся уложить меньшие объекты в оставшееся место.
Гпава 17 Вычислительная геометрия 609 Этот алгоритм легко реализуется и имеет время исполнения O(n\gn + bri), где b< min(n, т)— количество фактически использованных контейнеров. Просто для каждого объекта выполняем перебор контейнеров за линейное время. Возможен более быстрый алгоритм с временем исполнения O(n\gn), использующий двоичное дерево для отслеживания свободного места, оставшегося в каждом контейнере. Чтобы удовлетворить специфическим ограничивающим условиям, можно разнообра- зить порядок размещения объектов. Например, ящики с маркировкой "не складывать в штабель" следует размещать в последнюю очередь (возможно, предварительно искус- ственно понизив высоту контейнеров, чтобы оставить достаточно свободного места), а ящики с маркировкой верха— в первую очередь (чтобы иметь больше свободы при укладывании ящиков сверху). Укладывать ящики легче, чем объекты произвольной геометрической формы. Это на- столько легче, что общее правило укладки произвольных объектов состоит в предвари- тельной упаковке в индивидуальные ящики, после которой уже выполняется укладка ящиков в целевые контейнеры. Найти охватывающий прямоугольник для объекта- многоугольника не составляет труда: просто находим верхнюю, нижнюю, левую и пра- вую касательные, идущие в заданном направлении. Выяснить ориентацию и миними- зировать площадь (объем) такого прямоугольника (ящика) намного труднее, но все- таки возможно как на плоскости, так и в трехмерном пространстве (см. [O'R.85]). В случае невыпуклых объектов большой объем полезного пространства может ока- заться неиспользованным из-за наличия пустот, получившихся после помещения дета- ли в ящик. Одно из решений этой проблемы — найти наибольший пустой прямоуголь- ник внутри каждого размещенного объекта и. если этот прямоугольник достаточно ве- лик, поместить в него другие объекты. Реализации. Коллекцию реализаций алгоритмов на языке FORTRAN для разных вер- сий задачи о рюкзаке можно загрузить с веб-сайта http://www.or.deis.unibo.it/kp.html. Там же доступна электронная версия книги [МТ90а]. Хорошо организованную коллекцию реализаций алгоритмов на языке С для решения разных видов задачи о рюкзаке и родственных ей задач, таких как разложение по кон- тейнерам и загрузка контейнера, можно найти на веб-сайте http://www.diku.dk/ -pisinger/codes.html. Первым шагом упаковки объектов произвольной формы в контейнер является их раз- мещение по индивидуальным прямоугольным контейнерам минимального объема. Этот алгоритм имеет почти линейное время исполнения (см. [ВНО!]). Примечания Обзоры литературы, посвяшенной задаче разложения по контейнерам и задаче раскроя, представлены в работах [CFC94], [CGJ96] и [LMM02], Самым свежим справочником по различным вариантам задачи о рюкзаке является книга [К.РР04]. Экспериментальные ре- зультаты эвристических алгоритмов решения задачи разложения по контейнерам изложе- ны в работах [BJLM83] и ]МТ87]. Существуют эффективные алгоритмы поиска наибольшего пустого прямоугольника в многоугольнике (см. [DMR97]) и наборе точек (см. [CDL86]). Упаковка шаров является важным и хорошо изученным частным случаем разложения по контейнерам. Общеизвестна "гипотеза Кеплера", касающаяся поиска наиболее плотной 20 Зак. 3741
610 Часть II. Каталог алгоритмических задач упаковки единичных трехмерных шаров. Справедливость этой гипотезы была доказана в 1998 г. Хэйлзом (Hales) и Фергюсоном (Ferguson). Доказательство представлено в работе [Szp03], Самым лучшим справочником по задаче упаковки шаров и родственным ей зада- чам является книга [CS93]. Миленкович (Milenkovic) много работал над двумерной задачей разложения по контейне- рам для швейной промышленности, т. е. над минимизацией количества материала, необ- ходимого для изготовления предметов одежды. Отчеты об этой работе содержатся в кни- гах [DM97] и [МП97]. Родственные задачи. Задача о рюкзаке (см. раздел 13.10), задача упаковки множества (см. раздел 18.2). 17.10. Преобразование к срединной оси Вход. Многоугольник или многогранник Р. Задача. Найти набор точек внутри объекта Р, имеющих более чем одну ближайшую точку на границе Р (рис. 17.10). ВХОД ВЫХОД Рис. 17.10. Преобразование к срединной оси Обсуждение. Преобразование к срединной оси применимо для "утончения" много- угольников или. другими словами, поиска скелета. Нашей целью является простое, устойчивое представление формы многоугольника. На рис. 17.10 "тонкие" версии букв А и В отражают их форму и будут устойчивы к изменению толщины линии или добав- лению декоративных элементов, таких как засечки. Скелет также представляет собой центральное множество данной фигуры, и это свойство может быть использовано в других приложениях, таких как восстановление формы и планирование перемещений. Результатом преобразования к срединной оси многоугольника является дерево, что позволяет использовать динамическое программирование для вычисления "расстояния редактирования" между скелетом известной модели и скелетом неизвестного объекта. Когда два скелета достаточно близки друг к другу, неизвестный объект классифициру- ется как экземпляр модели. Этот метод применяется в области компьютерного зрения и оптического распознавания текста. Скелетом многоугольника с отверстиями (как буква А или В) является не дерево, а вложенный планарный граф, но с ним тоже легко ра- ботать.
Гпава 17. Вычислительная геометрия 611 Существует два разных подхода к выполнению преобразований к срединной оси. в за- висимости от того, имеем ли мы на входе произвольные геометрические точки или растровые изображения: ♦ геометрические данные. Вспомним, что диаграмма Вороного для множества точек 5(см.раздел 17.4) разбивает плоскость на области вокруг каждой точки s, е Этаким образом, что все точки в области вокруг точки st находятся ближе к ней, чем к лю- бой другой точке из множества 5. Аналогичным образом, диаграмма Вороного для множества L отрезков разбивает плоскость на области вокруг каждого отрезка 1 е L таким образом, что все точки внутри области вокруг точки /, находятся ближе к не- му, чем к любому другому отрезку из множества!. Любой многоугольник определяется набором отрезков таких, что отрезок I, имеет общую вершину с отрезком /,+ ). Преобразование к срединной оси многоугольника Р сводится к получению той части диаграммы Вороного для отрезков, которая на- ходится внутри этого многоугольника. Таким образом, для утончения многоуголь- ника подойдет любая программа создания диаграмм Вороного для отрезков. Прямолинейным скелетом называется структура, родственная срединной оси мно- гоугольника, за исключением того, что биссектрисы равноудалены не от его ребер, а от вспомогательных линий этих ребер. Для выпуклых многоугольников прямоли- нейный скелет, срединная ось и диаграмма Вороного идентичны, но в скелете об- щего вида биссектрисы могут не проходить по центру многоугольника. Прямоли- нейный скелет похож на результат преобразования к срединной оси, но его легче искать с помощью компьютера. В частности, все ребра прямолинейного скелета яв- ляются многоугольными; ♦ изображения. Оцифрованные изображения можно рассматривать как множества точек, расположенных на узлах целочисленной решетки. Таким образом, мы можем извлечь из изображения многоугольник, представляющий объект, и обработать его с помощью ранее описанных алгоритмов вычислительной геометрии. Но внутрен- ние вершины скелета, скорее всего, не будут располагаться на узлах решетки. При- менение алгоритмов вычислительной геометрии к задачам обработки изображений часто приводит к неудаче, потому что изображения состоят из отдельных пикселов и не являются непрерывными. В простейшем подходе к созданию скелета пиксельного объекта применяется метод "brush fire" (лесной пожар). Представьте себе огонь, охвативший все ребра много- угольника и продвигающийся с постоянной скоростью внутрь многоугольника. Скелет образуется точками, в которых сталкиваются две или более стены огня. Та- кой алгоритм обходит все граничные пикселы объекта, идентифицирует вершины, принадлежащие скелету, удаляет остальную часть границы и повторяет процесс. Алгоритм прекращает работу, когда все пикселы являются крайними, и возвращает объект толщиной всего в один или два пиксела. При правильной реализации время исполнения этого алгоритма будет линейным по отношению к количеству пикселов в изображении. Алгоритмы, которые манипулируют непосредственно пикселами, обычно легко поддаются реализации, по той причине, что в них не используются сложные струк- туры данных. Но при подходах, основанных на обработке пикселов, результаты по-
612 Часть II. Каталог алгоритмических задач лучаются не вполне корректными. Например, скелет многоугольника не всегда бу- дет деревом и не обязательно будет связным, а точки скелета могут оказаться "не совсем" равноудаленными от двух граничных ребер. Когда вы пытаетесь применять методы непрерывной геометрии в дискретном мире, у вас нет возможности решить задачу до конца, и с этим нужно смириться. Реализации. Библиотека CGAL (www.cgal.org) содержит пакет процедур для вычис- ления прямолинейного скелета многоугольника Р. Библиотека также содержит про- цедуры создания смещенных контуров, определяющих области внутри многоугольни- ка Р. точки которых находятся, по меньшей мере, на расстоянии d от границ много- угольника. Программа VRONI (см. [HelOl]) является эффективным средством создания диаграмм Вороного’на плоскости для отрезков, точек и дуг. Программа может с легкостью вы- полнять преобразования к срединной оси многоугольников, т. к. она поддерживает создание диаграмм Вороного для произвольных отрезков. Программа была протести- рована на огромном количестве искусственно созданных и реальных наборов данных, причем некоторые из них содержали свыше миллиона вершин. Дополнительную ин- формацию можно найти на веб-сайте разработчика http://www.cosy.sbg.ac.at/ ~held/projects/vroni/vroni.html. Другие программы создания диаграмм Вороного рас- сматриваются в разделе 17.4. Программы для реконструкции или интерполирования облаков точек часто основаны на преобразованиях к срединной оси. Программное обеспечение Сосопе выполняет приблизительные преобразования к срединной оси многогранной поверхности, интер- полируя точки в Е3. Дополнительную информацию можно найти на веб-сайте разра- ботчиков http://www.cse.ohio-state.edu/~tamaldey/cocone.html. Теоретические основы программы Сосопе излагаются в работе [Dey06]. Программа Powercrust (см. [АСКО 1а] и [ACKOlb]) выполняет дискретное приближенное преобразование к срединной оси, после чего реконструирует исходную поверхность на основе результатов этого преоб- разования. При достаточной плотности точек выборки алгоритм гарантированно вы- полняет геометрически и топологически правильную аппроксимацию поверхности. Примечания Всесторонние обзоры методов утончения в обработке изображений содержатся в работах [LLS92] и [Ogn93]. Преобразование к срединной оси было впервые использовано для изу- чения схожести фигур в биологии (см. [В1и67]). Применение преобразований к срединной оси в области распознавания образов рассматривается в работе [DHSOO]. Преобразование к срединной оси является фундаментальной операцией для алгоритма Powercrust, который воссоздает поверхность по точкам выборки (см. [ACKOla] и [ACKOlb]). Обсуждение пре- образований к срединной оси можно найти в книгах [dBvKOSOO], [O'ROl и [Pav82]. Срединную ось произвольного «-угольника можно вычислить за время O(Hlgw) (см. [Lee82]), хотя для выпуклых многоугольников существуют алгоритмы с линейным временем исполнения (см. [AGSS89]). Алгоритм, выполняющий преобразование к сре- динной оси с временем г/сполнения O(«lg«), представлен в работе [Kir79]. Прямолинейные скелеты обсуждаются в работе [AAAG95], а алгоритм с временем испол- нения меньше квадратичного — в работе [ЕЕ99]. Интересное применение прямолинейных скелетов для создания крыш виртуальных зданий описано в работе [LD03] .
Глава 17 Вычислительная геометрия 613 Родственные задачи. Диаграммы Вороного (см. раздел 17.4), сумма Минковского (см. раздел 17.16). 17.11. Разбиение многоугольника на части Вход. Многоугольник или многогранник Р. Задача. Разбить объект Р на небольшое количество простых (обычно выпуклых) час- тей (рис. 17.11). Обсуячденне. Разбиение многоугольников на части является важным шагом предвари- тельной обработки во многих алгоритмах вычислительной геометрии, т. к. на выпук- лых объектах геометрические задачи решаются легче, чем на невыпуклых. Работать с небольшим количеством выпуклых фрагментов проще, чем с одним невыпуклым многоугольником. В зависимости от конкретного приложения могут возникать разные варианты задачи разбиения многоугольника. Чтобы распознать их. ответьте на следующие вопросы. ♦ Нужно ли, чтобы все фрагменты разбиения были треугольниками? Триангуляция является самой главной из задач разбиения многоугольников, поскольку в ней об- ластями разбиения являются треугольники. Треугольник имеет всего лишь три сто- роны и является выпуклым, что делает его простейшим возможным многоугольни- ком. Любое триангуляционное разбиение п-вершинного многоугольника содержит ровно п-2 треугольника. Поэтому триангуляция не подходит для тех случаев, когда тре- буется разбиение на небольшое количество выпуклых частей. Критерием "хорошей" триангуляции является не количество треугольников, а их внешний вид. Триангуля- ция обсуждается в разделе 17.3. ♦ Требуется покрытие или разбиение многоугольника? Под разбиением многоуголь- ника подразумевается разделение его внутренней области на части, не перекры-
614 Часть II. Каталог алгоритмических задач вающие друг друга. А покрытие означает, что части многоугольника могут пере- крываться. Оба способа могут оказаться полезными в разных ситуациях. Так. при декомпозиции сложного многоугольника во время подготовки к поиску в области (см. раздел 17.6), нам требуется разбиение, чтобы каждая искомая точка находилась ровно в одной области разбиения. А при декомпозиции многоугольника с целью его закрашивания будет достаточно покрытия, поскольку двойная заливка области цве- том не влечет за собой никаких проблем. Мы будем рассматривать разбиение, т. к. его проще выполнить, и оно приемлемо для любого приложения, где требуется по- крытие. Единственный недостаток этого подхода состоит в том, что разбиения мо- гут потребовать больше памяти, чем покрытия. ♦ Можно ли добавлять дополнительные вершины? Можем ли мы добавлять в много- угольник вершины Штейнера (разделяя ребра или вставляя внутренние точки) или нам разрешено только соединять ребрами две существующие вершины? В первом случае мы получим разбиение с меньшим количеством областей, но будем вынуж- дены использовать более сложные алгоритмы обработки и, возможно, получим ме- нее аккуратные результаты. Существует простой и эффективный эвристический алгоритм Хертеля-Мельхорна для разбиения многоугольников на выпуклые области с помощью диагоналей. Алгоритм сначала выполняет произвольную триангуляцию многоугольника, а потом убирает все диагонали, после удаления которых остаются только выпуклые области. Удаление диа- гонали создает невыпуклую область только тогда, когда в результате получается внут- ренний угол, превышающий 180°. Выяснить, появится ли такой угол, можно за посто- янное время, рассмотрев диагонали и стороны многоугольника, окружающие удаляе- мую диагональ. Полученное количество выпуклых областей никогда не превысит минимально возможное их количество более, чем в четыре раза. Если минимизация количества областей разбиения не является вашей целью, то я ре- комендую использовать этот эвристический подход. Экспериментируя с разными вари- антами триангуляции и разным порядком удаления диагоналей, вы, возможно, сумеете получить более качественные разбиения. С помощью динамического программирования можно выяснить, чему равно мини- мальное количество диагоналей, использованных в декомпозиции. Самая простая реа- лизация, которая отслеживает количество областей для всех О(п~) частей многоуголь- ника, разделенных ребром, имеет время исполнения О(п). Более быстрые алгоритмы используют более сложные структуры данных и имеют время исполнения О(п + r’min(H. п)), где г— количество вершин с внутренним углом, превышающим 180°. Существует алгоритм с временем исполнения О(п), который еще больше умень- шает количество областей разбиения, добавляя внутренние вершины. Но этот алгоритм сложен, и его трудно реализовать. В другом варианте задачи многоугольник разбивается на монотонные области. Вер- шины ^монотонного многоугольника можно разделить на две цепочки таким образом, что любая горизонтальная линия будет пересекать любую из этих цепочек не более одного раза. Реализации. Многие программы триангуляции начинают работу с трапецевидного или монотонного разбиения многоугольника. Кроме этого, триангуляция является про-
Гпава 17. Вычислительная геометрия 615 стейшим способом разбиения многоугольника на выпуклые области. В поисках от- правной точки в выборе подходящего кода обратитесь к реализациям, упомянутым в разделе 17.3. Библиотека CGAL (www.cgal.org) содержит набор процедур для разбиения много- угольников. включающий в себя реализации эвристического алгоритма Хертеля- Мельхорна для разбиения многоугольника на выпуклые области, алгоритма динамиче- ского программирования с временем исполнения О(п) для поиска оптимального раз- биения на выпуклые области и эвристического алгоритма на основе метода заметаю- щей прямой с временем исполнения C>(r?logA?) для разбиения на монотонные много- угольники. Для решения задач триангуляции особенно хорошо подходит пакет GEOMPACK (http://members.shaw.ca/bjoe/), состоящий из набора процедур на языке FORTRAN 77. Пакет позволяет выполнять триангуляцию Делоне и разбиение многоугольников и многогранников на выпуклые области, а также триангуляцию Делоне в пространстве произвольной размерности. Примечания Сравнительно недавние обзоры содержатся в работах [KeiOO] и [OS04]. В книге [KS85] дается отличный обзор существующего материала по разбиению и покрытию много- угольников. Описание эвристического алгоритма Хертеля-Мельхорна (см. [НМ83]) можно найти в книге [O'ROl], Алгоритм динамического программирования для минимального разбиения на выпуклые области с временем исполнения O(n + r"min(r‘, л)) был предложен в работе [KS02]. Алгоритм с временем исполнения O(rJ + и) для минимизации количества выпуклых областей разбиения посредством вставки точек Штейнера представлен в книге [CD85]. В статье [LA06] содержится эффективный эвристический алгоритм с временем исполнения О(пг) для разложения многоугольников с внутренними пустотами на "почти выпуклые" многоугольники, а впоследствии этот алгоритм был обобщен для работы с многогранниками. С задачей покрытия многоугольника связана интересная задача о картинной галерее, в ко- торой требуется разместить в данном многоугольнике минимальное количество охранни- ков таким образом, чтобы каждая внутренняя точка этого многоугольника просматрива- лась, по крайней мере, одним охранником. Это соответствует покрытию данного много- угольника минимальным количеством звездообразных многоугольников. Прекрасной книгой, в которой представлена задача о картинной галерее и ее многие варианты, являет- ся [O'R87]. К сожалению, эта книга больше не издавалась. Родственные задачи. Триангуляция (см. раздел 17.3), покрытие множества (см. раз- дел 18.1). 17.12. Упрощение многоугольников Вход. Многоугольник или многограннику, имеющий л вершин. Задача. Найти многоугольник или многогранник р', имеющий только л' вершин, наи- более близкий по форме к исходному объекту р (рис. 17.12). Обсуждение. Задача упрощения многоугольников имеет, в основном, два приложения. Первое касается удаления шума из фигуры, полученной, например, сканированием
616 Часть II. Каталог алгоритмических задач изображения объекта. Упрощение ее позволит удалить шум и восстановить первона- чальный объект. А второе относится к сжатию данных, когда требуется избавиться от незначительных деталей большого и сложного объекта, по возможности сохранив его первоначальный вид. Это может быть особенно полезным в области компьютерной графики, поскольку меньшая модель отображается на мониторе значительно быстрее. ВХОД ВЫХОД Рис. 17.12. Упрощение многоугольников При решении задачи упрощения многоугольников возникаег несколько вопросов. ♦ Нужна чи выпуклая оболочка? Самым простым решением будет выпуклая оболочка вершин объекта (см. раздел 17.2\ Выпуклая оболочка многоугольника удаляет все его неровности и хорошо подходит для такой задачи, как упрощение перемещений робота. Однако использование выпуклой оболочки в системе оптического распо- знавания текста недопустимо, т. к. вогнутости символов являются их важнейшим свойством. Например, выпуклая оболочка буквы "X" идентична выпуклой оболочке буквы "Н", т. к. обе оболочки— просто прямоугольники. Другая проблема заклю- чается в том. что выпуклая оболочка выпуклого многоугольника никак его не уп- рощает. ♦ Можно.чи вставлять точки или разрешается только удалять их? Типичной целью упрощения объекта является по возможности точное его представление с использо- ванием заданного количества вершин. При самом простом подходе выполняются локальные модификации границы для уменьшения количества вершин. Например, если три последовательные вершины образуют небольшой треугольник, то цен- тральную вершину можно удалить и заменить два ребра одним, не внося при этом значительных искажений в многоугольник. Однако подход, при котором вершины только удаляются, быстро изменит форму многоугольника до неузнаваемости. Более устойчивые эвристические методы пере- мещают вершины, чтобы закрыть промежутки, возникающие после удаления вер- шин. Такой подход типа "разделить и объединить" иногда позволяет добиться желаемого, хотя ничего не гарантирует. Получение хороших результатов гораздо вероятнее при использовании алгоритма Дугласа-Пекера. описанного далее. ♦ Может ли получившийся многоугольник иметь самопересечения? Серьезным не- достатком инкрементальных процедур является то. что они не обеспечивают полу- чение простых многоугольников, т. е. многоугольников, не содержащих самопере- сечений. Вследствие этого, "упрощенные" многоугольники могут иметь серьезные
Глава 17. Вычислительная геометрия 617 дефекты, которые вызовут проблемы на последующих этапах обработки. Если важ- но получить простой многоугольник, то все его отрезки нужно проверить на попар- ное пересечение (см. раздел 17.8). Подход к упрощению многоугольников, который гарантирует простую аппроксима- цию, основан на поиске путей, состоящих из минимального количества отрезков. Реберной длиной пути между точками х и t называется количество отрезков в этом пути. Реберная длина прямого пути равна единице, а в общем случае она на единицу превышает количество поворотов в пути. В помещении с препятствиями реберная длина пути между точками s и t определяется минимальной реберной длиной по всем путям между этими точками. Подход к упрощению многоугольника, основанный на вычислении реберной длины, заключается в "утолщении" границы многоугольника на некоторую приемлемую величину е, в результате чего многоугольник оказывается в своеобразном канале. Замкнутый путь с минимальной реберной длиной в этом канале представляет про- стейший многоугольник, границы которого не отличаются от границ исходного больше, чем на е. Легко вычисляемая аппроксимация реберной длины сводит задачу к поиску в ширину. В канале размещается дискретный набор возможных точек по- ворота. после чего пары точек, находящихся в прямой видимости, соединяются реб- рами. ♦ Требуется очистить изображение от шума (а не упростить многоугольник)? При общепринятом подходе к очистке изображения от шума выполняется преобразова- ние Фурье на этом изображении, из него отфильтровываются высокочастотные элементы, а потом выполняется обратное преобразование, восстанавливающее изо- бражение. Подробную информацию о быстром преобразовании Фурье см. в раз- деле 13.11. Вместо того, чтобы пытаться упростить сложный многоугольник, алгоритм Дутласа- Пекера для упрощения многоугольников находит примитивную аппроксимацию, а по- том стремится улучшить ее. Для начала выбираем две вершины v, и v2 многоугольника Р, а вырожденный многоугольник vb v2 и V| рассматриваем в качестве простой аппрок- симации Р' Проходим через все вершины многоугольника Р и выбираем самую даль- нюю от соответствующего ребра многоугольника Р'. Вставка этой вершины добавляет треугольник к многоугольнику Р', минимизируя максимальное отклонение от много- угольника Р. Таким образом точки можно вставлять до тех пор. пока не будет получен удовлетворительный результат. Процедура вставки к точек занимает время О(кп), где \Р\=п. В трехмерном пространстве задача упрощения становится намного труднее. Более то- го, задача поиска минимальной поверхности, разделяющей два многогранника, являет- ся NP-полной. В качестве эвристического алгоритма упрощения многогранников мож- но использовать какой-либо многомерный аналог рассматриваемых здесь плоскостных алгоритмов. Дополнительную информацию см. в подразделе "Примечания". Реализации. Алгоритм Дугласа-Пекера достаточно прост. Реализация этого алгоритма на языке С, показывающая хорошую производительность в наихудшем случае, пред- ставлена в работе [HS94]. Программа доступна по адресу http://www.cs.unc.edu/ -snoeyink/papers/DPsinip.arch.
618 Часть II. Каталог алгоритмических задач Для автоматического генерирования иерархических структур, определяющих уровень детализации многоугольных моделей, применяется метод упрощающих оболочек. Пользователь указывает максимальное отклонение поверхности упрощенной модели от поверхности исходной модели, после чего генерируется новая, упрощенная модель. Реализацию такого подхода можно загрузить с веб-страницы http://www.cs.unc.edu/ ~geom/envelope.html. Данная процедура сохраняет отверстия и не допускает самопере- сечений. Алгоритм QSlim на основе квадратичного упрощения может довольно быстро созда- вать высококачественные аппроксимации триангулированных поверхностей. Реализа- ция алгоритма доступна по адресу http://graphics.cs.uiuc.edu/~garland/software.html. Еще один подход к упрощению многоугольников основан на обработке результата преобразования к срединной оси многоугольника. Преобразование к срединной оси (см. раздел 17.10) создает скелет многоугольника, который можно упростить. После этого выполняется обратное преобразование, дающее в результате более простой мно- гоугольник. Программное обеспечение Сосопе может создавать аппроксимирующие преобразования к срединной оси многогранной поверхности, интерполируя точки в Е'. Дополнительную информацию можно найти на веб-сайте разработчиков http://www.cse.ohio-state.edu/~tamaldey/cocone.html. Теоретические основы програм- мы Сосопе излагаются в работе [Dey06]. Программа Powercrust (см. [АСКОI а] и [ACKOlb]) выполняет дискретное приближенное преобразование к срединной оси, по- сле чего реконструирует исходную поверхность на основе результатов этого преобра- зования. При достаточной плотности точек выборки алгоритм гарантированно выпол- няет геометрически и топологически правильную аппроксимацию поверхности. Библиотека CGAL (www.cgal.org) содержит процедуры для упрощения многоугольни- ков и многогранников и поиска наименьшей охватывающей окружности или сферы. Примечания Алгоритм Дугласа-Пекера (см. [DP73]) составляет основу большинства схем упрощения очертаний. Быстрые реализации этого алгоритма представлены в работах [HS94] и [HS98], Подход к упрощению многоугольников, основанный на вычислении реберной длины, представлен в работе [GHMS93J. Задача упрощения очертаний становится значи- тельно сложнее в трех измерениях. Даже задача построения выпуклого многоугольника с минимальным количеством вершин, находящегося между двумя вложенными выпуклыми многоугольниками, является NP-сложной (см. [DJ92]), хотя существуют аппроксимирующие алгоритмы для ее реше- ния (см. [MS95b]). Обзор алгоритмов для упрощения очертаний представлен в работе [HG97]. Использова- ние преобразований к срединной оси (см. раздел 17 10) для упрощения очертаний рас- сматривается в работе [ТНОЗ]. Проверку многоугольника на простоту можно выполнить за линейное время, по крайней мере теоретически, благодаря существованию линейного алгоритма триангуляции Шазе- ля; см. [Cha91]. Родственные задачи. Преобразование Фурье (см. раздел 13 11), выпуклая оболочка (см. раздел 17.2).
Глава 17. Вычислительная геометрия 619 17.13. Выявление сходства фигур Вход. Две фигуры, Р, и Р2- Задача. Выявить степень сходства этих фигур (рис. 17.13). ВХОД ВЫХОД Рис. 17.13. Выявление сходства фигур Обсуждение. Задача выявления сходства фигур относится к области распознавания образов. Рассмотрим, например, систему оптического распознавания текста. Дана биб- лиотека моделей фигур, представляющих буквы, и неизвестные фигуры, полученные в результате сканирования текста. Нам нужно идентифицировать неизвестные фигуры, сопоставив их с наиболее похожими моделями. Задача выявления сходства является плохо определенной по самой сути, т. к. смысл слова "сходство" зависит от конкретного приложения. Вследствие этого не существует единого алгоритмического подхода для решения всех задач выявления сходства фигур. Какой бы метод вы ни выбрали, вам придется потратить много времени на его на- стройку, чтобы добиться максимальной производительности. Возможны следующие подходы к решению этой задачи: ♦ использование расстояния Хемминга. Допустим, что сравниваемые многоугольники наложены друг друга. Расстояние Хемминга определяется площадью симметриче- ской разности между этими двумя многоугольниками, иными словами, площадью области, внутри одного из этих многоугольников, но не внутри обоих. Когда много- угольники идентичны и должным образом выровнены, расстояние Хемминга равно нулю. Если многоугольники отличаются только небольшим шумом вдоль границы, то расстояние Хемминга будет невелико. Вычисление площади области симметрической разности сводится к задачам поиска пересечения и объединения двух многоугольников (см. раздел 17.8), с последую- щим вычислением площадей (см. раздел 17.1). Трудным моментом при вычислении расстояния Хемминга является правильное выравнивание многоугольников. Задача выравнивания упрощается в таких приложениях, как оптическое распознавание тек- ста, поскольку символы текста естественно выровнены в строках на странице. Су- ществуют эффективные алгоритмы оптимизации наложения выпуклых многоуголь- ников без использования вращения. Простые, но эффективные эвристические мето- ды для решения этой задачи основаны на определении контрольных ориентиров для
620 Часть II. Каталог алгоритмических задач каждого многоугольника (таких как центр тяжести, ограничивающий прямоуголь- ник или экстремальные вершины) с последующим сопоставлением подмножеств этих ориентиров при выравнивании многоугольников. Расстояние Хемминга легко вычисляется на растровых изображениях, поскольку после выравнивания изображений остается лишь сложить расстояния между соот- ветствующими пикселами. Хотя расстояние Хемминга имеет концептуальный смысл и легко поддается реализации, оно отражает форму фигур весьма приблизи- тельно и, скорее всего, окажется неэффективным в большинстве приложений; ♦ использование хаусдорфова расстояния. Альтернативной метрикой расстояний яв- ляется хаусдорфово расстояние, равное расстоянию между точкой на многоуголь- нике Р\, максимально удаленной от многоугольника Р2, и многоугольником Р2. Эта метрика несимметрична. Например, длинный и тонкий выступ многоугольника Р\ может значительно увеличить хаусдорфово расстояние от Р| до Р2, даже если каж- дая точка Pi находится поблизости от какой-либо точки Р\. Небольшое утолщение всей границы одной из моделей (что может случиться при наличии шума на грани- це) может значительно увеличить расстояние Хемминга, но при этом мало повлияет на хаусдорфово расстояние. Какая же из этих двух метрик лучше? Все зависит от характера вашего приложения. Кроме прочего, правильное выравнивание многоугольников может быть сопряжено с трудностями и отнимает много времени; ♦ сравнение скелетов. При более эффективном подходе к выявлению сходства фигур применяется утончение (см. раздел 17 10), позволяющее получить древоподобный скелет каждого объекта, отражающий многие характеристики исходной фигуры. После этого задача сводится к сравнению двух скелетов с использованием таких их свойств, как топология дерева и длина и наклон ребер. Это сравнение можно смоде- лировать в виде выяснения изоморфизма подграфов (см. раздел 16.9), при котором ребра считаются совпадающими при достаточной схожести их длины и наклона: ♦ метод опорных векторов. Наконец, можно воспользоваться каким-либо обучаю- щим методом, например, нейронной сетью или более мощным методом опорных векторов. Применение этих методов является разумным подходом к решению задач распознавания, когда имеется большой объем данных, но нет ясной идеи, что делать с этими данными. В первую очередь нам нужно определить набор таких характери- стик фигуры, которые было бы легко найти с помощью компьютера. Например, это может быть площадь, количество сторон или количество отверстий. Исходя из этих характеристик программа-"черный ящик" (т. е. алгоритм обучения методом опор- ных векторов) обрабатывает обучающие данные и создает классифицирующую функцию. Эта функция принимает в качестве входа значения заданных характери- стик и возвращает меру фигуры, т. е. степень ее близости к определенной фигуре. Каково качество получаемых классифицирующих функций? Все зависит от особен- ностей вашего приложения. Подобно любому специализированному методу, метод опорных векторов требует серьезной настройки, если вам требуется полностью ис- пользовать его потенциал. Кроме того, нужно иметь в виду следующее. Если вы не знаете, как классифици- рующие функции "черного ящика" принимают решения, то вы не сумеете распо-
Гпава 17. Вычислительная геометрия 621 знать неверное решение, если они его выдадут. В связи с этим будет интересным пример системы, созданной для армии и предназначенной для различения танков и автомобилей. Система прекрасно работала на тестовых изображениях, но не вы- держала полевых испытаний. Наконец, кто-то сообразил, что фотографии автомо- билей снимались в солнечный день, а танков— в облачный, и система различала два типа объектов исключительно по присутствию облаков на заднем плане! Реализации. Реализация на языке С алгоритма для сравнения изображений с исполь- зованием хаусдорфова расстояния доступна по адресу http://www.cs.cornell.edu/ vision/hausdorff/hausmatch-html. Альтернативная метрика сходства многоугольников основана на угле проворота (см. [АСН+91]). Программа на языке С, использующая эту метрику, доступна по адресу http://www.cs.sunysb.edu/~algorith. Существует также несколько отличных реализаций алгоритмов, основанных на методе опорных векторов, такие как библиотека Kernel-Machine (http://www.terborg.net/ research/knil/), программа SVM/,l/,z (http://svmlight.joachims.org/) и широко исполь- зуемая и хорошо поддерживаемая библиотека L1BSVM (http://www.csie.ntu.edu.tw/ -cjlin/libsvm/). Примечания Среди книг обшей тематики по алгоритмам классификации образов можно назвать [DHSOO] и [JD88], Было предложено большое количество разнообразных подходов к ре- шению задачи выявления сходства фигур. См., например, [AMWW88], [АСН+91]. [Ata84], [АЕ83], [ВМ89] и [OW85]. Хороший обзор содержится в работе [AGOO], Оптимальное выравнивание при сдвиге (но не при вращении) сравниваемых многоуголь- ников из к и т вершин можно вычислить за время О((л + m)log(/7 + т)) (см. [dBDK+98]). Аппроксимация оптимального наложения при сдвиге и вращении была представлена в ра- боте [АСР+07]. Алгоритм с линейным временем исполнения для вычисления хаусдорфова расстояния между двумя выпуклыми многоугольниками был представлен в работе [Ata83], а алгорит- мы для общего случая — в работе [НК90]. Родственные задачи. Изоморфизм графов (см. раздел 16.9). преобразование к средин- ной оси (см. раздел 17.10). 17.14. Планирование перемещений Вход. Робот многоугольной формы, начинающий движение в точке 5 в комнате, со- держащей многоугольные препятствия, и конечная точка /. Задача. Найти самый короткий маршрут от точки 5 до точки /. не пересекающий ника- ких препятствий (рис. 17.14). Обсуждение. Сложность задачи планирования перемещений знакома каждому, кто пытался внести мебель в небольшую квартиру. Эта задача также возникает в области молекулярного докинга. Многие лекарства являются небольшими молекулами, дейст- вие которых основано на связывании с некоторой целевой молекулой. Поиск доступ- ных участков связывания при разработке новых лекарств является экземпляром задачи планирования перемещений. Классическим приложением задачи планирования пере- мещений является разработка маршрутов роботов.
622 Часть II. Каталог алгоритмических задач Наконец, планирование перемещений применяется в компьютерной анимации. Для данного набора объектов и их местоположения в сценах 5| и si алгоритм планирования перемещений может создать короткую последовательность промежуточных перемеще- ний, чтобы преобразовать сцену S| в сцену s-2. Эти перемещения можно использовать для того, чтобы заполнить промежуточные сцены между сценами Л| и s2. что значи- тельно облегчит работу аниматора. Сложность задач планирования перемещений зависит от многих факторов. ♦ Является ли робот точкой? Задача планирования перемещения точечного робота сводится к поиску кратчайшего пути от точки 5 к точке / без столкновений с препят- ствиями. Проще всего реализуется подход, который заключается в создании графа видимости многоугольных препятствий и точек s и /. У этого графа видимости име- ется вершина для каждой вершины препятствия, а две вершины препятствия соеди- няются ребром тогда и только тогда, когда между ними нет ребра препятствия, ме- шающего им "видеть" друг друга. Граф видимости можно создать, перебирая все кандидаты на ребро между (2) пар вершин и проверяя, пересекаются ли они с каждым из п ребер препятствий. Впро- чем. существуют и более быстрые алгоритмы. Каждому ребру графа видимости присваивается вес, равный его длине. Тогда кратчайший путь от точки v к точке t можно найти, используя алгоритм Дейкстры для поиска кратчайшего пути (см. раз- дел 15.4). Время исполнения этого алгоритма будет ограничено временем, требуе- мым для создания графа видимости. ♦ Какие действия может выполнять робот? Задача планирования перемещений становится значительно труднее, когда робот— не точка, а многоугольник. Тогда ширина всех используемых для его перемещения коридоров должна быть достаточ- ной, чтоб робот мог пройти по ним. Алгоритмическая сложность зависит от числа степеней подвижности (degrees of freedom) робота. В частности, может ли робот, кроме перемещения, совершать по- вороты? Есть ли у робота конечности, способные сгибаться или вращаться незави- симо от него, как. например, рука? Каждая степень подвижности соответствует из-
Гпава 17. Вычислительная геометрия 623 мерению пространства поиска возможных конфигураций. Дополнительная степень подвижности означает большую вероятность существования короткого маршрута, но при этом задача поиска этого маршрута также становится труднее. ♦ Можно ли упростить форму робота? Алгоритмы планирования перемещений обычно сложны и трудоемки. Вам будет полезно все, что позволит упростить зада- чу. В частности, рассмотрите возможность помещения робота в охватывающую ок- ружность. Если существует маршрут для этой окружности, то он будет также и маршрутом для находящегося внутри него робота. Кроме этого, любая ориентация окружности эквивалентна любой другой ее ориентации, поэтому повороты не будут играть никакой роли при поиске пути. Тогда все движения будут ограничены про- стым перемещением. ♦ Ограничены ли движения одним лишь перемещением? Когда повороты робота не разрешаются, можно воспользоваться методом расширения препятствий, чтобы свести задачу планирования перемещения робота-многоугольника к ранее решен- ной задаче планирования перемещения робота-точки. Для этого выбираем на роботе базисную точку и заменяем каждое препятствие его суммой Минковского с много- угольником робота. В результате получаем препятствие большего размера, вклю- чающее в себя след, оставляемый роботом, когда он движется вокруг препятствия, прикасаясь к нему. Поиск маршрута среди таких увеличенных препятствий от ис- ходной базисной точки до цели определяет допустимый маршрут робота- многоугольника в исходном окружении. ♦ Известны ли препятствия заранее? До сих пор мы предполагали, что для планиро- вания перемещений робота мы имеем карту с расположением всех препятствий. Но это невозможно в приложениях с движущимися препятствиями. Для решения задач планирования перемещений без карты существует два подхода. При первом подхо- де мы исследуем окружение, создаем карту и на ее основе разрабатываем маршрут от начальной точки к целевой. Другой, более простой подход напоминает переме- щение в тумане с помощью компаса. Идем в направлении цели до тех пор, пока наш путь не будет перекрыт препятствием. Потом движемся вокруг препятствия, пока опять не появится возможность продолжить движение в прежнем направлении. К сожалению, этот подход неприменим в достаточно сложной обстановке. Наиболее практичным подходом к решению общей задачи планирования перемещения будет произвольная выборка данных конфигурационного пространства робота. Кон- фигурационное пространство определяет набор допустимых положений робота с ис- пользованием одного измерения для каждой степени подвижности. Плоский робот, обладающий возможностью перемещения и поворота, имеет три степени подвижности, а именно х- и у-координату базисной точки на роботе и угол 6 поворота относительно этой точки. Некоторые точки в этом пространстве соответствуют разрешенным пози- циям, а другие — препятствиям. Набор разрешенных точек конфигурационного пространства создается случайной вы- боркой. Для каждой пары точек р\ и pi выясняем, существует ли между ними прямой путь, не пересекающий препятствий. Таким образом строится граф, вершины которого представляют допустимые точки конфигурационного пространства, а ребра— свобод- ные от препятствий пути между этими точками. Задача планирования перемещения
624 Часть II. Каталог алгоритмических задач теперь сводится к поиску прямого пути между начальной/конечной точкой маршрута и какой-либо вершиной графа и последующему поиску кратчайшего пути между этими двумя вершинами. Существует много способов улучшения этого базового метода, например, путем добав- ления дополнительных вершин в области, представляющих особый интерес. Создание такой дорожной карты позволяет точно решать задачи, которые в противном случае выглядят очень запутанными. Реализации. Пакет Motion Planning Kit содержит библиотеку процедур на языке C++ и набор средств для разработки планировщиков перемещений для одного и нескольких роботов. В состав набора входит SBL — быстрый вероятностный планировщик мар- шрутов на дорожной карте. Набор можно найти по адресу http://robotics.stanford.edu/ ~mitul/mpk/. Группа GAMMA из университета Северной Каролины разработала несколько эффек- тивных библиотек для обнаружения столкновений (строго говоря, это не то же самое, что планирование перемещений), из которых самой последней является библиотека SW1FT++ (см. [EL01 ]). Эта библиотека поддерживает выявление пересечений, вычис- ление приблизительных и точных расстояний между объектами, а также обнаружение контакта между парой объектов в сценах, составленных из негибких многогранных моделей. Подробную информацию об этих библиотеках, включая условия загрузки и использования, можно найти на веб-сайте http://wvvvv.cs.unc.edu/~georn/collide/. Библиотека вычислительной геометрии CGAL (www.cgal.org) содержит большое ко- личество реализаций алгоритмов, касающихся задачи планирования перемещений, включая процедуры создания графов видимости и вычисления сумм Минковского. В книге [O'ROl] предоставлена реализация алгоритма для планирования перемещений на плоскости двухшарнирного робота-манипулятора. Подробности см. в разделе 19.1.10. Примечания В книге [Lat91] обсуждаются практические подходы к планированию перемещений, включая описанный выше метод случайной выборки. Две другие заслуживающие внима- ния книги по предмету планирования перемещений можно загрузить бесплатно. [LaV06] — по адресу http://planning.cs.uiuc.edu, a [Lau98] по адресу http://homepages. Iaas.fr/jpl/book.html. Задача планирования перемещений изначально изучалась Шварцем (Schwartz) и Шариром (Sharir). Разработанное ими решение создает полное пространство позиций робота, не пе- ресекающихся с препятствиями, после чего находит кратчайший путь в соответствующей компоненте связности. Эти описания свободного от препятствий пространства очень сложны. Доклады, посвященные задаче планирования перемещений, приводятся в книге [HSS87], а в книге [Sha04] дается обзор последних результатов. Самый лучший результат для подхода, основанного на использовании свободного от пре- пятствий пространства, был приведен в [Сап87], где показано, что любую задачу с d сте- пенями подвижности можно решить за время Olri'ign), хотя для частных случаев обшей задачи планирования перемещений существуют более быстрые алгоритмы. Подход к за- даче планирования перемещений, основанный на методе расширения препятствий, был представлен в работе [LPW79]. Обсуждение эвристического метода "с компасом в тума- не" содержится в работе [LS87].
Глава 17. Вычислительная геометрия 625 Временная сложность алгоритмов, основанных на методе свободного от препятствий про- странства, зависит от комбинаторной сложности расположения поверхностей, опреде- ляющих свободное пространство. Алгоритмы для поддержания таких компоновок пред- ставлены в разделе 17.15. В анализе таких компоновок часто возникают последовательно- сти Дэвенпорта-Шинцеля. Всестороннее рассмотрение последовательностей Дэвенпорта- Шинцеля и их значимость для задачи планирования перемещений приводится в книге [SA95], Граф видимости п отрезков с £ парами видимых вершин можно создать за время O(nlg« + £) (см. [GM91] и [PV96]), что является оптимальным результатом. Алгоритм с временем исполнения O(nlgn) для поиска кратчайшего пути для робота-точки среди пре- пятствий-многоугольников приводится в работе [HS99], А в работе [Che85] приводится алгоритм с временем исполнения O(n~lgn) для поиска кратчайшего пути среди препятст- вий-многоугольников для робота-окружности. Родственные задачи. Задача поиска кратчайшего пути (см. раздел 15 4), сумма Мин- ковского (см. раздел 17.16). 17.15. Конфигурации прямых Вход. Набор прямых .... Задача. Найти разбиение плоскости, определяемое набором прямых .... (рис. 17.15). ВХОД ВЫХОД Рис. 17.15. Разбиение плоскости Обсуждение. Явное создание областей, формируемых пересечениями набора п пря- мых. является одной из фундаментальных задач вычислительной геометрии. Многие задачи сводятся к созданию и анализу конфигурации некоторого набора прямых. Мож- но привести два примера подобных задач: ♦ проверка на вырожденность. Для данного набора из п прямых на плоскости выяс- нить, проходят ли какие-либо три из них через одну и ту же точку. Проверка всех таких триад методом исчерпывающего перебора займет время О(п). В качестве альтернативы мы можем создать конфигурацию прямых, а потом рассмотреть каж-
626 Часть II. Каталог алгоритмических задач дую вершину и явно подсчитать ее степень, причем все это делается за квадратич- ное время: ♦ удовлетворение максимальному количеству линейных ограничений. Допустим, что нам дан набор линейных ограничений, каждое в виде у < аре + Ь,. Нам нужно найти точку на плоскости, которая удовлетворяет самому большому количеству таких ог- раничений. Для решения этой задачи сначала создаем конфигурацию прямых. Все точки в любой области (или ячейке), образуемой пересекающимися линиями, удов- летворяют одному и тому же набору ограничений, поэтому, чтобы найти глобаль- ный максимум, нам нужно проверить только одну точку в каждой ячейке. При разработке алгоритмов бывает полезно формулировать геометрические задачи в терминах свойств конфигурации. К сожалению, следует признать, что на практике конфигурации не пользуются такой популярностью, как можно было бы предполагать. Основной причиной является то обстоятельство, что для правильного применения конфигураций требуется определенный уровень знаний. Библиотека вычислительной геометрии CGAL предоставляет общую и достаточно устойчивую реализацию, оправ- дывающую усилия по использованию конфигураций. Столкнувшись с задачей конфи- гурации прямых, постарайтесь ответить на следующие вопросы. ♦ Какой метод лучше всего подходит для создания конфигураций прямых? Для этой цели используются инкрементальные алгоритмы. Начинаем с конфигурации из од- ной или двух линий, в которую добавляем по одной новые прямые, получая конфи- гурации все большего размера. Чтобы добавить в конфигурацию новую прямую, начинаем с самой левой ячейки, содержащей эту прямую, и идем по конфигурации вправо, перемещаясь от текущей ячейки к смежной ячейке и разбивая на две части те ячейки, которые содержат новую прямую. ♦ Каким будет размер создаваемой конфигурации? Согласно теореме о зоне, k-я до- бавленная прямая пересекает к ячеек конфигурации, причем О{к) ребер образуют границы этих ячеек. Это означает, что мы можем перебрать все ребра каждой ячей- ки. обнаруживаемой в процессе добавления прямых, и быть уверенным, что общий объем работы, выполненный при добавлении прямой в конфигурацию, будет ли- нейным. Поэтому общее время добавления всех п прямых для создания полной конфигурации будет О(и2). ♦ Какова цель создания конфигурации? Часто требуется найти ячейку данной конфи- гурации, содержащую заданную точку. Это задача выяснения местоположения точ- ки {см. раздел 17.7). А для данной конфигурации прямых или отрезков часто требу- ется вычислить все точки пересечения этих прямых. Задача выявления пересечений обсуждается в разделе 17.8. ♦ Входные данные — это набор точек, а не прямых? Хотя прямые и точки являются разными геометрическими объектами, они могут заменить друг друга. Используя преобразования двойственности, можно преобразовать прямую L в точку р и на- оборот: К. у = lax — b<^>p: {а, Ь) Важность двойственности состоит в том, что мы теперь можем применять конфигу- рации прямых в задачах на точках, нередко получая неожиданные результаты.
Глава 17. Вычислительная геометрия 627 Рассмотрим пример. Имеется множество п точек и требуется узнать, не располага- ются ли какие-либо три из этих точек на одной и той же прямой. Эта задача похожа на задачу проверки на вырожденность, рассматриваемую ранее. В действительно- сти, это та же самая задача, но точки и прямые в ней поменялись ролями. Мы мо- жем выполнить преобразование двойственности точек в прямые, как описано выше, а потом выполнить поиск вершины, через которую проходят три прямых. Преобра- зование двойственности этой вершины определяет прямую, на которой располага- ются три исходные вершины. Часто требуется перебрать все многоугольники существующей конфигурации ровно один раз. Такие алгоритмы называются алгоритмами заметающей прямой и рассмат- риваются в разделе 17.8. Базовая процедура таких алгоритмов состоит в упорядочива- нии точек пересечения по х-координате и выполнении обхода слева направо с сохране- нием информации, обнаруженной в процессе этого обхода. Реализации. Библиотека CGAL (wvvvv.cgal.org) содержит пакет общих процедур для работы с конфигурациями кривых (не только прямых) на плоскости. Эта библиотека должна быть отправной точкой для любого проекта, в котором используются конфигу- рации. Устойчивый код на языке C++ для создания и топологического заметания конфигура- ций можно найти на веб-сайте http://vvvvvv.cs.tufts.edu/rescarch/geometry/other/svveep В библиотеке CGAL предоставлено расширение топологического заметания для рабо- ты с комплексом видимости набора попарно непересекающихся выпуклых плоских множеств. Пакет Arrange на языке С предназначен для размещения многоугольников на плоско- сти или на сфере. Многоугольники могут быть вырожденными, и в таком случае работа сводится к размещению отрезков. В пакете используется рандомизированный инкре- ментальный алгоритм и поддерживается эффективное выяснение местоположения точки в разбиении. Пакет Arrange был разработан Майклом Голдвассером (Michael Goldwasser) и доступен на веб-сайте http://euler.slu.edu/~goldvvasser/publications. Примечания Подробное изложение комбинаторной теории конфигураций и соответствующие алгорит- мы представлены в книге [Ede87]. Эта книга— основной справочник для каждого, кто серьезно интересуется данной темой. Свежие обзоры комбинаторных и алгоритмических результатов приводятся в работах [ASOO] и [На104]. Обсуждение принципов создания конфигураций можно найти в книгах [dBvKOSOO] и [O'ROl]. Вопросы реализации про- цедур библиотеки CGAL рассматриваются в работах [ННОО] и [FWH04]. Конфигурации естественным образом обобщаются для случая многомерных пространств. В трех измерениях разбиение пространства определяется плоскостями, а в многомерных пространствах— гиперплоскостями. Теорема о зоне утверждает, что сложность любой конфигурации из п «/-мерных гиперплоскостей равна а любая одиночная гиперпло- скость пересекает ячейки со сложностью O(nJ '). Это дает основание для алгоритма ин- крементального создания конфигураций. Обход вдоль границы ячейки с целью поиска следующей ячейки, пересекаемой гиперплоскостью, занимает время, пропорциональное количеству ячеек, создаваемых добавлением гиперплоскости.
628 Часть II. Каталог алгоритмических задач Теорема о зоне имеет несколько запутанную историю. Первоначальные доказательства оказались ошибочными для случая многомерных пространств. Обсуждение теоремы и верное доказательство приведены в работе [ESS93]. Теория последовательностей Дэвен- порта-Шинцеля тесно связана с изучением конфигураций (см. [SA95]). Простой алгоритм заметания конфигурации прямых сортирует п точек пересечения по х-координате и поэтому занимает время (9(H"lgn). Использование топологического замета- ния (см. [EG89] и [EG91 ]) избавляет от необходимости в сортировке, вследствие чего об- ход конфигурации выполняется за линейное время. Этот алгоритм легко поддается реали- зации и его можно использовать, чтобы ускорить работу многих алгоритмов, в основе ко- торых лежит метод заметающей прямой. Устойчивая реализация и экспериментальные результаты представлены в работе [RSS02], Родственные задачи. Выявление пересечений (см. раздел 17.8), выяснение местопо- ложения точки (см. раздел 17.7). 17.16. Сумма Минковского Вход. Наборы точек или многоугольники А и В, содержащие п и т вершин соответст- венно. Задачи. Найти сумму Минковского: А + В= {х+_у|х е А, у е В} (рис. 17.I6). ВХОД ВЫХОД Рис. 17.16. Сумма Минковского Обсуждение. Вычисление суммы Минковского— полезная геометрическая операция, с помощью которой можно увеличивать объекты, чтобы они удовлетворяли опреде- ленным требованиям. Например, в известном подходе к планированию перемещений роботов многоугольной формы в комнате с препятствиями многоугольной формы (см. раздел 17 14) размер каждого из препятствий увеличивается на величину суммы Мин- ковского препятствия и робота. Таким образом, задача сводится к более простому слу- чаю планирования перемещения робота-точки. Другим применением суммы Минков- ского является задача упрощения формы многоугольников (см. раздел 17.12). Здесь мы делаем границу объекта толще, чтобы создать вокруг него канал, а потом выясняем упрощенную форму объекта, для чего находим путь внутри этого канала, состоящий из минимального количества отрезков. Наконец, сумма Минковского для объекта, имею- щего очень неровную границу, и небольшого круга поможет сгладить его границу, устранив небольшие впадины и выступы.
Глава 17. Вычислительная геометрия 629 Далее приводится определение суммы Минковского (при этом предполагается, что многоугольники А и В находятся в некоторой системе координат): А + В = {.х +_у| х е А, у G В}, где х + у— сумма соответствующих векторов. В терминах операции переноса сумма Минковского — это объединение всех переносов многоугольника А на расстояние, оп- ределяемое точкой внутри многоугольника В. При вычислении суммы Минковского возникают следующие вопросы. ♦ Входные объекты представляют собой растровые изображения или многоугольни- ки? Если А и В являются растровыми изображениями, то определение суммы Мин- ковского подсказывает простой алгоритм для ее вычисления. Инициализируем дос- таточно большую матрицу пикселов, вычислив сумму Минковского ограничиваю- щих прямоугольников для многоугольников А и В. Для каждой пары точек из А и В складываем их координаты и уменьшаем яркость соответствующего пиксела. Эти алгоритмы становятся более сложными, если требуется явное представление суммы Минковского в виде многоугольника. ♦ Требуется ли увеличить размер объекта на заданное значение? При типичной опе- рации увеличения объекта размер модели М увеличивается на определенное значе- ние допуска /, называемое смещением (offsetting). Как показано на рис. 17.16, ре- зультат достигается путем вычисления суммы Минковского для модели М и круга радиусом /. Базовые алгоритмы продолжают работать, хотя полученный объект не является многоугольником. Теперь его граница состоит из дуг и отрезков. ♦ Являются ли входные объекты выпуклыми? Сложность вычисления суммы Мин- ковского во многом зависит от формы многоугольников. Если оба многоугольника выпуклые, то сумму Минковского можно вычислить за время О(п + т), проведя один многоугольник по контурам другого. Если один из многоугольников не явля- ется выпуклым, то размер результирующего объекта может достичь значения 0(шл). Если же невыпуклыми являются оба многоугольника, то размер результата может достичь значения О(п~т~). Суммы Минковского для невыпуклых много- угольников часто имеют эстетически непривлекательный вид, поскольку отверстия в них возникают или исчезают самым неожиданным образом. Простейший подход к вычислению сумм Минковского основан на триангуляции и объединении. Сначала выполняем триангуляцию каждого многоугольника, а затем вы- числяем сумму Минковского для каждого треугольника из А с каждым треугольником из В. Сумма двух треугольников легко вычисляется и является частным случаем суммы выпуклых многоугольников, рассматриваемым далее. Объединением этих О(пт) вы- пуклых многоугольников будет сумма А + В. Алгоритмы поиска объединения много- угольников основаны на методе заметающей прямой (см. раздел 17.8). Вычислить сумму Минковского для двух выпуклых многоугольников легче, чем для общего случая, т. к. эта сумма всегда будет выпуклой. Когда многоугольники выпук- лые. проще перемещать многоугольник А вдоль границы многоугольника В и вычис- лять сумму для каждого ребра. Подход, в котором каждый многоугольник разбивается на небольшое количество выпуклых фрагментов (см. раздел 17.11), а потом вычисляет- ся сумма Минковского для каждой пары фрагментов, обычно намного эффективнее, чем обработка двух многоугольников, полностью подвергнутых триангуляции.
630 Часть II. Каталог алгоритмических задач Реализации. Пакет библиотеки CGAL предоставляет эффективные процедуры вычис- ления суммы Минковского для двух произвольных многоугольников, а также вычис- ления точных и приблизительных смещений. Реализация алгоритма вычисления суммы Минковского для двух выпуклых много- гранников в трех измерениях описана в работе [FH06] и доступна по адресу http://vvvvvv.cs.tau.ac.il/~erif/CD/. Примечания Обсуждение алгоритмов вычисления суммы Минковского можно найти в книгах [dBvKOSOO] и [O'ROl]. Самые быстрые алгоритмы вычисления суммы Минковского представлены в работах [KOS91] и [Sha87]. Практическая эффективность вычисления суммы Минковского в обшем случае зависит от того, каким образом многоугольники разбиты на выпуклые фрагменты. Разбиение много- угольников на минимальное количество выпуклых фрагментов не обязательно будет оп- тимальным решением. Всестороннее обсуждение методов разбиения многоугольников для вычисления суммы Минковского представлено в работе [AFH02]. Комбинаторная сложность суммы Минковского для двух выпуклых многогранников в трех измерениях полностью определена в работе [FHW07]. Реализация алгоритма вычис- ления суммы Минковского для таких многогранников описана в работе [FH06]. В работе [KS90] представлен эффективный алгоритм, основанный на вычислении сумм Минковского для планирования перемещений роботов многоугольной формы. Родственные задачи. Преобразования к срединной оси (см. раздел 17 10), планирова- ние перемещений (см. раздел 17.14), упрощение многоугольников (см. раздел 17.12).
ГЛАВА 18 Множества и строки Как множества, так и строки являются коллекциями объектов, и разница между ними состоит в том. имеет ли значение порядок элементов коллекции. Множества— это коллекции символов, порядок которых не имеет значения, в то время как строки опре- деляются как последовательность символов. Тот факт, что элементы в строках упорядочены, позволяет решать задачи со строками намного эффективнее, чем задачи с множествами, благодаря возможности использо- вать такие методы, как динамическое программирование, и такие развитые структуры данных, как суффиксные деревья. Причиной повышения интереса, проявляемого к ал- горитмам обработки строк, и важности этих алгоритмов являются такие приложения, как биоинформатика, поиск в Интернете и другие приложения обработки текста. Среди последних книг по алгоритмам для обработки строк можно назвать следующие: ♦ [Gus97]) — пожалуй, самое лучшее введение в обработку строк. Эта книга содержит подробное обсуждение суффиксных деревьев, а также современные формулировки классических алгоритмом для точного сравнения строк; ♦ [CHL07] — подробное описание алгоритмов обработки строк, автор которого явля- ется признанным лидером в данной области. Перевод с французского на англий- ский; ♦ [NR07] — краткое, но имеющее практическую ценность обсуждение алгоритмов поиска по образцу, ориентированное на конкретные реализации. Особое внимание уделяется подходам, в которых применяется параллелизм на уровне битов; ♦ [CR03]—обзор некоторых специальных тем, имеющих отношение к алгоритмам обработки строк, с акцентом на теорию. Ежегодная конференция СРМ (Combinatorial Pattern Matching, комбинаторное сравне- ние строк) является основным форумом, посвященным практическим и теоретическим аспектам алгоритмов обработки строк. 18.1. Поиск покрытия множества Вход. Коллекция подмножеств 5= {Л’|.S',») универсального множества (/= {1,.... н). Задача. Найти наименьшую коллекцию Т подмножеств множества S. объединение ко- торых равно универсальному множеству, т. е. U/=|7^=6/ (рис. 18.1). Обсуждение. Задача о покрытии множества возникает, когда мы стремимся экономно приобрести товары, разложенные по наборам. Мы хотим получить, как минимум, по одному товару каждого вида, покупая при этом как можно меньшее количество набо-
632 Часть II. Каталог алгоритмических задач ров. Задача поиска хоть какого-нибудь покрытия множества не представляет сложно- сти. т. к. можно купить все предлагаемые наборы товаров. Но поиск наименьшего по- крытия множества позволяет нам минимизировать наши расходы. Задача о покрытии множества позволила упростить формулировку задачи оптимизации выбора лотерей- ных билетов, обсуждавшейся в разделе 1.6. В той задаче нам требовалось купить наи- меньшее количество лотерейных билетов, покрывающее все комбинации данного на- бора. Другим интересным применением задачи о покрытии множества является задача оп- тимизации булевой логики. Рассмотрим, например, булеву функцию с к переменными, возвращающую 0 или 1 для каждого из возможных 2* входных векторов. Нам нужно найти простейшую логическую схему, которая реализует эту функцию. Один из подхо- дов— определить на этих переменных и их дополнениях булеву формулу в дизъюнк- тивной нормальной форме (ДНФ), например: хрс2 Для каждого входного векто- ра мы могли бы построить один член И. а потом выполнить операцию ИЛИ над всеми этими членами И. однако мы существенно сэкономим, если вынесем за скобки общие подмножества переменных. Имея набор выполнимых членов И, каждый из которых покрывает какое-либо подмножество нужных нам векторов, мы хотим объединить операцией ИЛИ наименьшее количество членов, которые реализует данную функцию. Это как раз и есть задача о покрытии множества. Существует несколько разновидно- стей задачи о покрытии множества, и для их распознавания нужно ответить на сле- дующие вопросы. ♦ Разрешено ли повторное включение элементов в покрытие? Если любой элемент может входить только в одно подмножество, то задача о покрытии множества пре- вращается в задачу укладки множества, которая рассматривается в разделе 18.2. Если есть возможность включать один элемент в несколько подмножеств, то сле- дует воспользоваться ею, т. к. в результате обычно удается получить меньшее по- крытие. ♦ В задаче рассматривается множество ребер или множество вершин графа? Зада- ча о покрытии множества имеет общий характер и включает в себя несколько по- лезных задач на графах в качестве частных случаев. Допустим, что нужно найти
Глава 18. Множества и строки 633 наименьшее множество ребер графа, которое затронет каждую вершину, по мень- шей мере, один раз. Решением данной задачи будет максимальное паросочетание в графе (см. раздел 15.6), к которому добавлены любые ребра, позволяющие охватить все вершины, не вошедшие в паросочетание. А теперь допустим, что нужно найти наименьшее множество вершин графа, которое задействует каждое ребро, по край- ней мере, один раз. Эта задача является задачей о вершинном покрытии, эвристиче- ские методы для решения которой рассматриваются в разделе 16.3. Здесь полезно продемонстрировать, как смоделировать вершинное покрытие в виде экземпляра покрытия множества. Пусть универсальное множество U соответствует множеству ребер {еь ..., ет}. Создаем п подмножеств, где подмножество S, состоит из ребер, инцидентных вершине v,. Хотя задача о вершинном покрытии является лишь замаскированным экземпляром задачи о покрытии множества, имеет смысл воспользоваться более эффективными эвристическими методами решения частной задачи о вершинном покрытии. ♦ Каждое подмножество содержит только два элемента? Если в любом подмно- жестве содержится, самое большее, два элемента, считайте, что вам повезло. Для этого частного случая можно получить оптимальное решение, т. к. он сводится к поиску максимального паросочетания графа. К сожалению, как только количество элементов во всех подмножествах возрастает до трех, задача становится NP-полной. ♦ Нужно найти множества, содержащие элементы, или элементы, содержащиеся во множествах? В задаче о минимальном множестве представителей (hitting set) нужно найти множество элементов, которые совместно представляют каждое под- множество из данной коллекции. Пример минимального множества представителей показан на рис. 18.2. Оптимальное решение экземпляра задачи о минимальном множестве представителей получается при выборе элементов 1 и 3 или элементов 2 и 3 (а). Эту задачу можно преобразовать в экземпляр двойственной задачи о покры- тии множества, оптимальным решением которой будет выбор подмножеств 1 и 3 или подмножеств 2 и 4 (б). Рис. !8.2. Пример минимального множества представителей Входной экземпляр задачи о минимальном множестве представителей идентичен входному экземпляру задачи о покрытии множества, но здесь требуется найти такое минимальное подмножество элементов ТcU, чтобы каждое подмножество S, со- держало, по крайней мере, один элемент подмножества Т. Таким образом. S, n ТФ О
634 Часть II. Каталог алгоритмических задач для всех 1 </ <т. Допустим, что мы хотим создать небольшой парламент, в кото- рый входил бы, по меньшей мере, один представитель от каждой этнической груп- пы. Если этническая группа определяется как подмножество всего населения, то решение задачи о минимальном множестве представителей будет решением задачи создания политкорректного парламента. Задача о минимальном множестве представителей двойственна задаче о покрытии множества. Заменим каждый элемент множества U множеством имен подмножеств, содержащих его. Теперь множества S' и U поменялись ролями, т. к. мы ищем мно- жество подмножеств множества U, чтобы покрыть все элементы в множестве S. Мы получили задачу о покрытии множества, поэтому можем использовать любую про- грамму для ее решения, чтобы решить задачу о минимальном множестве представи- телей. Пример изображен на рис. 18.2. Задача о покрытии множества должна быть, по меньшей мере, так же сложна, как и задача о вершинном покрытии, поэтому она тоже является NP-полной. На самом деле, она еще сложнее. Решение, выдаваемое аппроксимирующими алгоритмами для задачи о вершинном покрытии, хуже оптимального не более чем в два раза, а для задачи о по- крытии множества самое лучшее решение хуже оптимального в ®(lgn) раз. Самым естественным и эффективным подходом к решению задачи о покрытии множе- ства будет использование "жадного"" эвристического алгоритма. Для начала выбираем самое мощное подмножество для покрытия, после чего удаляем все его элементы из универсального множества. Добавляем подмножество, содержащее наибольшее коли- чество неохваченных элементов, и повторяем это действие, пока все элементы не будут покрыты. Количество подмножеств покрытия множества, выдаваемых таким эвристи- ческим алгоритмом, никогда не превысит оптимальное более, чем в Inn раз, причем на практике этот коэффициент намного меньше. Простейшая реализация "жадного" эвристического алгоритма просматривает на каж- дом шаге весь входной экземпляр из т подмножеств. Впрочем, используя такие струк- туры данных, как связные списки и очереди с приоритетами ограниченной высоты (см. рейде. 112.2). можно реализовать "жадный" эвристический алгоритм с временем испол- нения O(S). где S = и™, | S, | — размер входного экземпляра. Полезно проверить существование элементов, содержащихся лишь в небольшом коли- честве подмножеств, в идеале, только в одном. При наличии таких элементов следует выбрать наибольшие содержащие их подмножества в самом начале работы алгоритма. В конечном счете нам все равно придется их выбрать, но они содержат другие элемен- ты. покрытие которых потребует дополнительных затрат, если выбрать эти подмноже- ства не с самого начала. Метод имитации отжига, скорее всего, даст лучшие результаты, чем эти простые эври- стические подходы. Чтобы гарантировать оптимальное решение, можно использовать поиск с возвратом, но выгода от этого не оправдает дополнительные расходы. Еще один, более мощный подход, основан на переформулировке задачи о покрытии множества в терминах целочисленного программирования. Пусть целая переменная s„ принимающая два значения, 0 и 1, обозначает, выбрано ли подмножество S для данно- го покрытия. Для каждого элемента х из универсального множества добавляется огра-
Глава 18 Множества и строки 635 ничение £5, >1. гарантирующее, что он будет покрыт хотя бы одним выбранным хеХ, подмножеством. Минимальное покрытие множества удовлетворяет всем ограничи- вающим условиям и одновременно минимизирует х, . Эту задачу целочисленного программирования можно с легкостью обобщить до задачи о взвешенном покрытии множества (если допустить разную стоимость разных подмножеств). Ослабив эту зада- чу до задачи линейного программирования (т. е., позволив каждой переменной s, нахо- диться в диапазоне 0<л, <1, не ограничивая ее только двумя значениями 0 или 1), можно получить эффективный эвристический алгоритм, основанный на методах ок- ругления. Реализации. Как "жадный" эвристический подход, так и подход с использованием це- лочисленного линейного программирования является достаточно простым в своей об- ласти, что его нужно реализовать с чистого листа. Реализация на языке Pascal алгоритма исчерпывающего перебора для решения задачи укладки множества, а также для задачи о покрытии множества, дается в книге [SDK83]. Подробности см. в разделе 19.1.10. Пакет SYMPHONY содержит процедуру для решения задачи разбиения множества методами смешанного линейного программирования. Загрузить ее можно с веб-сайта http://branchandcut.org/SPP. Примечания В работе [ВР76] представлен классический обзор методов решения задачи о покрытии множества, а более свежий обзор аппроксимирующих методов с анализом их сложности дан в работе [Pas97]. Результаты исследований эвристических методов целочисленного программирования и точные алгоритмы решения задачи о покрытии множества изложены в работах [CFT99J и [CFTOO]. Отличное обсуждение алгоритмов и правил сведения для задачи о покрытии множества представлено в книге [SDK83J. Среди хороших работ, посвященных "жадным" эвристическим алгоритмам решения зада- чи о покрытии множества, можно назвать книги [CLRSOI] и [Нос96]. Пример, демонст- рирующий, что решение задачи о покрытии множества, выдаваемое "жадным” эвристиче- ским алгоритмом, может быть в lg/7 хуже оптимального, представлен в работах [Joh74] и [PS98]. Но такой результат не является следствием дефекта в алгоритме. Доказана слож- ность получения приблизительного решения задачи о покрытии множества с коэффици- ентом, лучшим, чем (1 - о(1))1пл, (см. [Fei98]). Родственные задачи. Паросочетание (см. раздел 15.6), вершинное покрытие (см. раз- дел 16.3), укладка множества (см. раздел 18.2). 18.2. Задача укладки множества Вход. Множество подмножеств S = {Si,.... S„,} универсального множества U= {1,....»}. Задача. Выбрать коллекцию (в идеале, небольшую) взаимно пепересекающихся под- множеств из множества S, объединением которых будет универсальное множество (рис. 18.3).
636 Часть II. Каталог алгоритмических задач ВХОД ВЫХОД Рис. 18.3. Пример укладки множества Обсуждение. Задачи укладки множества возникают в приложениях, имеющих строгие ограничивающие условия на разрешенное разбиение. Главной особенностью задач ук- ладки множества является условие, согласно которому ни один элемент не может быть покрыт больше, чем одним выбранным подмножеством. Эта задача в определенной степени аналогична задаче поиска независимого множества в графах (см. раздел 16.2), где требуется найти наибольшее подмножество вершин гра- фа G, такое что каждое ребро инцидентно не более чем одной из выбранных вершин. Смоделируем задачу поиска подмножества вершин в виде задачи укладки множества. Пусть универсальное множество состоит изо всех ребер графа G, а подмножество S, состоит изо всех ребер, инцидентных вершине г,. Дополнительно определим одноэле- ментное множество для каждого ребра. Любая укладка множества определяет множе- ство вершин, не имеющих общих ребер, другими словами, независимое множество. Одноэлементные множества используются для покрытия ребер, не охваченных вы- бранными вершинами. Еще одним применением укладки множества будет комплектование экипажей самоле- тов. На каждый самолет авиакомпании нужно назначить экипаж, состоящий из двух пилотов и штурмана. К составу экипажа предъявляются определенные требования, та- кие как умение управлять самолетом данного типа, психологическая совместимость, а также рабочее расписание. Для всех возможных комбинаций экипажей и самолетов, каждая из которых представлена подмножеством элементов, нужно найти такой вари- ант, при котором каждый самолет и каждый член экипажа присутствуют только в од- ной комбинации. При данных ограничивающих условиях на подмножества нам нужно найти безупречную укладку. Задача укладки множества включает в себя несколько типов NP-полных задач на мно- жествах. Для их распознавания нужно ответить на следующие вопросы. ♦ Обязательно ли, чтобы каждый элемент входил только в одно выбранное под- множество? В задаче о точном покрытии требуется найти коллекцию подмно- жеств. в которой каждый элемент покрыт ровно один раз. Рассмотренная выше за- дача комплектования экипажей самолетов близка к задаче точного покрытия, т. к. в ней должны<5ыть задействованы каждый самолет и каждый экипаж.
Глава 18. Множества и строки 637 К сожалению, задача поиска точного покрытия ставит нас в ситуацию, аналогичную поиску гамильтонова цикла в графах. Если действительно нужно покрыть все эле- менты только по одному разу, а эта задача является NP-полной, то единственным способом ее решения является поиск с экспоненциальным временем исполнения. Стоимость такого подхода окажется очень высокой, если только вам не повезет, и вы не натолкнетесь на решение в начале поиска. ♦ Есть ли у каждого элемента свое одноэлементное множество? Ситуация намного улучшается, если нам достаточно частичного решения, например, такого, при кото- ром каждый элемент универсального множества U определяется как одноэлемент- ное подмножество множества S. Тогда мы можем расширить любую укладку мно- жества до точного покрытия, покрыв не попавшие в укладку элементы U одноэле- ментными множествами. После этого наша задача сводится к поиску укладки множества минимальной мощности, решение которой можно получить с помощью эвристических методов. ♦ Какова плата за двойное покрытие элементов? В задаче о покрытии множества (см. раздел 18.1) включение одного элемента в несколько выбранных подмножеств не влечет за собой никаких отрицательных последствий. Однако в задаче точного покрытия такие многократные вхождения запрещены. Во многих приложениях за- преты не такие строгие. В одном из подходов к решению таких задач можно увели- чить стоимость выбора подмножеств, содержащих элементы, входящие в уже вы- бранные подмножества. Самым лучшим подходом к решению задачи укладки множества является использова- ние "жадных" эвристических алгоритмов, подобных тем, что применяются для реше- ния задачи о покрытии множества (см. раздел 18.1). Если нам требуется найти укладку, содержащую наибольшее (наименьшее) количество подмножеств, тогда мы выбираем наименьшее (наибольшее) подмножество, удаляем все конфликтующие с ним подмно- жества из множества S и повторяем процесс. Как обычно, если мы дополним этот под- ход каким-либо методом исчерпывающего поиска или рандомизации (например, мето- дом имитации отжига), мы, вероятно, получим более удачные укладки за счет увеличе- ния времени работы. Еще один, более мощный подход, основан на переформулировке задачи о покрытии множества в терминах целочисленного программирования, подобно тому, как мы де- лали это в задаче о покрытии множества. Пусть целая переменная s,. принимающая два значения, 0 и 1, обозначает, выбрано ли подмножество S, для данного покрытия. Для каждого элемента х из универсального множества добавляется ограничение =1. гарантирующее, что он будет покрыт только одним выбранным подмножеством. Ми- нимизация или максимизация si ПРИ соблюдении этих ограничивающих условий позволяет нам регулировать количество подмножеств в покрытии. Реализации. Поскольку задача о покрытии множества более популярна и легче реша- ется. чем задача укладки множества, для ее решения проще найти подходящую реали- зацию. Рассматриваемые в разделе 18.1 реализации без труда поддаются модификации. что позволяет соблюдать конкретные ограничивающие условия задачи упаковки мно- жества.
638 Часть II. Каталог алгоритмических задач Реализация на языке Pascal алгоритма исчерпывающего перебора для решения задачи упаковки множества, а также для задачи о покрытии множества дается в книге [SDK83]. Информацию по загрузке этих программ можно найти в разделе 19.1.10. Пакет SYMPHONY содержит процедуру для решения задачи разбиения множества методами смешанного линейного программирования. Загрузить ее можно с веб-сайта http://branchandcut.org/SPP. Примечания Среди обзорных статей, посвященных задаче укладки множества, можно назвать такие, как [ВР76] и [Pas97], Стратегии предложения цены в аукционах, в которых товары пред- лагаются в наборах партий, обычно сводятся к решению задач укладки множества, как описывается в работе [dVV03]. Ослабленные ограничивающие условия для задач целочисленного программирования, со- ответствующих задаче укладки множества, представлены в работе [BWOO]. Отличное об- суждение алгоритмов и правил сведения для задачи о покрытии множества представлено в книге [SDK83]. В ней же вы найдете описанное ранее приложение комплектования эки- пажей самолетов. Родственные задачи. Независимое множество (см. раздел 16.2), вершинное покрытие (см. раздел 16.3). 18.3. Сравнение строк Вход. Текстовая строка / из п символов, строка-образецр длиной в т символов. Задача. Найти в текстовой строке / первое (или все) вхождение строки-образца р (рис. 18.4). " You will always have my love, my love, for the love I love is lovely as love itself." |love ? " You will always have my | love my love , for the | love 11 love is| love ly as | love itself." вход выход Рис. 18.4. Поиск в строке по образцу Обсуждение. Задача сравнения строк возникает почти во всех приложениях обработки текста. Любой текстовый редактор имеет механизм поиска произвольной строки в те- кущем докумензе. Языки программирования Perl и Python обладают широкими воз- можностями поиска подстрок благодаря наличию встроенных примитивов сравнения строк, и это позволяет писать на них программы, способные фильтровать и модифици- ровать текст. Наконец, программы проверки правописания ищут в своем словаре каж- дое слово проверяемого текста и помечают слова, отсутствующие в словаре.
Гпава 18. Множества и строки 639 При выборе правильного алгоритма сравнения строк для конкретного приложения воз- никает несколько вопросов. ♦ Какова длина образцов для поиска? При недлинных образцах и нечастых запросах на поиск достаточно использовать простой алгоритм поиска со временем исполне- ния О(тп). Для каждой возможной начальной позиции 1 <i<n-m + 1 в строке по- иска выполняется проверка подстроки длиной т символов на идентичность со стро- кой-образцом. Реализация этого алгоритма на языке С приводится в разделе 2.5.3. При очень коротких образцах (т < 5) нет смысла пытаться улучшить этот простой алгоритм в надежде на повышение производительности. Кроме этого, для типичных строк ожидаемое время исполнения будет намного лучше, чем О(тп), т. к. мы про- двигаем образец дальше, как только обнаруживаем его несовпадение с подстрокой текста. Более того, этот простой алгоритм обычно исполняется за линейное время. Впрочем, наихудший случай вполне вероятен, — возьмем, например, образец/? = а и текст t = (cTW'. ♦ Как поступать с длинными текстами и образцами? В действительности, поиск строк может осуществляться за линейное время в наихудшем случае. Обратите внимание, что при обнаружении несовпадающих символов нет необходимости во- зобновлять поиск с начала, т. к. префикс образца и текст одинаковы вплоть до точки несовпадения. Имея длинное частичное совпадение, заканчивающееся в позиции /, мы переходим на первый символ в образце или тексте, который может предоставить новую информацию о тексте в позиции / + 1. Алгоритм Кнута-Морриса-Пратта вы- полняет предварительную обработку образца для эффективного создания такой таб- лицы переходов. Подробности довольно сложны, но имеющийся алгоритм позволя- ет создавать короткие, простые программы. ♦ Велика ли вероятность найти совпадение с образцом? Алгоритм Бойера-Мура сравнивает образец с текстом справа налево, что позволяет избежать просмотра больших фрагментов текста при отсутствии совпадения. Допустим, что для образца поиска используется строка abracadabra, а в одиннадцатой позиции в тексте нахо- дится буквах. Этот образец не может совпасть с первыми одиннадцатью символами текста, поэтому следующей точкой проверки в тексте является двадцать второй символ. Если нам повезет, то проверить придется всего лишь п/т символов. В слу- чае несовпадения алгоритм Бойера-Мура использует два набора таблиц переходов: один построен на основе текущих совпадений, а второй — на символе, вызвавшем несовпадение. Хотя этот алгоритм сложнее, чем алгоритм Кнута-Морриса-Иратта. он оправдывает себя на практике для строк-образцов длиной свыше пяти символов, при условии от- сутствия частых вхождений образца в текст поиска. Время исполнения алгоритма в наихудшем случае равно О(п + гт), где г— количество вхождений образца р в текст /. ♦ Будет ли выполняться повторный поиск в одном и том же тексте? Допустим, что вы создаете программу для многократных запросов на поиск в некоторой текстовой базе. Так как текст поиска не изменяется, имеет смысл создать структуру данных, позволяющую ускорить выполнение запросов. Подходящими структурами данных
640 Часть II. Каталог алгоритмических задач для этой цели будут суффиксные деревья и суффиксные массивы, рассматриваемые в разделе 12.3. ♦ Планируется ли поиск одного и того же образца в разных текстах? Допустим, что вы создаете программу, чтобы отфильтровывать нецензурные выражения из текста. В этом случае набор образцов остается постоянным, а текст поиска может меняться. В таких приложениях может потребоваться найти все вхождения в текст любого из к разных образцов, при этом значение к может быть довольно большим. Линейное время для поиска каждого образца означает алгоритм с общим временем исполнения ()(к(т + /?)). Для больших значений к существует лучшее решение, соз- дающее один конечный автомат, который распознает эти образцы и возвращается в соответствующее начальное состояние при любом несовпадении символов. Алго- ритм Ахо-Корасика создает такой автомат за линейное время. Оптимизируя автома- ты распознавания образов (см. раздел 18.7}, можно достичь экономии памяти. Этот подход был использован в первоначальной версии программы fgrep. Иногда несколько образцов можно указать не в виде списка строк, а сжато, в форме регулярного выражения. Например, регулярное выражение а(а + h + с)*а соответст- вует любой строке алфавита (а, Ь, с), которая начинается и заканчивается на букву а Самый лучший способ проверки, описывается ли входная строка данным регу- лярным выражением R, заключается в создании конечного автомата, эквивалентно- го R, с последующей эмуляцией этого автомата на данной строке. Подробности соз- дания автомата на основе регулярных выражений приводятся в разделе 18.7. Когда вместо регулярных выражений для указания образцов используются контек- стно-свободные грамматики, задача превращается в задачу синтаксического разбора (см. развел 8 6). ♦ Как поступить, если текст или образец содержит орфографические ошибки? Рас- сматриваемые здесь алгоритмы пригодны только для точного сравнения строк. Если необходим допуск на орфографические ошибки, то задача превращается в задачу нечеткого сравнения строк (см. раздел 18.4). Реализации. Программы на языке С коллекции strmat реализуют алгоритмы для точ- ного сравнения образцов, включая реализацию нескольких вариантов алгоритмов Кну- та-Морриса-Пратта и Бойера-Мура. Самым лучшим справочным материалом по этим алгоритмам является книга [Gus97], Загрузить пакет можно с веб-сайта http:// vvwvv.cs.ucdavis.edu/~gusfield/strmat.html. Пакет для распознавания образов SPARE Parts, написанный на языке C++ (см. [WC04a]), предоставляет пригодные для коммерческого использования реализации всех основных вариантов классических алгоритмов сравнения строк (алгоритмы Кну- та-Морриса-Пратта и Бойера-Мура), как для одного образца, так и для нескольких (ал- горитмы Ахо-Корасика и Комменца-Вальтера). Загрузить пакет можно с веб-сайта http://vvvvvv.fastar.org/. Свободно доступны несколько версий программы сравнения регулярных выражений grep. Вариант GNU программы grep можно найти по адресу http://directory.fsf.org/ project/grep/; эта версия заменяет такие предыдущие версии программы, как egrep и fgrep. Версия GNU grep использует гибрид быстрого детерминистического алгоритма
Глава 18 Множества и строки 641 отложенных состояний с алгоритмом Бойера-Мура для поиска строк фиксированной длины. Библиотека Boost содержит реализации на языке C++ алгоритмов обработки строк, включая поиск. Примечания Все книги по алгоритмам обработки строк, включая такие как [CHL07], [NR07] и [Gus97], содержат исчерпывающее обсуждение точного сравнения строк. Хорошие описания алго- ритмов Бойера-Мура (см. [ВМ77]) и Кнута-Морриса-Пратта (см. [КМР77]) представлены в книгах [BvG99], [CLRSOl] и [Мап89] В истории создания алгоритмов сравнения строк были и неудачи, — некоторые из опубликованных работ содержали ошибки. Подробности см. в книге [Gus97], В книге [Aho90| дается хороший обзор алгоритмов поиска образцов в строках, в частно- сти. алгори тмов поиска с помощью регулярных выражений. Алгоритм Ахо-Корасика опи- сывается в работе [АС75]. Эксперименты по сравнению алгоритмов обработки строк представлены в книгах [DB86], [Ног80], [Lec95] и [dVS82] Качество работы каждого конкретного алгоритма зависит от свойств строк и размера алфавита. Для работы с длинными образцами и большими тек- стами я рекомендую использовать самые лучшие реализации алгоритма Бойера-Мура, ко- торые вы сможете найти. В алгоритме Карпа-Рабина (см. [KR87]) для сравнения строк используется хэш-функция, что позволяет получить линейное ожидаемое время исполнения. Но время исполнения этого алгоритма в наихудшем случае остается квадратичным, а его производительность на практике оказывается несколько хуже, чем у описанных ранее методов сравнения симво- лов. Этот алгоритм обсуждается вразоеле 3.7.2. Родственные задачи. Суффиксные деревья (см. раздел 12.3), нечеткое сравнение строк (см. раздел 18.4) 18.4. Нечеткое сравнение строк Вход. Текстовая строка / и строка-образец р. Задача. Найти самый дешевый способ преобразования строки I в строку р. в котором используются вставки, удаления и замены символов (рис. 18.5). Обсуждение. Задача нечеткого сравнения строк является одной из фундаментальных задач, т. к. мы живем в мире, где нет четких границ. Программы проверки правописа- ния должны находить наиболее подходящее слово для любой строки символов, отсут- ствующей в словаре (т. е. базе данных образцов). Предоставляя поддержку эффектив- ного поиска сходных последовательностей в больших базах данных последовательно- стей ДНК. компьютерная программа BLAST коренным образом изменила направление исследований в молекулярной биологии. Допустим, что вы исследовали некоторый ген человека, и оказалось, что он аналогичен гену, вырабатывающему гемоглобин в орга- низме крысы. Скорее всего, исследуемый ген также отвечает за выработку гемоглоби- на, а разница между этими двумя генами является результатом эволюционных му- таций. 21 Зак. 3741
642 Часть II. Каталог алгоритмических задач misplace misspelled mislead m i s р е 1 d misspelled вход выход Рис. 18.5. Нечеткое сравнение строк Мне однажды пришлось столкнуться с задачей нечеткого сравнения строк при оценке качества работы системы оптического распознавания текста. В ней нужно было срав- нить ответы, выдаваемые системой для тестового документа, с правильными результа- тами. Чтобы улучшить систему, нам требовалось идентифицировать ошибочно распо- знаваемые буквы и "мусор", т. е. несуществующие буквы. Было принято решение вы- полнить нечеткое сравнение ответов и тестовых документов. В таком случае вставки и удаления соответствовали бы "мусору", а замены указывали бы на ошибки в системе распознавания. Аналогичный принцип используется в программах сравнения файлов, которые выявляют строки, отличающиеся в двух разных версиях файла. Когда ошибки недопустимы, задача сводится к точному сравнению строк, которая рас- сматривается в разделе 18.3. Здесь же мы будем рассматривать только строки с ошиб- ками. Динамическое программирование предоставляет нам базовый подход к нечеткому сравнению строк. Пусть D[i.J] обозначает стоимость редактирования первых i симво- лов строки образца р в первые j символов текста t. Отсюда следует существование ре- куррентного соотношения, т. к. мы должны были что-то сделать с концевыми симво- лами р, и t, У нас имелись следующие возможности: зафиксировать совпадение или заменить один символ другим, удалить р, или, наконец, вставить символ, совпадающий с Таким образом, значение D[i,j] является минимальным из следующих трех значе- ний стоимости: ♦ если р, = то £>[/ - 1, j - 1 ], иначе D[i - 1, j - 1 ] + стоимость замены; ♦ D[i - 1, j] + стоимость удаления р,: ♦ D[i,j - 1 ] + стоимость удаления t,. Общая реализация этого алгоритма на языке С и более подробное обсуждение приво- дятся в разделе 8.2. Прежде чем использовать это рекуррентное соотношение, ответьте на несколько вопросов. ♦ Образец должен совпасть со всем текстом или только с подстрокой? Разница между алгоритмами сравнения строк и алгоритмами сравнения подстрок определя- ется граничными условиями этого рекуррентного соотношения. Допустим, что мы хотим сравнить весь образец со всем текстом. Тогда стоимость £>[/, 0] должна рав-
Глава 18. Множества и строки 643 няться стоимости удаления первых / символов образца, поэтому £>[/, 0] = i. Подоб- ным образом. £)[0. /] =j. Теперь допустим, что образец может встретиться в любом месте текста. Коррект- ным значением стоимости £)[0,/] будет 0, т. к. в этом случае не должно быть штра- фа за то, что выравнивание начинается в j-й позиции текста. Стоимость D[i. 0J по- прежнему остается равной /, т. к. единственным способом получения совпадения первых i символов образца с пустой строкой является удаление всех этих символов. Стоимость самого лучшего совпадения образца с подстрокой будет min"^ D\m,k}. ♦ Как определить стоимость операций замены, вставки и удаления? Базовый алго- ритм можно с легкостью модифицировать, установив разную стоимость вставки, удаления и замены символов. Конкретное значение стоимости каждой операции за- висит от того, что вы намереваетесь делать с выровненными текстами. Самым распространенным вариантом является назначение одинаковой стоимости для операций вставки, удаления и замены. Более высокая стоимость замены, чем суммарная стоимость вставки и удаления гарантирует, что замена никогда не будет выполнена, т. к. редактирование вне строки обойдется дешевле. Если выполняются только операции вставки и удаления, то задача сводится к задаче поиска макси- мальной общей подстроки, рассматриваемой в разделе 18.8. Часто бывает полезно слегка изменить стоимость расстояний редактирования и исследовать полученные результаты, повторяя процесс до тех пор, пока не будут определены наилучшие па- раметры для данной задачи. ♦ Как узнать, какие операции фактически привели к выравниванию строк? Ранее бы- ло сказано, что рекуррентное соотношение дает только стоимость оптимального выравнивания образца с текстом, но не последовательность операций редактирова- ния, приводящую к этому выравниванию. Получить список операций можно, двига- ясь в обратном направлении от матрицы полной стоимости D. Чтобы добраться до ячейки D[m. н], мы должны были выйти из ячейки D[m -1, и] (удаление образ- ца/вставка текста), D|/«, п- 1] (удаление текста/вставка образца) или D[m- I, и- 1] (замена/совпадение). Опцию, которая была выбрана фактически, можно выяснить по этим значениям стоимости и по символам р,„ и Продолжая двигаться в обрат- ном направлении к предыдущей ячейке, можно восстановить все операции редакти- рования. Реализация этого алгоритма на языке С приводится в разделе 8.2. ♦ Как поступить, если обе строки очень похожи друг на друга? Алгоритм динамиче- ского программирования позволяет найти кратчайший путь в решетке размером т х п. где стоимость каждого ребра зависит от представляемой им операции. Чтобы найти выравнивание, включающее в себя комбинацию из не более чем d вставок, удалений и замен, нужно только обойти полосу из O(dn) ячеек на расстоянии d в каждую сторону от главной диагонали. Если в этой полосе нет выравнивания низ- кой стоимости, его нет и во всей матрице стоимости. Можно также использовать фильтрацию — быстрое удаление тех участков строки, в которых образец наверняка отсутствует. Разбиваем образец длиной т символов на d + 1 частей. Если существует соответствие, имеющее максимум d отличий, то хотя бы одна из этих частей имеет точное совпадение в оптимальном выравнивании. Та-
644 Часть II. Каталог алгоритмических задач ким образом, мы можем идентифицировать все возможные точки приблизительных совпадений, выполняя точный поиск по нескольким фрагментам образца, а затем более внимательно рассмотретыолько потенциальные кандидаты. ♦ Насколько длинной является строка-образец? Недавно появился новый подход к сравнению строк, в котором используется то обстоятельство, что современные ком- пьютеры могут выполнять операции на словах длиной в 64 бита. В таком машинном слове можно разместить восемь 8-битовых символов ASCII, что стимулирует разра- ботку алгоритмов с параллелизмом на уровне битов, в которых за одну операцию выполняется несколько сравнений. Основная идея далеко не тривиальна. Для каждой буквы а алфавита создаем бито- вую маску Ви, в которой /-й бит B„[i] равен 1 тогда и только тогда, когда /-м симво- лом образца является а. Допустим теперь, что имеется такой битовый вектор совпа- дения М, для /-й позиции в текстовой строке, что А/Д/] = 1 тогда и только тогда, ко- гда первые / битов образца точно совпадают с символами текста, начиная с (/’-/ + 1)-го п заканчивая /-м. Мы можем найти все биты А/,, । посредством лишь двух операций, а именно, сдвинув вектор М, на один бит вправо, а потом выполнив для него побитовую операцию И с маской В„. где а— символ в (j + 1 )-й позиции текста. Такой алгоритм с параллелизмом на уровне битов, обобщенный до нечеткого срав- нения, используется в рассматриваемой далее программе agrep. Подобные алгорит- мы легко поддаются реализации, а работают они во много раз быстрее алгоритмов динамического программирования. ♦ Как минимизировать требования к памяти? Квадратичный обьем памяти, требуе- мый для хранения таблицы динамического программирования, представляет про- блему. более серьезную, чем время исполнения соответствующих алгоритмов. К счастью, для вычисления D[m. и] требуется только C?(inin(/n,«)) памяти. Для вы- числения окончательного значения нужно сопровождать только две активные стро- ки (или столбца) матрицы. Вся матрица понадобится только в том случае, когда по- требуется воссоздать последовательность операций, в результате которых было по- лучено данное выравнивание. Для эффективного вычисления за линейное время оптимального выравнивания можно использовать рекурсивный алгоритм Хиршберга. За один проход описанного выше алгоритма с линейной сложностью по памяти для вычисления D[m, /г] мы вы- ясняем, какая ячейка среднего элемента D[inl2, х] была использована для оптимиза- ции D\m, л]. Это сводит нашу задачу к задаче поиска наилучших путей от £>[1. 1] до О[х/2, х] и от D[m/2, х] до D[m/2, и], причем оба пути могут быть найдены рекурсив- но. Каждый раз мы исключаем из рассмотрения половину элементов матрицы, по- этому общее время остается равным О(тп). Этот алгоритм с линейной сложностью по памяти обладает хорошей производительностью на длинных строках, но про- граммировать его затруднительно. ♦ Сцедует ли по-разному оценивать многократные повторы операций вставки и уда- ления? Во многих приложениях сравнения строк оказывается предпочтение вырав- ниваниям. в которых операции вставки и/или удаления сгруппированы в небольшое количество последовательностей. Удаление слова из текста, по идее, должно стоить
Гпава 18. Множества и строки 645 меньше, чем удаление соответствующего количества одиночных символов, т. к. при этом выполняется одна (хотя и сложная) операция редактирования. Применение штрафор за появление разрывов при сравнении строк позволяет кор- ректно учитывать такие операции. Обычно для каждой операции вставки/удаления / последовательных символов устанавливается стоимость А + Bl. где А — стоимость создания разрыва, а В— стоимость удаления одного символа. Если значение А ве- лико по сравнению со значением В. то при выравнивании появится стимул выпол- нять относительно небольшое количество последовательных операций удаления. Сравнение строк при таких аффинных штрафах за разрывы можно выполнять за квадратичное время, как и обычное вычисление расстояния редактирования. Мы будем использовать различные рекуррентные соотношения Е и F для операций вставки и удаления, чтобы вычислить стоимость пребывания в режиме разрыва, предполагая, что цена за создание разрыва уже заплачена: l'\i./) = шах(£(/. /). F(i,j), C(i.j)) G(i.j) = !'(/- I.j - I) + match(Z.y) E(i,j) = max(£(q- I), ly.j- Г)-А)-В F(i.j) = max(F(/ - l,j). f’(/ - l.j) -A) - В При постоянном времени обращения к каждой ячейке время исполнения этого ал- горитма равно ()(тп). ♦ Считаются чи похожими строки с одинаковым произношением? Для некоторых приложений лучше подходят другие модели нечеткого сравнения строк. Особый интерес представляет схема хэширования Soundex, в которой одинаково звучащим словам присваивается одинаковый индекс. Эта схема может быть полезной при проверке, не являются ли два по-разному написанных слова в действительности од- ним и тем же словом. Например, мою фамилию часто пишут как "Skina", "Skinnia". Schiena", а иногда и "Skiena". Всем этим по-разному написанным словам присваива- ется одинаковый индекс Soundex — S25. Алгоритм Soundex отбрасывает гласные, непроизносимые и повторяющиеся буквы, а потом назначает оставшимся буквам числа в соответствии со следующими клас- сами: BFPV— I, CGJKQSXZ— 2. DT —3. L — 4, MN — 5. R —6. Код слова со- стоит из первой буквы, за которой следуют максимум три цифры. Хотя такой под- ход кажется не очень естественным, на практике он работает довольно хорошо. А практика уже довольно продолжительная — Soundex используется с 1920-х годов. Реализации. Для нечеткого сравнения строк существует несколько отличных про- граммных средств. Программа agrep (см. [WM92a, WM92b]) поддерживает поиск в тексте с орфографическими ошибками. Последнюю версию этой программы можно загрузить с веб-страницы http://wwvv.tgrics,de/agrep. Программа nrgrep (см. [NavOlb]) сочетает параллелизм на уровне битов с фильтрацией и имеет постоянное время ис- полнения, хотя не всегда работает быстрее, чем agrep. Загрузить программу можно с веб-сайта http:/Avww.dcc.uchilc.cl/~gnavarro/sofbvare/. Библиотека TRE поиска совпадений с регулярными выражениями предназначена для поиска точных и приблизительных совпадений и обладает более широкими возможно-
646 Часть II. Каталог алгоритмических задач стями, чем программа agrep. Временная сложность программы в худшем случае равна О(пт~). где т— длина списка используемых регулярных выражений. Загрузить про- грамму можно с веб-сайта http://www.dcc.uchile.cl/~gnavarro/software/. В Wikipedia можно найти программы для вычисления расстояния редактирования, в частности, расстояния Левенштейна, написанные на разных языках (включая Ada, C++, Emacs Lisp, lo, JavaScript, Java, PHP, Python, Ruby, VB и С#). Дополнительную ин- формацию можно найти на веб-сайте http://en.wikibooks.org/wiki/Algorithm_ Implemcntation/Strings/Levenshteindistancc. Примечания В последнее время наблюдаются определенные успехи в области нечеткого сравнения строк, особенно в алгоритмах, применяющих параллелизм на уровне битов. Алгоритмам обработки строк посвящены книги [CHL07] и [Gus97], а самым лучшим справочником по методам сравнения строк является [NR07]. Считается, что базовый алгоритм динамического программирования для получения вы- равниваний впервые был описан в работе [WF74], хотя, по-видимому, он был известен и раньше. Широта диапазона применения нечеткого сравнения строк была продемонстри- рована в книге [SK99], которая по сей день является полезным историческим справочни- ком в данной области. Обзоры решений задачи нечеткого сравнения строк представлены в работах [HD80] и [NavOla]. Расстояние редактирования между двумя строками иногда на- зывают расстоянием Левенштейна. Алгоритм Хиршберга с линейной сложностью по па- мяти (см. [Hir75]) рассматривается в книгах [CR03] и [Gus97]. В работе [МР80] представлен алгоритм для вычисления расстояния редактирования меж- ду строками длиной в т и п символов за время O(/w?/log(min{ш, «})) для алфавитов посто- янного размера. В алгоритме использованы идеи из алгоритма четырех русских для ум- ножения булевых матриц (см. [ADKF70]). Формулировка задачи нечеткого сравнения строк в терминах поиска кратчайшего пути позволяет создать множество алгоритмов, дающих хорошие результаты для небольших расстояний редактирования, включая алгоритмы с временем исполнения O(«lg« +г/2) (см. [Муе86]) и O(dn) (см [LV88]) Максимальную возрастающую последовательность можно вычислить за время O(«lg«) (см. [HS77]), как описано в [Мап89]. В числе алгоритмов нечеткого сравнения строк, применяющих параллелизм на уровне би- тов, можно назвать алгоритм Майерса (Myers) (см. [Муе99Ь]) с временем исполнения О(тп/м), где ir— количество бит компьютерного слова. Результаты экспериментальных исследований алгоритмов с параллелизмом на уровне битов представлены в работах [FN04], [HFN05] и [NR,OO], Система Soundex была изобретена и запатентована Оделл (М. К. Odell) и Расселлом (R. С. Russell). Описание этой системы можно найти в работах [BR95] и [Кпи98]. Недавняя раз- работка Metaphone (см. [BR95] и [Раг90]) представляет собой попытку создания системы, превосходящей Soundex. Применение таких систем фонетического хэширования для уни- фикации названий задач рассматривается в работе [LMS06]. Родственные задачи. Поиск строк (см. раздел 18.3). поиск максимальной общей под- строки (см. раздез 18.8).
Гпава 18. Множества и строки 647 18.5. Сжатие текста Вход. Текстовая строка S. Задача. Преобразовать строку S в более короткую строку S', из которой можно пра- вильно воссоздать исходную строку 5' (рис. 18.6). 1 oursum* and seven у eats ag«> our lather brought forth nil this continent a new nation conceived in Liberty and dedicated to the proposition that «ill men aiv created equal Now we are engaged in a great civil war testing whether that nation or any nation so conceived and so dedicated can long endure We arc met on a great battlefield ot that wai. Wc have come to dedicate a portion <>l that field as a final resting place Гог those who here gave their lives that the nation might live. It is altogether fitting and we can not consecrate we can not hallow this groud. The brave men living «ind dead who struggled here have consecrated il for above our poor power in add or detract. The world will little note hoi long remember what we say here but il can never loigct what they did here. It U> lor us the living here have thus lar.su nobly advanced. It is rather for us to be here dedicated to the gicat tusk remaining before us that hum these hoiioied dead we take increased devotion to that cause tor which they here gave the List lull measure of devotion that we here highly icsolve that these dead shall not have died in vam that this nation under God shall have a new birth of freedom and that government of the people by the people Гог the people shall not perish Irom the cailh. вход выход Рис. 18.6. Сжатие текста Обсуждение. Хотя емкость внешних устройств хранения данных удваивается каждый год, ее все время не хватает. Снижение цен на устройства хранения ничуть не умень- шило интерес к сжатию данных, вероятно, потому что данных, подлежащих сжатию, становится все больше. Сжатие данных является алгоритмической задачей, в которой требуется найти экономичную кодировку для файла указанного типа. Развитие компь- ютерных сетей поставило новую цель перед задачей сжатия данных— повышение эф- фективной пропускной способности сети посредством уменьшения количества переда- ваемых битов. Создается впечатление, что разработчикам нравится изобретать специальные методы сжатия данных для каждого конкретного приложения. Иногда производительность этих специализированных методов даже превосходит производительность методов общего назначения. При выборе правильного алгоритма сжатия возникает несколько вопросов. ♦ Требуется ли точное восстановление исходных данных после сжатия? Основной вопрос, возникающий при сжатии данных, заключается в том, выполнять сжатие с потерями или без потерь. Приложения для хранения документов обычно требуют обратимого сжатия (без потерь), т. к. пользователи не хотят, чтобы их данные под-
648 Часть II. Каталог алгоритмических задач вергались изменениям. При сжатии изображений или видеофайлов точность данных не настолько важна, т. к. небольшие погрешности незаметны для зрителя. Примене- ние сжатия с потерями позволяет получить значительно более высокую степень сжатия, вследствие чего этот тип сжатия применяется в большинстве приложений для сжатия аудио- и видеоданных и изображений. ♦ Можно ли упростить данные перед сжатием? Самым эффективным способом увеличения свободного дискового пространства является удаление ненужных фай- лов. Аналогичным образом любая предварительная обработка, уменьшающая объем информации в файле, приводит к более эффективному сжатию. Постарайтесь выяс- нить. можно ли удалить из файла избыточные пробельные символы, преобразовать все буквы текста в заглавные или снять форматирование. Особый интерес представляет упрощение данных с помощью преобразования Бар- роуза-Уитри. По ходу этого преобразования выполняется сортировка всех п цикли- ческих сдвигов «-символьной входной строки, а затем возвращается последний символ каждого сдвига. Например, строка АВАВ имеет циклические сдвиги АВАВ, ВАВА, АВАВ и ВАВА. После сортировки получаем строки АВАВ, АВАВ, ВАВА и ВАВА. Считывая последний символ каждой из этих строк, получаем результат пре- образования — ВВАА. При условии, что последний символ входной строки уникален (например, представ- ляет собой символ конца строки), это преобразование является полностью обрати- мым. Строка, предварительно обработанная с помощью преобразования Барроуза- Уилера. сжимается на 10-15% лучше, чем первоначальный текст, т. к. повторяю- щиеся слова превращаются в блоки повторяющихся символов. Кроме этого, это преобразование можно осуществить за линейное время. ♦ Что делать, если алгоритм сжатия запатентован? Некоторые алгоритмы сжатия данных были запатентованы. Одним из самых известных примеров является рас- сматриваемая далее версия LZW алгоритма Лемпеля-Зива. К счастью, в настоящее время срок действия этого патента истек, чего нельзя сказать об алгоритме сжатия JPEG, который все еще является предметом судебных разбирательств. Обычно у любого алгоритма сжатия имеются версии, которыми можно пользоваться без огра- ничений, причем работают они так же хорошо, как и запатентованный вариант. ♦ Как сжимать изображения? Самым простым обратимым алгоритмом сжатия изо- бражений является кодирование длин серий. При таком способе сжатия последова- тельности одинаковых пикселов (серии) заменяются одним экземпляром пиксела и числом, указывающим длину последовательности. Этот метод хорошо работает с двоичными представлениями изображений, имеющими большие области одина- ковых пикселов, например, с отсканированным текстом. Но на изображениях, имеющих много градаций цвета и/или содержащих шум. производительность пада- ет. Сильное влияние на качество сжатия оказывают два фактора: выбор размера по- ля, содержащего длину последовательности, и выбор порядка обхода при преобра- зовании двумерного изображения в поток пикселов. Для профессиональных приложений сжатия аудиоданных, видеоданных и изобра- жений я рекомендую использовать один из популярных методов сжатия с потерями и не пытаться реализовать свой собственный. Стандартным высокопроизводитель-
Гпава 18. Множества и строки 649 ным методом сжатия изображений является JPEG, a MPEG предназначен для сжа- тия видеоданных. ♦ Должно ли сжатие выполняться в реальном времени? Быстрое восстановление данных нередко оказывается важнее, чем их быстрое сжатие. Например, видеодан- ные на YouTube сжимаются только один раз, но восстанавливаются при каждом его воспроизведении. Впрочем, существует и противоположный пример. Операцион- ной системе для увеличения эффективной емкости диска посредством сжатия файлов требуется симметричный алгоритм, обладающий также быстрым временем сжатия. Существуют десятки алгоритмов для сжатия текста, но их можно разбить на две груп- пы в соответствии с применяемым подходом. Статические алгоритмы, такие как ме- тод Хаффмана, создают одну кодовую таблицу, исследуя полностью весь документ. Адаптивные алгоритмы. такие как алгоритм Лемпеля-Зива, динамически создают ко- довую таблицу, которая адаптируется под локальное распределение символов доку- мента. В большинстве случаев следует предпочитать адаптивный алгоритм. ♦ Код Хаффмана. Алгоритм Хаффмана заменяет каждый символ алфавита кодовой строкой битов переменной длины. Использование восьми биг для кодирования каждой буквы неэкономично, т. к. некоторые символы (например, буква "е") встре- чаются намного чаще, чем другие (например, буква "q"). При кодировании методом Хаффмана букве "е" присваивается короткий код. а букве "q" — бочее длинный, та- ким образом сжимая текст. Код Хаффмана можно создать, используя "жадный" алгоритм. Сначала сортируем символы в порядке возрастания частоты их вхождения в текст. Два самых редких символа х и у объединяются в один новый символ ху с частотой вхождения, равной сумме частот вхождения двух его родительских символов. Таким образом мы полу- чаем меньший набор символов. Эта операция повторяется и- I раз. пока все симво- лы не будут слиты попарно. Посредством таких операций объединения строится корневое двоичное дерево, в котором исходные символы алфавита представлены листьями. Биты слова двоичного кода для каждого символа определяются выбором правого или левого пути в проходе от корня к листу. При написании программы упорядочивания символов по частоте использования можно применять очереди с приоритетами, что позволит выполнять эту операцию за время (7(nlgn). Несмотря на свою популярность, алгоритм Хаффмана имеет три недостатка. Во- первых, для того чтобы закодировать документ, по нему необходимо выполнись два прохода — первый, чтобы создать таблицу кодирования, а второй, чтобы выполнить само кодирование документа. Во-вторых, таблицу кодирования нужно явно сохра- нять вместе с закодированным документом, чтобы можно было декодировать его, и это приводит к неэкономному расходу памяти при кодировании коротких докумен- тов. Наконец, алгоритм Хаффмана эффективен при неравномерном распределении символов, в то время как адаптивные алгоритмы способны распознавать повторяю- щиеся последовательности, например, 0101010101. ♦ Алгоритмы Лемпеля-Зива. Алгоритмы Лемпеля-Зива (включая популярный вариант LZW) сжимают текст, создавая таблицу кодирования по ходу чтения документа. Эта таблица корректируется на каждой позиции в тексте. Благодаря использованию хо-
650 Часть II. Каталог алгоритмических задач рошо продуманного протокола, программы котирования и декодирования работают с одной и той же таблицей, что позволяет избежать потери информации. Алгоритмы Лемпеля-Зива создают кодовые таблицы часто встречающихся под- строк. размер которых может быть очень большим. Таким образом алгоритмы этого типа могут использовать часто встречающиеся слоги, слова и фразы, чтобы выпол- нить кодирование наилучшим образом. Алгоритм приспосабливается к локальным изменениям в распределении элементов текста, что является важным, т. к. многие документы обладают значительной пространственной локальностью. Замечательным свойством алгоритма Лемпеля-Зива является его устойчивость при обработке данных разных типов. Разработать для конкретного приложения специа- лизированный алгоритм, превосходящий алгоритм Лемпеля-Зива. довольно трудно. Я рекомендую не предпринимать таких попыток. Если имеется возможность устра- нить специфическую для приложения избыточность на стадии предварительной об- работки. вое пользуйтесь ею. Но не тратьте понапрасну время, пытаясь создать соб- ственный алгоритм сжатия. Вряд ли вам удастся получить значительно лучшие ре- зультаты. чем выдает программа gzip или другая распространенная утилита. Реализации. Самой популярной программой сжатия текста, вероятно, является про- грамма gzip. которая реализует публично доступный вариант алгоритма Лемпеля-Зива. Эта программа распространяется с лицензией GNU; загрузить ее можно с веб-сайта http://www.gzip.org. Всегда приходится искать разумный компромисс между коэффициентом сжатия и вре- менем работы алгоритма. Альтернативой программе gzip является программа сжатия bzip2. в которой используется преобразование Барроуз-Уилера. Она обеспечивает луч- ший коэффициент сжатия, чем программа gzip, но за счет увеличения времени испол- нения. Создатели некоторых алгоритмов впадают в крайность, жертвуя скоростью ради повышенной компактности. Типичные представители таких алгоритмов собраны на веб-сайте http://www.cs.fit.edu/~ninialioney/conipression. Авторитетное сравнение программ сжатия, включающее в себя ссылки на все доступное программное обеспечение, представлено на веб-сайте http://www.maxiniiimcompression.coni/. Примечания Существует большое количество книг, посвященных сжатию данных. Из недавно вышед- ших можно назвать книги [SayO5] и [Sal06], Также можно порекомендовать уже не новую книгу [ВС W90]. Обзор алгоритмов сжатия текстов приведен в работе [СЕ98]. Хорошее описание алгоритма Хаффмана (см [Huf52]) можно найти в книгах [AHU83], [CLRS] и [Мап89]. Алгоритм Лемпеля-Зива и его варианты описаны в работах [Wel84] и [ZL78]. Преобразование Барроуз-Уилера было представлено в работе [BW94], Главным форумом по обмену информацией в области сжатия данных является ежегодная тематическая конференция института IEEE. Подробности см. на веб-сайте http:// www.cs.brandeis.cdu/~dcc. Сжатие данных является хорошо развитой технической обла- стью, и в последнее время усилия разработчиков сосредоточены на весьма незначитель- ных улучшениях алгоритмов, особенно в случае сжатия текстовых данных. Однако радует то обстоятельство, что ежегодная конференция IEEE проводится на горнолыжном курор- те мирового класса в штате Юта.
Глава 18. Множества и строки 651 Родственные задачи. Поиск минимальной общей надстроки (см. раздел 18.9). крипто- графия (см. раздел 18.6). 18.6. Криптография Вход. Открытое сообщение Т или зашифрованный текст Е и ключ к. Задача. С помощью ключа к зашифровать сообщение Т (расшифровать текст Е). полу- чив в результате зашифрованный текст Е (открытое сообщение Т) (рис. 18.7). The magic words are Squeamish Ossifrage. I5&AE<&UA9VEC'=0 <Fls"F%R92!3<75E96Ul<V V @ *3 W-S :69R86=E+ @ К вход выход Рис. 18.7. Пример шифрования текста Обсуждение. Поскольку широкое распространение компьютерных сетей предоставля- ет все больше возможностей несанкционированного доступа к конфиденциальной ин- формации. криптография играет важную роль в сохранении целостности такой инфор- мации. Криптография повышает безопасность обмена сообщениями, позволяя обраба- тывать их таким образом, что их будет невозможно прочесть, если они попадут в чужие руки. Хотя этой дисциплине, по крайней мере, две тысячи лет, ее алгоритмиче- ские и математические основы лишь недавно сформировались настолько, что появи- лась возможность рассуждать о доказуемо безопасных системах шифрования. Идеи и приложения шифрования выходят за рамки общеизвестных понятий "шифро- вания" и "аутентификации". Сейчас эта область включает такие важные математиче- ские конструкции, как криптографические хэши. цифровые подписи, а также базовые протоколы, предоставляющие гарантии безопасности. Существует три класса систем шифрования: ♦ шифр Цезаря. В самых старых шифрах каждый символ алфавита заменялся другим символом этого же алфавита. В наиболее слабых шифрах выполняется циклический сдвиг алфавита на заданное количество символов (часто 13), вследствие чего они имеют только 26 возможных ключей (речь идет об английском алфавите — прим, перев). Лучше использовать произвольную перестановку букв, что дает 26! возмож- ных ключей. Но даже и в этом случае такие системы шифрования можно с лег- костью взломать, подсчитав частоту появления каждого символа в тексте и приме- нив знание частотного распределения букв в текстах данного алфавита. Хотя суще- ствуют варианты этого шифра, более устойчивые к взлому, ни один из них не будет таким надежным, как шифр AES или RSA; ♦ блочные шифры. В шифрах этого типа биты текста многократно перемешиваются в соответствии с заданным ключом. Классическим примером такого шифра являет-
652 Часть II. Каталог алгоритмических задач ся шифр Data Encryption Standard (DES). Хотя он был одобрен в 1976 г. в качестве Федерального стандарта США для обработки информации, в настоящее время 56-битовый ключ этого шифра считается слишком коротким для приложений, тре- бующих значительного уровня безопасности. В частности, компьютер, созданный специально для взлома этого шифра, продемонстрировал, что этот шифр можно взломать быстрее, чем за один день. С 19 мая 2005 г. шифр DES официально пере- стал быть Федеральным стандартом США и был заменен более сильным шифром Advanced Encryption Standard (AES). Но простой вариант шифра DES под названием "тройной DES" позволяет осущест- влять шифрование с помощью ключа длиной в 112 битов за счет троекратного при- менения шифра DES с двумя 56-битовыми ключами. Для этого текст сначала за- шифровывается ключом keyl. потом расшифровываемся ключом кеу2. а затем опять зашифровывается ключом keyl. Использование трех раундов шифрования вместо двух имеет целью обратную совместимость со старым шифром DES, у которого keyl = кеу2. 11ациональный институт стандартов и технологий США (NIST) недавно одобрил применение тройного DES для шифрования важной правительственной информации вплоть до 2030 г.; ♦ шифрование с открытым ключом. Если вы боитесь, что ваши недоброжелатели читают ваши сообщения, вы также должны держать в секрете ключ, необходимый для их расшифровки. Однако в системах с открытым ключом для шифрования и дешифрования сообщений используются разные ключи. Так как ключ шифрования бесполезен для расшифровки, он может быть открытым, без риска для безопасно- сти. Решение о распределении ключей лежит в основе упеха шифрования с откры- тым ключом. Классическим примером системы шифрования с открытым ключом является систе- ма RSA. названная так в честь ее изобретателей — Инвеста (Rivest), (Памира (Shamir) и Адлемана (Adleman). Безопасность системы RSA основана на вычисли- тельной сложности задач разложения на множители и проверки чисел на простоту (см. раздел 13.8). Процесс шифрования протекает быстро, т. к. для создания ключа используется проверка на простоту, а сложность расшифровки вытекает из сложно- сти разложения на множители. Однако система шифрования RSA работает медлен- но по сравнению с другими системами. В частности, она примерно в 100-1 000 раз медленнее, чем шифр DES. Важнейшим фактором, влияющим на выбор системы шифрования, является требуемый уровень безопасности. От кого вы пытаетесь защитить ваши сообщения — от своей бабушки, квартирных воров, организованной преступности или спецслужб? Если вы можете использовать современную реализацию шифра AES или RSA. вы будете чувст- вовать себя в безопасности, по крайней мере, в ближайшее время. Однако постоянно растущая мощность компьютеров способна на удивление быстро расшатать криптоси- стему. Вспомните, что шифр DES продержался в качестве надежной системы менее тридцати лет. Обязательно используйте ключи максимальной длины и следите за ново- стями в области шифрования, если вы планируете долгосрочное хранение конфиден- циальной информации. Что касается меня лично, признаюсь, что я использую DES для шифрования задач вы- пускного экзамена в конце каждого семестра. Этого шифра оказалось более чем доста-
Глава 18 Множества и строки 653 точно, когда один решительно настроенный студент проник в мой офис в поисках за- дач. Результат был бы иным, если бы взломщиком оказался агент спецслужб. При этом важно понимать, что самые серьезные бреши в безопасности обусловлены человече- ским фактором, а не качеством алгоритма. Использовать достаточно длинный, трудный для отгадывания пароль и не записывать его на бумажке гораздо важнее, чем излишне усердствовать в выборе алгоритма шифрования. При одинаковой длине ключа большинство симметричных систем шифрования труд- нее поддается взлому, чем системы с открытым ключом. Это означает, что для симмет- ричного шифрования можно использовать намного более короткие ключи, чем для шифрования с открытым ключом. Институт NIST и лаборатория RSA предоставляют списки рекомендуемых размеров ключей для безопасного шифрования, и на момент написания этой книги они рекомендуют 80-битовые симметричные ключи, как эквива- лентные 1024-битовым асимметричным. Эта разница объясняет, почему алгоритмы шифрования симметричным ключом работают на несколько порядков быстрее алго- ритмов шифрования с открытым ключом. Простые шифры, вроде шифра Цезаря, легко программируются. По этой причине их можно использовать в приложениях, требующих минимального уровня безопасности, например, для тою. чтобы скрыть ответы на загадки. Поскольку эти шифры легко под- даются взлому, их никогда не следует использовать в приложениях, требующих серь- езного уровня безопасности. А еще никогда не следует пытаться разработать собственную систему шифрования. Безопасность шифров DES и RSA общепризнанна, т. к. они выдержали многолетнее испытание массовым использованием. За это время было предложено много других систем шифрования, которые оказались уязвимыми к взлому. Разработкой алгоритмов шифрования должны заниматься только профессионалы. Если же вам поручили соз- дать систему безопасности, начните с изучения какой-либо признанной программы, например PGP, чтобы разобраться, как в ней решаются вопросы выбора и распростра- нения ключей. Стойкость любой системы безопасности определяется ее самым слабым звеном. На практике часто возникают следующие вопросы, связанные с шифрованием. ♦ Как проверить, что данные не были повреждены случайно? Часто требуется удо- стовериться в том. что при передаче данных они не подверглись случайным иска- жениям, и полученные данные идентичны отправленным. Одно из решений этой за- дачи — передать полученные данные обратно отправителю для подтверждения идентичности двух текстов. Но этот метод не срабатывает, если при обратной пере- даче происходит в точности противоположная ошибка. К тому же. при такой схеме пропускная способность канала уменьшается вдвое, что представляет серьезное не- удобство. Более эффективным методом будет использование контрольной суммы, т. е. хэши- рование длинного текста с помощью простой математической функции в небольшое число и передача этой контрольной суммы вместе с текстом. Принимающая сторона вычисляет контрольную сумму принятого текста и сравнивает ее с первоначальной. В самой простой схеме с контрольной суммой вычисляется сумма байтов текста по модулю некоторой константы, например, 2х = 256. К сожалению, такая схема не вы-
654 Часть II. Каталог алгоритмических задач являет ошибку, при которой несколько символов меняются местами, поскольку сложение коммутативно. Более надежный способ вычисления контрольной суммы включает в себя использо- вание цикчического избыточного кода. Он применяется в большинстве систем связи и в компьютерах для проверки целостности обмена данных с диском. Для получе- ния этой контрольной суммы вычисляется остаток от деления двух многочленов, причем делитель является функцией входного текста. Разработка этих многочленов является сложной математической задачей, но такая контрольная сумма обеспечи- вает обнаружение всех достаточно вероятных ошибок. Подробности эффективной реализации весьма сложны, поэтому рекомендуется начинать с уже существующих реализаций. ♦ Как проверить, что данные не были повреждены преднамеренно? Метод контроля с помощью циклического избыточного кода хорошо распознает случайные искаже- ния, но не годится для выявления преднамеренных. Для таких целей лучше подхо- дят криптографические функции хэширования, такие как MD5 или SHA-256, кото- рые легко вычисляются для любого документа, но являются труднообратимыми. Это означает, что по данному хэш-значению х трудно создать такой документ d, для которого H(d) = x. Это свойство хэш-функций делает их полезными для использова- ния в качестве цифровых подписей и в других приложениях. ♦ Как доказать, что файл не был изменен? Если я вышлю вам контракт в электрон- ной форме, ничто не помешает вам отредактировать файл и утверждать, что ваша версия отражает нашу договоренность. Мне нужен способ доказать, что модифици- рованная версия документа является подделкой. Цифровая подпись представляет собой криптографический способ подтверждения подлинности документа. Я могу вычислить контрольную сумму для данного файла и зашифровать ее с по- мощью закрытого ключа. Вместе с файлом контракта я высылаю его зашифрован- ную контрольную сумму. Вы можете отредактировать файл, но, чтобы обмануть суд. вам надо будет отредактировать и зашифрованную контрольную сумму так, чтобы после ее расшифровки получилась исходная. При использовании достаточно хорошей функции вычисления контрольной суммы создание другого файла с такой же контрольной суммой будет невыполнимой задачей. Для полной безопасности мне понадобятся услуги пользующейся всеобщим доверием третьей стороны, кото- рая удостоверит подлинность цифровой подписи и свяжет закрытый ключ со мной. ♦ Как ограничить доступ к материалам, защищенным авторским правом? Важным новым применением шифрования является управление авторскими правами на цифровые аудио- и видеоматериалы. Принципиальным вопросом здесь является скорость декодирования, т. к. она должна соответствовать скорости пересылки дан- ных в реальном времени. Такие потоковые шифры обычно включают в себя гене- рирование потока псевдослучайных битов с использованием, например, генератора кода на регистре сдвига. Операция исключающего ИЛИ над этими битами и пото- ком данных дает зашифрованную последовательность. Для восстановления исход- ных данных опять выполняется операция исключающего ИЛИ над зашифрованным потоком данных с тем же самым потоком псевдослучайных битов, который исполь- зовался для шифрования.
Глава 18. Множества и строки 655 Было доказано, что высокоскоростные системы потокового шифрования сравни- тельно легко поддаются взлому. Современное решение этой проблемы включает принятие специальных законов и уголовное преследование за их нарушение. Реализации. Криптографическая библиотека Nettle содержит, среди прочего, функции хэширования MD5 и SHA-256. а также блочные шифры DES, AES и некоторые другие. В библиотеку включена и реализация системы RSA. Загрузить библиотеку Nettle мож- но по адресу http://www.lysator.liu.se/~nisse/nettle/. Подробный обзор алгоритмов шифрования, включающий оценку их устойчивости к взлому, доступен на веб-странице http://www.cryptolounge.Org/wiki/Category': Algorithm. На странице http://csrc.nist.gov/groups/ST/toolkit институт NIST представ- ляет коллекцию стандартов и руководств по разработке алгоритмов шифрования. Большая библиотека Crypto++ классов языка C++ схем шифрования содержит все рас- смотренные в этом разделе системы шифрования. Загрузить библиотеку можно с веб- сайта http://ww w .cryptopp.com. Многие популярные утилиты с открытым кодом используют профессиональные схемы шифрования и служат хорошими моделями текущей практики в этой области. Про- грамму GnuPG, версию с открытым кодом программы PGP. можно загрузить по адресу http://www.gnupg.org/. Программу OpenSSL для аутентификации доступа к компью- терным системам можно загрузить по адресу http://www.openssl.org/. Библиотека Boost CRC Library содержит несколько реализаций алгоритмов вычисления контрольной суммы с помощью циклического избыточного кода. Загрузить библиотеку можно с веб-сайта http://www.boost.org/libs/crc/. Примечания В книге [MOV96] приведено описание технических деталей 'всех аспектов криптогра- фии. Электронную версию книги можно найти на веб-сайте http://www.cacr.math. uwaterloo.ca/hac/. В книге [Sch96] содержится всестороннее обсуждение разных алгорит- мов шифрования, но, пожалуй, книга [FS03] является лучшим введением в эту область. В книге [Kah67] увлекательно изложена история криптографии, начиная с древних времен до 1967 г. Обсуждение алгоритма RSA (см. [RSA78]) приводится, в частности, в книге [CLRS01], Обширная информация представлена на домашней странице RSA по адресу http://www.rsa.com/rsalabs/. Главным источником информации о современном состоянии крипто! рафии является Агентство национальной безопасности США (National Security Agency, NSA). История шифра DES хорошо представлена в книге [Scli96], Особенно спорным было решение NSA ограничить длину ключа 56-ю битами. Хэш-функция MD5 (см. [Riv92]) используется в программе PGP для вычисления цифро- вых подписей. Ее описание можно найти, например, в [Sch96] и [Sta06]. Недавно были выявлены серьезные проблемы с безопасностью этой функции (см. [WY05]). Семейство хэш-функций SHA (в частности, функции SHA-256 и SHA-512) производит впечатление более надежного. Родственные задачи. Разложение на множители и проверка чисел на простоту (см. раздел 13.8), сжатие текста (см. раздел 18.5).
656 Часть II Каталог алгоритмических задач 18.7. Минимизация конечного автомата Вход. Детерминированный конечный автомат М Задача. Создать наименьший детерминированный конечный автомат Л-7’, поведение которого идентично поведению автомата М(рис. I8.8). ВХОД ВЫХОД Рис. 18.8. Пример минимизации конечною автомата Обсуждение. Задача создания и минимизации конечных автоматов часто возникает при разработке программного и аппаратного обеспечения. Конечные автоматы широко используются в приложениях, связанных с распознаванием образов. С овременные язы- ки программирования, такие как Java и Python, содержат встроенные средства под- держки регулярных выражений, являющихся самым естественным способом определе- ния автоматов. Конечные автоматы часто применяются в системах управления и ком- пиляторах для кодирования текущего состояния и связанных с ним действий или переходов. Минимизация размера автоматов понижает как затраты по хранению дан- ных, так и стоимость исполнения при использовании таких автоматов. Конечные автоматы определяются посредством ориентированных графов. Каждая вершина представляет состояние, а каждое помеченное символом алфавита ребро оп- ределяет переход из одного состояния в другое при получении данного символа. Пока- занные на рис. 18.8 автоматы анализируют последовательность бросков монеты. За- крашенные вершины обозначают состояние, при котором "орел" выпал четное количе- ство раз. Такие автоматы можно представить, используя граф (см. раздел 12.4) или матрицу переходов размером п х |S|, где Е — размер алфавита. Конечные автоматы часто используются для поиска образцов, представленных в виде регулярных выражений, которые являются результатом выполнения операций И, ИЛИ или повтора над регулярными выражениями меньшего размера. Например, регулярное выражение а(а + b + с)*а соответствует любой строке алфавита (a, h, с), которая начи- нается и заканчивается на букву а. Самый лучший способ проверки, описывается ли входная строка данным регулярным выражением R, заключается в создании конечного автомата, эквивалентного R, с последующей эмуляцией этого автомата на данной строке. Альтернативные подходы к решению задачи поиска строк рассматриваются в разделе 18.3.
Глава 18. Множества и строки 657 Здесь мы рассмотрим три задачи, связанные с конечными автоматами: ♦ минимизация детерминированных конечных автоматов. У сложных конечных ав- томатов матрицы переходов имеют недопустимо большой размер, что вызывает не- обходимость в более "плотных" структурах. Самым простым подходом к решению данной проблемы является устранение повторяющихся состояний автомата. Как по- казано на рис. 18.8, автоматы самого разного размера могут выполнять одну и ту же функцию. Алгоритмы минимизации количества состояний детерминированных конечных ав- томатов можно найти в любом учебнике по теории автоматов. Основной подход к решению этой задачи — приблизительное разбиение состояний на классы эквива- лентности с последующим уточнением этого разбиения. Сначала состояния разде- ляются на три класса: допускающие, отвергающие и все остальные. Теперь перехо- ды из каждого узла ведут к данному классу по данному символу. Каждый раз, когда два состояния х и / из одного и того же класса С переходят в состояния из разных классов, класс С нужно разбить на два подкласса, один из которых содержит со- стояние s, а другой — состояние /. Этот алгоритм перебирает все классы, проверяя необходимость нового разбиения, и, если такая необходимость имеется, повторяет процесс с начала. Время исполне- ния этого алгоритма равно О(и"), т. к. нужно будет выполнить, самое большее, п - 1 проходов. Полученные после последнего прохода классы эквивалентности соответ- ствуют состояниям в минимальном конечном автомате. В действительности, суще- ствует более эффективный алгоритм с временем исполнения O(/?logn); ♦ преобразование недетерминированных автоматов в детерминированные. С детер- минированными конечными автоматами очень легко работать, т. к. в любой данный момент автомат находится в одном-единственном состоянии. Недетерминирован- ные конечные автоматы могут находиться в нескольких состояниях одновременно, поэтом) их текущее "состояние" представляет подмножество всех возможных со- стояний автомата. Однако любой недетерминированный конечный автомат можно механически пре- образовать в эквивалентный детерминированный конечный автомат, который потом удастся минимизировать, как описано выше. Это преобразование может вызвать экспоненциальный рост количества состояний, но не исключено, что оно умень- шится во время минимизации детерминированного конечного автомата. Такой экс- поненциальный "взрыв" делает большинство задач минимизации недетерминиро- ванных конечных автоматов PSPACE-полными. а это даже хуже, чем NP-полнота. Доказательства эквивалентности недетерминированных конечных автоматов, де- терминированных конечных автоматов и регулярных выражений достаточно эле- ментарны и рассматриваются в университетских курсах теории автоматов. Впрочем, при кодировании возникают неожиданные проблемы; ♦ создание автоматов на основе регулярных выражений. Для преобразования регу- лярного выражения в эквивалентный конечный автомат существуют два подхода, разница между которыми состоит в том, создается детерминированный или неде- терминированный автомат. Недетерминированные конечные автоматы легче созда- вать, но их сложнее эмулировать. 22 Зак 3741
658 Часть II Каталог алгоритмических задач Для создания недетерминированных конечных автоматов используются с-переходы. которые являются дополнительными переходами, не требующими входного симво- ла. Попав в некоторое состояние с помощью к-перехода, мы должны предполагать, что автомат может быть в любом состоянии. Использование с-переходов позволяет с легкостью создать автомат на основе обхода в глубину дерева синтаксического разбора регулярного выражения. Данный автомат будет иметь О(гп) состояний, где т — длина регулярного выражения. Кроме этого, эмуляция автомата на строке дли- ной п символов занимает время O(inn), т. к. каждую пару "состояние/префикс" нуж- но рассмотреть только один раз. Создание детерминированного конечного автомата начинается с построения дерева синтаксического разбора регулярного выражения с соблюдением того условия, что каждый лист должен представлять символ алфавита в образце. Распознав префикс текста, мы оказываемся в некотором подмножестве возможных позиций, которое будет соответствовать состоянию конечного автомата. Алгоритм Бржозовского (Brzozowski) создает этот автомат по одному состоянию за раз по мере надобности. Но даже в этом случае для некоторых регулярных выражений длиной в т сим- волов потребуется О(2‘") состояний в любом реализующем их детерминирован- ном конечном автомате. В качестве примера можно привести выражение (а + Ь)*а(а + Ь)(а + Ь)...(а + Ь). Нет никакой возможности избежать этого экспонен- циального "взрыва" сложности по памяти. К счастью, эмуляция входной строки за- нимает линейное время на любом детерминированном конечном автомате, незави- симо от размера автомата. Реализации. Пакет Grail+ на языке C++ предназначен для символьных вычислений над конечными автоматами и регулярными выражениями. Пакет позволяет выполнять преобразования между разными представлениями автомата и минимизировать автома- ты. Пакет может обрабатывать автоматы большого размера, определенные на больших алфавитах Код и документацию можно найти на веб-сайте http://www.csd.uwo.ca/ Research/grail/, где также предоставлены ссылки на многие другие пакеты. Использо- вание пакета Grail в коммерческих целях требует предварительного разрешения, но для образовательных целей пакет предоставляется бесплатно, как для студентов, так и для преподавателей. Библиотека FSM компании AT&T представляет собой программный пакет под UNIX, предназначенный для создания, комбинирования, оптимизирования и исследования взвешенных конечных автоматов, как распознавателей, так и преобразователей. Биб- лиотека поддерживает автоматы с более чем десятью миллионами состояний и перехо- дов. Подробности см. на веб-сайте http://www.research.att.com/~fsmtools/fsm. Пакет JFKAP содержит графические инструменты для изучения основных понятий теории автоматов. В пакет включены функции для выполнения преобразований между детерминированными конечными автоматами, недетерминированными конечными автоматами и регулярными выражениями, а также функции для минимизации автома- тов. Поддерживаются автоматы более высокого уровня, в том числе контекстно- свободные языки и машины Тьюринга. Пакет JFLAP можно загрузить с веб-сайта http://wvvw.jflap.org. Там же доступна книга [RF06] Пакет FIRE Engine предоставляет коммерческие реализации алгоритмов работы с ко- нечными автоматами и регулярными выражениями. Пакет также содержит реализации
Глава 18. Множества и строки 659 нескольких алгоритмов минимизации конечных автоматов, включая алгоритм Хопкрофта с временем исполнения O(«lg«). Поддерживаются как детерминирован- ные, так и недетерминированные автоматы. Пакет можно загрузить с веб-сайта http://www.fastar.org/, а улучшенную версию — с веб-сайта http://www.eti.pg.gda.pl/ ~jandac/min ini.html. Примечания В работе [Aho90] предоставлен хороший обзор алгоритмов поиска по образцу, в том чис- ле и для случая, когда образцы представлены регулярными выражениями. Метод поиска образцов в виде регулярных выражений, использующий £-переходы, был представлен в работе [Tho68]. Описание методов поиска по образцу с помощью конечных автоматов можно найти в книге [AHIJ74], Обсуждение конечных автоматов и теории вычислений представлено в книгах [HMU06] и [SipO5J. Основным форумом специалистов в этой области является конференция С1АА (Conference on Implementation and Application of Automata, конференция по реализации и применению автоматов). На веб-сайте http://tln.li.univ-tours.fr/ciiia/ доступны ссылки на материалы предыдущих конференций и соответствующее программное обеспечение. Оптимальный алгоритм с временем исполнения (9(«lg«) для минимизации количества со- стояний детерминированного конечного автомата был представлен в работе [Нор711. Ме- тод производных для создания конечного автомата из регулярного выражения был пред- ставлен в работе [Brz64] и изложен более подробно в работе [BS86]. Среди описаний ал- горитма Бржозовского можно назвать книгу [Соп71]. Среди последних работ по инкрементальному созданию и оптимизации автоматов можно назвать статью [WatO3], Задача преобразования детерминированного конечного автомата в минимальный неде- терминированный конечный автомат (см. [JR93]), а также задача проверки на эквивалент- ность двух недетерминированных конечных автоматов (см. [SM73]) являются PSPACE- полными. Родственные задачи. Задача выполнимости (см. раздел 14.10), поиск строк (см. раз- дел 18.3). 18.8. Максимальная общая подстрока Вход. Множество S строк ..., Sn. Задача. Найти такую максимальную строку S', все символы которой входят в виде подстроки в каждую строку S,, где 1 < i < п (рис. 18.9). Обсуждение. Задача поиска максимальной общей подстроки или подпоследовательно- сти возникает при сравнении текстов. Особенно важным приложением является выяс- нение степени сходства биологических последовательностей. Гены белков эволюцио- нируют со временем, но их важнейшие участки изменяться не должны. Максимальная общая подпоследовательность вариаций одного гена в разных биологических видах позволяет получить представление о генном материале, не подвергшемся изменениям. Задача поиска максимальной общей подстроки является частным случаем задачи вы- числения расстояния редактирования (см. раздел 18.4), в котором замены символов недопустимы, а разрешаются только вставка и удаление. При этих условиях расстояние редактирования между строками Р и Т равно п + m~2\lcs(P, 7)|, т. к. мы можем удалять
660 Часть II. Каталог алгоритмических задач GCAAGTCTAATA CAAGGTTATATA GCAATTCTATAA CAATTGATATAA GCAAT САТАТАТ С т с А А ВХОД ВЫХОД Рис. 18.9. Поиск максимальной общей подстроки из Р символы, отсутствующие в lcs(P, Т), и вставлять символы строки Т, чтобы преоб- разовать строку Р в строку Т. При решении данной задачи возникают следующие вопросы. ♦ Требуется ли найти общую подстроку? При проверке текстов на уникальность нам нужно найти самую длинную общую фразу в двух и более документах. Так как фра- зы являются строками символов, ищется максимальная подстрока, общая для всех текстов. Максимальную общую подстроку для множества строк можно найти за линейное время, используя суффиксные деревья (см. раздел 12.3). Решение заключается в том, чтобы создать суффиксное дерево, содержащее все строки, пометить каждый лист входной строкой, которую он представляет, а потом выполнить обход в глуби- ну и найти самый дальний узел с потомками из каждой входной строки. ♦ Требуется ли найти общую подпоследовательность разбросанных символов? Далее в этом разделе мы будем рассматривать только задачу поиска общих подпоследова- тельностей разбросанных символов. Алгоритм поиска таких подпоследовательно- стей является частным случаем алгоритма динамического программирования для вычисления расстояния редактирования. Реализация этого алгоритма на языке С приводится в листинге 8.14. Пусть M[i,j]— количество символов самой длинной общей подстроки строк 5[1]. .... 5р] и 1 ].7р]. Когда Sp] 7р], совпадение последней пары символов невоз- можно ни при каких обстоятельствах, поэтому ЛУр,у] = тах(ЛУр,у - 1], Л7(*-l.j]). Но если Sp] = 7р], мы можем выбрать этот символ для нашей подстроки, поэтому А/[/, у] = тах(Л/р - 1, j - 1 ] + 1, M\i - 1 ,у], M[i, j - 1 ]). Данное рекуррентное соотношение вычисляет длину максимальной общей подпос- ледовательности за время О(пт). Саму общую подстроку можно воссоздать, двига- ясь в обратном направлении от М[п, т] и выясняя, какие символы совпали. ♦ Как поступать, если количество наборов совпадающих символов сравнительно не- велико? Для строк, содержащих не слишком много экземпляров одного и того же
Глава 18 Множества и строки 661 символа, существует более быстрый алгоритм. Пусть г— количество пар таких по- зиций (/,/). для которых S, = 7',. Таким образом, г может достичь значения тп, если обе строки состоят полностью из одного и того же символа, но г — п. если обе стро- ки являются перестановками множества {I,п}. Этот метод рассматривает нары /-, как определяющие точки на плоскости. Полный набор из г таких точек можно вычислить за время ()(п + т + г), используя метод раскладывания по корзинам. Для каждого символа алфавита с и каждой стро- ки (S или Г) создается корзина, после чего номера позиций каждого символа строки распределяются по соответствующим корзинам. Потом из каждой пары v е \ и / е Т, в корзинах \ и Д создается точка (5, /). Общая подпоследовательность описывает монотонно неубывающий путь по этим точкам, т. е. путь, идущий только вверх и вправо. Самый длинный такой путь мож- но найти за время О((и + r)lg«). Точки сортируются в порядке возрастания х-координаты; конфликты с одинаковым значением этой координаты разрешаются в пользу точки с большим значением у-координаты. В этом порядке мы вставляем точки по одной и отслеживаем минимальную конечную у-координату любого пути, проходящего ровно через к точек, для каждого к, где I <к<п. Новая точка изменяет только один из этих путей, либо определяя новую максимальную подпос- ледовательность, либо уменьшая значение у-координаты кратчайшего пути, конеч- ная точка которого находится выше точки pv. ♦ Как поступать, если строки являются перестановками? Перестановки — эго строки, в которых символы не повторяются. Две перестановки задают п пар совпа- дающих символов, и в этом случае приведенный ранее алгоритм исполняется за время (~Xn\gn). Важным частным случаем является поиск максимальной возрас- тающей подпоследовательности числовой последовательности. Отсортировав последовательность, а потом заменив каждое число его номером, мы получим пере- становку р. Максимальная общая подпоследовательность перестановки р и после- довательности (I, 2, 3, ..., п\ дает нам максимальную возрастающую подпоследова- тельность. ♦ Как поступать, если задано несколько строк? Базовый алгоритм динамического программирования можно обобщить для работы с к строками. Время исполнения в этом случае будет равно О(2кпк), где п— длина максимальной строки. Алгоритм имеет сложность, экспоненциально зависящую от количества строк к, и поэтому он годится только для небольшого количества строк. Кроме того, данная задача явля- ется NP-полной, поэтому не следует надеяться на появление более быстрого точно- го алгоритма в ближайшее время. Для решения задачи выравнивания нескольких последовательностей было предло- жено большое количество эвристических методов. Часто эти алгоритмы начинают работу с вычисления попарного выравнивания для каждой из (*) пар строк. В од- ном из подходов две наиболее похожие последовательности объединяются в одну и заменяются ею, и этот процесс повторяется до тех пор. пока все последовательности не будут слиты в одну. Проблема в том, что для двух строк часто существует много разных выравниваний оптимальной стоимости. Выбор самого подходящего из них
662 Часть II. Каталог алгоритмических задач зависит от оставшихся последовательностей, подлежащих слиянию, а они алгорит- му неизвестны. Реализации. Для выравнивания нескольких последовательностей ДНК существует много программ. В частности, популярным инструментом выравнивания протеиновых последовательностей является программа ClustalW (см [THG94]). Программу можно загрузить с веб-сайта http://www.ebi.ac.uk/Tools/clustahv/. Еще одним заслуживающим внимания программным средством выравнивания нескольких биологических последо- вательностей является пакет MSA, который можно загрузить с веб-страницы http://www.ncbi.nlni.nih.gov/CBBresearch/Schaffer/msa.html. Любую из программ динамического программирования для нечеткого сравнения строк, рассмотренных в разделе 18.4. можно использовать для поиска максимальной общей подстроки Специализированные реализации на языках Perl. Java и С доступны на веб- странице http://bix.ucsd.edu/bioalgorithms/downloads/code/. Библиотека Combinatorica (см. [PS03]) содержит реализацию (на языке пакета Mathematica) алгоритма создания максимальной возрастающей подпоследовательности перестановки, что является частным случаем максимальной общей подпоследователь- ности. Этот алгоритм основан не на динамическом программировании, а на использо- вании таблиц Янга. Подробности см. в разделе 19.1 9. Примечания Обзоры алгоритмов поиска максимальных общих подпоследовательностей можно найти в [BHROO] и [GBY91]. Алгоритм для последовательностей, имеющих мало или вовсе не имеющих совпадений, был представлен в работе [HS77]. Его описание приводится в кни- гах [Aho90] и [Мап89]. В последнее время наблюдается неожиданное повышение интере- са к этой задаче, в частности к разработке эффективных алгоритмов поиска максимальной общей подпоследовательности, применяющих параллелизм на уровне битов (см. [CIPROI]). В работе [МР80] представлен быстрый алгоритм поиска максимальной общей последова- тельности для алфавитов фиксированного размера. Алгоритм основан на методе четырех русских и имеет время исполнения, равное O(n/»/log(min{/n, л})). Создадим две случайные строки длиной п на алфавите из а символов. Какова ожидаемая длина максимальной общей подстроки? Эта задача является предметом активных иссле- дований, прекрасный обзор которых представлен в работе [Dan94]. Выравнивание нескольких биологических последовательностей представляет собой от- дельную область, хорошим введением в которую послужат книги [Gus97] и [DEK.M98]. Свежий обзор состояния дел вы найдете в работе [Not02], Сложность задачи выравнива- ния нескольких последовательностей следует из сложности задачи поиска минимальной общей подстроки для больших наборов строк (см. [Mai78]). Среди приложений задачи поиска максимальной общей подстроки мы упомянули воз- можность ее применения для выявления плагиата. Интересные рассуждения о том, как реализовать детектор плагиата для компьютерных программ, представлены в работе [SWA03], Родственные задачи. Нечеткое сравнение строк (см. раздел 18.4), поиск минимальной общей надстроки (см. раздел 18.9).
Гпава 18. Множества и строки 663 18.9. Поиск минимальной общей надстроки Вход. Множество строк S= {Sb Sm}. Задача. Найти минимальную строку S, содержащую в виде подстроки каждую строку S, (рис. 18.10). ABRACADABRA ABRAC А С A D А A D А В R D А В R А R А С A D ABRAC R А С A D А С A D А A D А В R D А В R А ВХОД выход Рис. 18.10. Поиск минимальной обшей надстроки Обсуждение. Задача поиска минимальной общей надстроки возникает в разных при- ложениях. Однажды заядлый игрок казино спросил у меня, как восстановить последо- вательность символов на диске игрового автомата. При очередном раунде игры каж- дый диск игрового автомата после вращения останавливается в случайной позиции, показывая выпавший символ, а также непосредственно предшествующий и следующий ему символы. При наблюдении за достаточно большим количеством раундов игры можно выяснить порядок символов каждого диска игрового автомата, решив задачу поиска минимальной общей (циклической) надстроки для трехсимвольных подстрок, полученных в результате наблюдения. Другое применение задачи поиска минимальной общей надстроки — сжатие матрицы. Предположим, имеется разреженная матрица М размером п * т, т. е. значение боль- шинства ее элементов равно нулю. Каждую строку матрицы можно разбить на т/k под- строки из к элементов, а потом создать минимальную общую надстроку S' из этих под- строк. Теперь матрицу можно представить этой надстрокой совместно с массивом ука- зателей размером п * т!к. обозначающих начало каждой из этих последовательностей в надстроке. Доступ к любому элементу по-прежнему занимает постоянное время, но если |S| « пт, то экономия памяти будет значительной. Возможно, самым популярным приложением задачи поиска минимальной общей над- строки является сборка ДНК. Роботы без труда собирают последовательности из 500 базовых пар ДНК, но наибольший интерес вызывает секвенирование длинных мо- лекул. В процессе сборки последовательностей методом "дробовика" создается множе- ство копий целевой молекулы, которые разбиваются на случайные фрагменты. Затем выполняется секвенирование этих фрагментов, и их минимальная надстрока считается корректной последовательностью. Найти надстроку набора строк нетрудно, т. к. мы можем просто конкатенировать ис- ходные строки. Проблема в том. чтобы найти кратчайшую надстроку. Задача поиска минимальной обшей надстроки является NP-полной для строк всех классов.
664 Часть II Каталог алгоритмических задач Задачу поиска минимальной общей надстроки можно с легкостью свести к задаче ком- мивояжера (см. раздел 16.4). Для этого создаем граф перекрытий G. в котором вершина v, представляет строку S,. Присваиваем ребру (v„ v,) вес. равный длине строки S,, уменьшенной на длину перекрытия строк S, и S,. Например, h(v„ у,) = 1 для S, = ahc и S, = bcd. Путь с минимальным весом, проходящий через все вершины, определяет ми- нимальную общую надстроку. Веса ребер несимметричны,— обратите внимание, что M'(vz. v,) = 3 на рис. 18.10. К сожалению, несимметричные задачи коммивояжера намно- го сложнее симметричных. Стандартным решением является аппроксимирование минимальной общей надстроки с помощью "жадного" алгоритма. Находим пар)' строк с максимальным перекрытием. Заменяем эти строки их объединением и повторяем процесс до тех пор. пока не оста- нется только одна строка. Для этого эвристического алгоритма можно создать реализа- цию с линейным временем исполнения. По-видимому, самым дорогим этапом является создание графа перекрытий. Поиск максимального перекрытия двух строк длиной / методом исчерпывающего перебора занимает время О(1г) для каждой из ()(п~) пар строк. Но для решения этой задачи существуют более быстрые методы, основанные на использовании суффиксных деревьев (см. раздел 12.3). Создаем дерево, содержащее все суффиксы всех строк множества S. Строка S, перекрывает строку S, тогда и только тогда, когда суффикс строки S, совпадает с префиксом строки S,, а это событие отобра- жается вершиной суффиксного дерева. Обход таких вершин в порядке увеличения рас- стояния от корня определяет соответствующий порядок слияния строк. Какова эффективность этого "жадного" эвристического алгоритма? Очевидно, сущест- вуют случаи, заставляющие его создать надстроку, вдвое превышающую оптимальную. Например, оптимальный порядок слияния строк c(ab)k, (ba)k и (ab)kc— слева направо. Но наш "жадный" алгоритм начинает со слияния первой и третьей строки, не оставляя для средней строки возможности перекрытия. Надстрока, возвращаемая "жадным" ал- горитмом. никогда не будет превышать оптимальную более, чем в 3,5 раза, а на прак- тике результат обычно даже лучше. Задача создания надстрок значительно усложняется, когда некоторые исходные строки, согласно условию, не могут быть подстроками конечной надстроки. Задача выявления таких подстрок является NP-полной. если только не разрешено добавить в алфавит до- полнительный символ, чтобы использовать его как разделитель. Реализации. Для сборки последовательностей ДНК существует несколько высокопро- изводительных программ. Такие программы исправляют ошибки секвенирования, так что конечный результат не обязательно будет надстрокой входных строк. Как мини- мум. эти программы могут послужить в качестве моделей, когда вам требуется найти недлинную корректную надстроку. Самыми последними разработками в этой серии являются программы САРЗ (см. [НМ99]) и РСАР (см. [HWA 03]), которые можно загрузить с веб-сайта http://seq.cs.iastate.edu/. Эти программы применялись в проектах сборки генов млеко- питающих, когда одновременно были задействованы сотни миллионов участков ДНК. Программа Celera. которая использовалась для первоначального секвенирования гено- ма человека, теперь доступна для общего пользования. Ее можно загрузить с веб- страницы http://sourceforge.net/projectsAvgs-assembler/.
Глава 18. Множества и строки 665 Примечания Описание задачи поиска минимальной общей подстроки и ее применения в сборке после- довательностей ДНК методом "дробовика" представлено в книге [МК.Т07] и в работе [Муе99а]. В статье [KM95J приводится алгоритм решения более общего случая задачи поиска минимальной общей надстроки, когда предполагается, что исходные строки со- держат ошибки. Эта статья рекомендуется для чтения всем, интересующимся сборкой фрагментов ДНК. В работе [BJL 94] были представлены первые аппроксимирующие алгоритмы поиска ми- нимальной общей надстроки, в которых применяется разновидность "жадного" эвристи- ческого метода и которые выдают решения, имеющие постоянный коэффициент. В ре- зультате более поздних исследований (см. [Swe99]) удалось уменьшить этот коэффициент до 2,5, что является шагом вперед на пути к ожидаемому коэффициенту 2. В настоящее время самое лучшее приблизительное решение, выдаваемое стандартным "жадным" эври- стическим алгоритмом, имеет коэффициент 3,5 (см. [KS05a]). Эффективные реализации такого алгоритма описаны в работе [Gus94]. Отчет об экспериментах с эвристическими методами поиска минимальных общих над- сгрок представлен в статье [RBT04], На основе этих экспериментов можно предположить, что для входных экземпляров разумного размера "жадные" эвристические методы обычно выдают решения, имеющие коэффициент лучший, чем 1,4%. Результаты экспериментов с использованием подходов, основанных на генетических алгоритмах, представлены в статье [ZS04]. В работе [YZ99] содержатся аналитические результаты, демонстрирую- щие очень небольшое сокращение минимальной общей надстроки случайных последова- тельностей, обусловленное тем, что ожидаемая длина перекрытия двух произвольных строк невелика. Родственные задачи. Суффиксные деревья (см. раздел 12.3), сжатие текста (см. раз- дел 18.5).
ГЛАВА 19 Ресурсы Эта глава содержит краткое описание ресурсов, с которыми должен быть знаком каж- дый разработчик алгоритмов. Хотя часть этой информации была представлена в других местах каталога задач, здесь собраны наиболее важные ссылки. 19.1. Программные системы В этом разделе дается описание нескольких наиболее полных пакетов с реализациями комбинаторных алгоритмов, любой из которых можно загрузить из Интернета. Хотя эти программы упоминаются в соответствующих разделах каталога задач, они заслу- живают особого внимания. Хороший разработчик алгоритмов не изобретает колесо, а хороший программист не создает код, который уже был создан другими. Лучше всего на эту тему выразился Пи- кассо: "Хорошие художники заимствуют. Великие художники крадут". Здесь необходимо сказать несколько слов по поводу воровства. Многие из программ- ных продуктов, описываемых в этой книге, доступны только для исследовательских или образовательных целей. Для использования их в коммерческих целях необходимо заключить лиценционное соглашение с авторами. Я настоятельно рекомендую вам со- блюдать это правило. Лицензионные условия большинства академических организаций обычно вполне скромны. Факт использования кода в коммерческих разработках часто бывает важнее для автора, чем финансовая сторона. Автор получает стимул для сопро- вождения кода и выпуска следующих версий. Поступайте честно и приобретайте необ- ходимые лицензии. Условия лицензии и контактные данные обычно указаны в доку- ментации на программное обеспечение или доступны на веб-сайте разработчика. Хотя многие из описанных здесь программ можно получить из нашего хранилища ал- горитмов (http://vvvvw.cs.sunysb.edu/~algorith), мы настоятельно рекомендуем загру- жать эти программы с сайтов их разработчиков. Во-первых, существует очень большая вероятность, что версия на сайте разработчика является самой свежей. Во-вторых, на сайте разработчика часто можно найти вспомогательные файлы и документацию, ко- торые не были скопированы в наше хранилище, но могут оказаться полезными. Нако- нец, многие разработчики следят за количеством загрузок своих программ, поэтому вы лишите их морального поощрения, если будете загружать их программы с других сай- тов. 19.1.1. Библиотека LEDA Библиотека LEDA (Library of Efficient Data Types and Algorithms, библиотека эффек- тивных типов данных и алгоритмов)— это, пожалуй, самый лучший из доступных ре-
Глава 19. Ресурсы 667 сурсов с реализациями комбинаторных методов. Библиотека была создана группой разработчиков из института Макса Планка в г. Саарбрюкен, Германия, в составе Курта Мельхорна (Kurt Mehlhorn), Штефана Наера (Stefan Naher), Штефана Ширры (Stefan Schirra), Кристиана Урига (Christian Uhrig) и Кристофа Бурникеля (Christoph Burnikel). Библиотека LEDA уникальна благодаря высочайшей квалификации ее создателей и большому количеству средств, вложенных в этот проект. Эта библиотека предлагает обширную коллекцию хорошо реализованных на языке C++ структур данных и типов. Особенно полезным является тип graph, который под- держивает все основные операции, хотя платой за такую универсальность являются несколько больший объем кода и меньшая скорость работы, чем у специализирован- ных реализаций. В библиотеку входит набор программ для работы с графами, наглядно демонстрирующих, как можно аккуратно и кратко реализовать алгоритмы, используя типы из библиотеки LEDA. В библиотеке имеются хорошие реализации самых важных структур данных, в частности словарей и очередей с приоритетами. Библиотека также содержит алгоритмы и структуры данных для приложений вычислительной геометрии, включая поддержку визуализации. Дополнительную информацию о библиотеке см. в книге [MN99], С 2001 г. библиотеку можно получить исключительно через компанию Algorithmic Solutions Software GmbH (http://www.algorithmic-solutions.com/). Таким образом обес- печивается профессиональная поддержка библиотеки и гарантируется регулярный вы- пуск новых версий. В феврале 2008 г. была выпущена бесплатная версия библиотеки, содержащая все основные структуры данных (включая словари, очереди с приоритета- ми, графы и числовые типы). В бесплатной версии отсутствуют исходный код и неко- торые особо сложные алгоритмы. Но плата за лицензию на использование полной вер- сии библиотеки невелика, к тому же у пользователя имеется возможность бесплатно загрузить пробную версию полной библиотеки. 19.1.2. Библиотека CGAL Библиотека CGAL (Computational Geometry Algorithms Library, библиотека алгоритмов вычислительной геометрии) включает в себя эффективные и надежные реализации ал- горитмов вычислительной геометрии на языке C++. Библиотека весьма обширна и со- держит широкий спектр реализаций алгоритмов создания триангуляционных разбие- ний. диаграмм Вороного, конфигураций прямых, альфа-очертаний, выпуклых оболо- чек, а также операций с многоугольниками и многогранниками. Предлагаемые реализации предназначены для работы в двух- и трехмерном пространстве, а некото- рые— и в пространствах с большим количеством измерений. Сайт библиотеки CGAL (www.cgal.org) — это первое место, где вы должны искать профессиональные программы для геометрических вычислений, хотя поначалу вам придется потратить какое-то время на то, чтобы разобраться в структуре этой библио- теки. Библиотека CGAL распространяется по схеме двойного лицензирования. Ее можно бесплатно использовать вместе с программным обеспечением с открытым ис- ходным кодом, но для использования в других ситуациях необходимо приобрести коммерческую лицензию.
668 Часть II. Каталог алгоритмических задач 19.1.3. Библиотека Boost Библиотека Boost (www.boost.org) содержит популярную коллекцию прошедших экс- пертную оценку бесплатных переносимых исходных кодов библиотек на языке C++. Условия лицензии библиотеки предусматривают как коммерческое, так и некоммерче- ское использование. Библиотека Boost Graph Library, пожалуй, лучше всего подходит для читателей этой книги. Руководство пользователя можно найти в [SLL02] и на странице http://www.boost.org/libs/graph/doc. Библиотека включает в себя реализации списков смежности, матриц смежности и списков ребер, а также неплохую коллекцию базовых алгоритмов для работы с графами. Интерфейс и компоненты библиотеки являются обобщенными в том смысле, в каком этот термин употребляется в стандартной биб- лиотеке шаблонов STL языка C++. Другие интересные коллекции из библиотеки Boost содержат программы обработки строк и пакеты для математических вычислений. 19.1.4. Библиотека GOBLIN Библиотека GOBLIN (Graph Object Library for Network Programming Problems, библио- тека объектов-графов для задач программирования сетей) содержит классы, написан- ные на языке C++, и посвящена, в основном, задачам оптимизации графов. Она вклю- чает в себя реализации нескольких разновидностей алгоритмов поиска кратчайшего пути, минимальных остовных деревьев и компонент связности, а также большой набор средств для решения задач о потоках в сети и паросочетаниях графов. Для решения таких сложных задач, как поиск независимого множества и вершинная раскраска, пре- доставляется модуль обобщенный, использующий метод ветвей и границ. Библиотека GOBLIN была разработана и поддерживается Кристианом Фремут- Пагером (Christian Fremuth-Paeger) из университета г. Аусберга. Библиотека доступна на условиях общественной лицензии GNU ограниченного применения и находится на веб-странице http://www.math.uni-augsburg.de/~fremuth/goblin.html. Библиотека имеет интерфейс на языке программирования сценариев Tcl/Tk. Предположительно, библиотека GOBLIN не является такой устойчивой, как библиотека Boost или LEDA, но она содержит ряд алгоритмов, отсутствующих в этих двух библиотеках. 19.1.5. Библиотека Netlib Библиотека Netlib (www.netlib.org)— это хранилище математического программного обеспечения, содержащее большой объем кода, а также множество таблиц и статей. Здесь собраны ресурсы из самых разных источников, дополненные подробным индек- сом и механизмом поиска. Важными достоинствами библиотеки Netlib являются ши- рокий спектр ее ресурсов и легкость доступа к ним. Если вам потребуется специализи- рованная математическая программа, начните ее поиски с библиотеки Netlib. Служба GAMS (Guide to Available Mathematical Software, путеводитель по существую- щему математическому программному обеспечению) индексирует библиотеку Netlib и другие хранилища математического программного обеспечения. Она поддерживается Национальным институтом стандартов и технологий США (NIST) и доступна по адре- су http://gams.nist.gov.
Глава 19. Ресурсы 669 19.1.6. Коллекция алгоритмов ассоциации АСМ Одним из первых механизмов распространения реализаций полезных алгоритмов была коллекция CALGO (Collected Algorithms of the ACM. сборник алгоритмов ассоциации АСМ). Впервые коллекция была представлена в журнале "Communications of the ACM" в I960 г., и тогда в нее вошли такие знаменитые алгоритмы, как алгоритм Флойда для создания пирамиды с линейным временем исполнения. В настоящее время коллекция сопровождается журналом "ACM Transactions on Mathematical Software". Для каждого алгоритма и его реализации дается краткое описание в статье журнала, а реализация проверяется и добавляется в коллекцию Программы доступны по адресу http://www.acm.org/ealgo и на веб-сайте библиотеки Netlib. На сегодняшний день в коллекции представлено свыше 850 алгоритмов. Большинство программ написано на языке FORTRAN и относится к математическим вычислениям, хотя несколько интересных комбинаторных алгоритмов нашли в ней свое место. По- скольку реализации прошли экспертную оценку, они считаются более надежными, чем аналогичные программы из других источников. 19.1.7. Сайты SourceForge и CPAN Сайт SourceForge (http://sourceforge.net) является самым крупным ресурсом про- граммного обеспечения с открытым исходным кодом, насчитывающим свыше 160 ты- сяч зарегистрированных проектов. Большинство из этих проектов представляет инте- рес лишь для узкого круга потребителей, но вы тоже сможете найти много полезного материала, в частности, библиотеки JUNG и JGraphT для работы с графами и средства оптимизации Ipsolve и JGAP. Архив CPAN (Comprehensive Perl Archive Network, всеобъемлющая сеть архивов Perl) (http://www.cpan.org) содержит громадную коллекцию модулей и сценариев, написан- ных на языке Perl. Прежде чем пытаться самостоятельно реализовать что-либо на этом языке, поищите существующую реализацию в этом архиве. 19.1.8. Система Stanford GraphBase Система Stanford GraphBase интересна во многих отношениях. Прежде всего, она была создана в соответствии с принципом "грамотного программирования", означающим, что ее можно читать. Если чьи-либо программы заслуживают, чтобы их читали, так это программы Дональда Кнута. Его книга [Knu94] содержит полный исходный код систе- мы Stanford GraphBase. В качестве среды программирования используется система CWEB, позволяющая эффективно сочетать описание программы с ее кодом. Система GraphBase содержит реализации многих важных комбинаторных алгоритмов, включая алгоритмы поиска паросочетаний, вычисления минимальных остовных де- ревьев и построения диаграмм Вороного, а также специализированные средства для создания графов-расширителей и создания комбинаторных объектов. Наконец, система содержит программы для решения многих развлекательных задач, включая создание "лесенок слов" (цепочек, в которых каждое слово отличается от предыдущего на одну букву) и определения доминирующих отношений среди футбольных команд. Домаш-
670 Часть II. Каталог алгоритмических задач няя страница системы находится в Интернете по адресу http://www-cs-faculty. stanford.edu/~knuth/sgb.litml. С системой GraphBase интересно экспериментировать, но она мало пригодна для соз- дания приложений общего характера. Ее можно применять, самое большее, как генера- тор экземпляров графов, которые могут послужить в качестве тестовых данных для других программ. Система содержит графы, полученные путем обработки взаимоот- ношений персонажей известных романов, статей из словаря синонимов Роджета, визу- альных характеристик "Моны Лизы", графов-расширителей, а также модели экономики США. Помимо прочего, в системе GraphBase используются аппаратно-независимые генераторы случайных чисел, и поэтому построенные с ее помощью случайные графы могут быть воссозданы на других компьютерах, что делает их прекрасным материалом для экспериментального сравнения алгоритмов. 19.1.9. Пакет Combinatorica Пакет Combinatorica (см. [PS03]) содержит реализации свыше 450 алгоритмов по ком- бинаторике и теории графов. Эти процедуры, написанные на языке пакета Mathematica. предназначены для работы в комплексе, что облегчает эксперименты с ними. Пакет Combinatorica широко используется как для исследовательских, так и для образова- тельных целей. Хотя (с моей точки зрения) пакет Combinatorica более полон и лучше интегрирован, чем другие библиотеки комбинаторных алгоритмов, он работает очень медленно. От- ветственность за его достоинства и недостатки лежит на среде Mathematica, которая предоставляет полнофункциональный интерпретируемый язык программирования высокого уровня, обладающий вследствие вышесказанного очень низкой эффектив- ностью. Пакет Combinatorica лучше всего подходит для решения небольших задач, а также может послужить источником кратких описаний алгоритмов, пригодных для перевода на другие языки программирования (если вы сумеете разобраться в коде Mathematica). Пакет Combinatorica и связанные с ним ресурсы доступны на веб-сайте http:// vvvvw.combinatorica.com. Пакет также входит в состав стандартного дистрибутива Mathematica и находится в папке Packages/DiscreteMath/Combinatorica.m. 19.1.10. Программы из книг Многие книги, посвященные алгоритмам, содержат работоспособные реализации алгоритмов, написанные на распространенных языках программирования. Хотя эти реализации предназначены, главным образом, для иллюстрации материала, их вполне можно использовать на практике. Поскольку они, как правило, аккуратно написаны и имеют небольшой размер, они могут послужить хорошей основой для простых прило- жений. Далее приводятся описания наиболее интересных программ этого типа. Большинство из них доступны в хранилище алгоритмов по адресу http://vvwvv.cs.sunysb.edu/ ~algorith.
Глава 19 Ресурсы 671 Книга "Programming Challenges" Если вам нравятся программы на языке С. приведенные в первой части этой книги, вас, вероятно, заинтересуют программы, которые я написал для своей книги [SR03]. Возможно, наибольшую пользу принесут дополнительные примеры алгоритмов дина- мического программирования, процедуры вычислительной геометрии, такие как создание выпуклой оболочки, а также пакет bignum для выполнения арифметических операций с большими целыми числами. Эта библиотека алгоритмов доступна на сайтах: http://www.cs.sunysb.edu/~skiena/392/progranis/ и http://www.programming- challenges.com. Книга "Combinatorial Algorithms for Computers and Calculators" Эта книга (см. [N W78]) посвящена алгоритмам создания элементарных комбинаторных объектов, таких как перестановки, подмножества и разбиения. У этих алгоритмов, как правило, очень короткие описания, но их трудно найти, и в них не всегда легко разо- браться. Для всех алгоритмов имеются реализации на языке FORTRAN, сопровождае- мые обсуждением теоретических основ. Эти программы обычно невелики, что позво- ляет легко переписать их на каком-либо современном языке программирования. Имен- но так я и поступил при разработке пакета Combinatorica (см. раздел 19.1.9). В книге приводятся алгоритмы как для случайного, так и для последовательного генерирования объектов. Описания недавно появившихся алгоритмов решения некоторых задач (без предоставления кода программ) можно найти в книге [WИ89]. Реализации этих алгоритмов можно загрузить из нашего хранилища алгоритмов. Нам удалось обнаружить их у Нила Слоуна (Neil Sloane), который хранил эти реализации на магнитной ленте, в то время как у самих авторов их не было! В книге [NW78] предло- жена процедура проверки статистического распределения случайных объектов, созда- ваемых генераторами, которая позволяет убедиться в равномерности этого распределе- ния. Мы настоятельно рекомендуем выполнять такое тестирование перед использова- нием этих программ для контроля того, что данные не были потеряны в процессе передачи. Книга "Computational Geometry in С" Эта книга (см. [O'ROl]) является, пожалуй, самым лучшим введением в вычислитель- ную геометрию, благодаря присутствию в ней аккуратно написанных на языке С реа- лизаций основных алгоритмов. Она содержит реализации всех алгоритмов для реше- ния элементарных задачи вычислительной геометрии, построения выпуклых оболочек, триангуляций и диаграмм Вороного, а также планирования перемещений. Хотя эти реализации, в основном, предназначены для иллюстрации материала, а не коммерче- ского использования, они, скорее всего, достаточно надежны. Программы можно за- грузить с веб-сайта http://maven.smith.edu/~orourke/code.html. Книга "Algorithms in C++" Эта книга (см. [Sed98] и [SS02]) издана в нескольких вариантах, ориентированных на разные языки программирования, включая языки С. C++ и Java. От других книг ее от-
672 Часть II Каталог алгоритмических задач личает широкий диапазон рассматриваемых тем, в том числе алгоритмов для работы с числовыми, строковыми и геометрическими объектами. Участки текста, специфичные для конкретного языка программирования, содержат много небольших фрагментов кода, а не законченные программы или процедуры, и относиться к ним следует как к моделям, а не рабочим реализациям. Фрагменты кода на языке C++ можно загрузить с веб-сайта http://wvvvv.cs.princeton.edu/~rs/. Книга "Discrete Optimization Algorithms in Pascal" Эта книга (см. http://wvvw.cs.princeton.edu/~rs/) содержит коллекцию из 28 программ для решения задач дискретной оптимизации. Сюда входят программы для решения задач целочисленного и линейного программирования, задач о рюкзаке, о покрытии множества, а также задач коммивояжера, вершинной раскраски, календарного плани- рования и оптимизации сетей. Программы доступны по адресу http://vvvvvv.cs.sunysb.edu/ ~algorith. Этот пакет программ интересен тем, что содержащиеся в нем задачи и алгоритмы имеют непосредственное отношение к такой математической дисциплине, как иссле- дование операций. Алгоритмы подбирались с учетом их ценности для решения прак- тических задач. 19.2. Источники данных При тестировании алгоритмов часто возникает необходимость использовать в качестве входа нетривиальные данные, позволяющие проверить правильность работы алгоритма или сравнить скорость работы разных алгоритмов. Поиск хороших тестовых данных может оказаться трудной задачей. Вот некоторые источники: ♦ библиотеки TSPLIB Эта широко известная библиотека содержит стандартную тес- товую коллекцию сложных экземпляров задачи коммивояжера (см. |Rei9l]). Имеющиеся в ней экземпляры задачи представляют собой большие графы, полу- ченные из реальных приложений, таких как проектирование печатных плат и сетей. Библиотека доступна по адресу http://vvww.iwr.uni-heidelberg.de/groups/comopt/ softvvare/TSPLIB95/. Относительно старые экземпляры задачи находятся также в библиотеке Netlib: ♦ Stanford GraphBase Этот пакет программ, написанных Дональдом Кнутом (см. раз- дел 19.1.8), содержит машинно-независимые генераторы разнообразных графов, включая графы, построенные на основе матриц расстояний и произведений искус- ства и литературы, а также графы, представляющие чисто теоретический интерес; ♦ материалы соревнований DIMACS. Некоторые соревнования D1MACS были посвя- щены реализациям алгоритмов работы с графами и различными структурами дан- ных, а также алгоритмов решения логических задач. Для этих соревнований были разработаны генераторы входных экземпляров задач каждого типа, причем особое внимание уделялось созданию трудных или репрезентативных тестовых данных. Материалы доступны по адресу http://dimacs.rutgers.edu/Challenges.
Глава 19. Ресурсы 673 19.3. Библиографические ресурсы Интернет предоставляет фантастические возможности людям, интересующимся алго- ритмами, как, впрочем, и тем, кого интересуют другие темы. Перечислю ресурсы, ко- торыми я пользуюсь чаще всего: ♦ цифровая библиотека АСМ. Эта коллекция библиографических ссылок содержит ссылки на практически все когда-либо опубликованные технические доклады по теории вычислительных систем. Коллекция доступна по адресу http:// portal.acm.org; ♦ Google Scholar. Этот бесплатный ресурс (http://scholar.google.com) ограничивается поиском среди научных работ, что позволяет получить более качественные резуль- таты, чем при универсальном поиске. Особенно полезной является возможность выяснить, в каких работах упоминается данная. Таким образом вы можете обновить свой старый справочный материал и увидеть, что нового произошло в той или иной области, а также определить научную ценность конкретной статьи; ♦ Amazon сот. Этот обширный каталог книг (vvvvvv.amazon.com) чрезвычайно полезен при поиске литературы, посвященной разработке алгоритмов, особенно с тех пор. как многие книги были оцифрованы и внесены в его индекс. 19.4. Профессиональные консалтинговые услуги Консалтинговая фирма Algorist Technologies (http://vvvvvv.algorist.com) предоставляет своим клиентам краткосрочные экспертные услуги по разработке и реализации алго- ритмов. Обычно консультант компании выезжает на объект клиента для интенсивного обсуждения задачи с местным коллективом разработчиков в течение срока от одного до трех дней. Компанией Algorist Technologies накоплен впечатляющий список удач- ных решений по повышению производительности во многих компаниях в различных сферах деятельности. Мы также предоставляем долгосрочные услуги на контрактной основе. Для получения дополнительной информации относительно услуг, предоставляемых компанией Algorist Technologies, звоните по телефону 212-222-9891 и пишите по адре- су электронной почты info(®algorist.com. Algorist Technologies 215 West 92nd St. Suite IF New York, NY 10025 http://www.algorist.com
Список литературы [AAAG95] О. Aichholzer. F Aurenhammer, D. Alberts, and В. Gartner. A novel type of skeleton for polygons. J. Universal Computer Science, 1:752-761, 1995. [ABCC07] D. Applegate, R. Bixby, V. Chvatal, and W. Cook. The Traveling Salesman Problem: A computational study. Princeton University Press, 2007. [Abd80] N. N. Abdelmalek. A Fortran subroutine for the L\ solution of overdetennined systems of linear equations. ACM Trans. Math. Softw., 6(2):228-2.30, June 1980. [ABF05] L Arge, G. Brodal, and R. Fagerberg. Cache-oblivious data structures. In D. Mehta and S. Sahni, editors. Handbook of Data Structures and Applications, p. 34:1—34:27. Chapman and Hal! ' CRC, 2005. [AC75] A. Aho and M. Corasick. Efficient string matching: an aid to bibliographic search. Communications of the ACM, 18:333-340, 1975. [AC9I] D. Applegate and W. Cook. A computational study of the job-shop scheduling problem. ORSA Journal on Computing, 3:149-156, 1991. [ACG+03] G. Ausiello, P. Crescenzi, G. Gambosi, V. Kann, S. Marchetti-Spaccamela, and M. Protasi. Complexity and Approximation: Combinatorial Optimization Problems and Their Approximahihty Properties. Springer, 2003 [АСН+91] E. M. Arkin, L. P. Chew, D. P. Huttenlocher, K. Kedem, and J. S. B. Mitchell. An efficiently computable metric for comparing polygonal shapes. IEEE Trans. PAMI, 13<3):209—216. 1991. [AC192] D. Alberts, G. Cattaneo. and G. Italiano. An empirical study of dynamic graph algorithms. In Proc. Seventh АСМ-SI AM Svmp. Discrete Algorithms (SODA), p. 192-201, 1992. [ACKOla] N. Amenta, S. Choi, and R. Kolluri. The power crust. In Proc. 6th ACMSymp. on Solid Modeling, p. 249-260, 2001. [ACKOlb] N. Amenta, S. Choi, and R. Kolluri. The power crust, unions of balls, and the medial axis transform. Computational Geometry: Theory and Applications, 19:127-153, 2001. [ACP+07] H. Ahn, O. Cheong, C. Park, C. Shin, and A. Vigneron. Maximizing the overlap of two planar convex sets under rigid motions. Computational Geometry: Theory and Applications, 37:3-15, 2007. [ADGM04] L Aleksandrov, H. Djidjev, H. Guo, and A. Maheshwari. Partitioning planar graphs with costs and weights. In Algorithm Engineering and Experiments 4th International Workshop. ALE.HEX 2002, 2004. ]ADKF70] V. Arlazarov, E. Dinic, M. Kronrod, and 1. Faradzev. On economical construction of the transitive closure of a directed graph. Soviet Mathematics. Doklady, 11:1209-1210, 1970.
676 Список литературы [Adl94| L. M Adleman. Molecular computations of solutions to combinatorial problems. Science, 266:1021-1024, November 1 1, 1994. [АЕ83] D. Avis and Fl. EIGindy. A combinatorial approach to polygon similarity. IEEE Trans. Inform. Theory, IT-2:148-150, 1983. [АЕ04] [AF96] G. Andrews and K. Eriksson. Integer Partitions. Cambridge Univ. Press. 2004. D. Avis and K. Fukuda. Reverse search for enumeration. Disc. Applied Math.. 65:21^16, 1996. [AFI102] P. Agarwal, E. Flato, and D. Halperin. Polygon decomposition for efficient construction of Minkowski sums. Computational Geometry: Theory and Applications, 21:39-61.2002. [AGOO] H. Alt and L. Guibas. Discrete geometric shapes: Matching, interpolation, and approximation. In J. Sack and J. Urrutia, editors. Handbook of Computational Geometry, p. 121-153. Elsevier, 2000. [Aga04] P. Agarwal. Range searching. In J. Goodman and J. O’Rourke, editors, Handbook of Discrete and Computational Geometry, p. 809-837. CRC Press. 2004. [AGSS89J A. Aggarwal, L. Guibas, J. Saxe, and P. Shor. A linear-time algorithm for computing the Voronoi diagram of a convex polygon. Discrete and Computational Geometry, 4:591-604, 1989. [AGU72] A. Aho, M. Garey, and J. Ullman. The transitive reduction of a directed graph. SIAM J. Computing, 1:131-137, 1972. [Aho90] A. Aho. Algorithms for finding patterns in strings. In .1. van Leeuwen, editor. Handbook of Theoretical Computer Science: Algorithms and Complexity, vol. A, p. 255-300. MIT Press, 1990. [AHU74] A. Aho, J. Hopcrott, and J. Ullman. The Design and Analysis of Computer Algorithms. Addison-Wesley, Reading MA, 1974. [AHU83] A. Aho, J. Hopcroft, and .1. Ullman. Data Structures and Algorithms. Addison- Wesley, Reading MA, 19g3. [Aig88] [A1TT00] M. Aigner. Combinatorial Search Wiley-Teubner, 1988. Y. Asahiro, K. Iwama, H. Tamaki, and T. Tokuyama. Greedily finding a dense subgraph. J. Algorithms. 34:203-221, 2000. [AK89] E. Aarts and J. Korst. Simulated annealing and Boltzman machines: A stochastic approach to combinatorial optimization and neural computing. John Wiley and Sons, 1989. [AKD83] J. H. Ahrens, K. D. Kohrt, and U. Dieter. Sampling from gamma and Poisson distributions. ACM Trans. Math. Softw., 9(2):255—257, June 1983. [AKSO4] M. Agrawal, N. Kayal, and N. Saxena. PRIMES is in P. Annals of Mathematics, 160:781-793,2004. [AL97] E. Aarts and J. K. Lenstra. Local Search in Combinatorial Optimization. John Wiley and Sons, West Sussex, England, 1997. [AM93] S. Arya and D. Mount. Approximate nearest neighbor queries in fixed dimensions. In Proc. Fourth ACM-S1AMSymp Discrete Algorithms (SODA), p. 271-280, 1993. [AM№ 98] S Arya, D. Mount, N. Netanyahu, R. Silverman, and A. Wu. An optimal algorithm for approximate nearest neighbor searching in fixed dimensions. J. ACM, 45:891-923, 1998.
Список литературы 677 [ЛМО93] [AMWW88] [And98] [AndO5] [АР72] [АРТ79] [Aro98J [ASOO] [Ata83] [Ata84] [Ata98] [Aur9l J [ВагОЗ] [BBF991 [BBPP99] [BCGR92] [BCPB04] [BCW90] [BD99] [BDH97] [BDNOI] R. Ahuja, T. iMagnanti, and .1. Odin. Network Flows. Prentice Hall, Englewood Cliffs NJ, 1993. II. Alt, K. Mchlhorn, H. Wagener, and E. WelzL Congruence, similarity and symmetries of geometric objects. Discrete Comput. Ceom., 3:237-256, 1988. G. Andrews. The Theory of Partitions. Cambridge Univ. Press, 1998. A. Andersson. Searching and priority queues in o(logn) time. In D. Mehta and S. Sahni. editors. Handbook of Data Structures and Applications, p. 39:1 39:14. Chapman and Hall / CRC. 2005. A. Aho and T. Peterson. A minimum distance error-correcting parser for context-free languages. SIAM J. Computing, 1:305-312, 1972. B. Aspvall, M. Plass, and R. Tarjan. A linear-time algorithm for testing the truth of certain quantified boolean formulas. Info. Proc. Letters, 8:121-123, 1979. S. Arora. Polynomial time approximations schemes for Euclidean TSP and other geometric problems. J. ACM, 45:753-782, 1998. P. Agarwal and M. Sharir. Arrangements. In J. Sack and J. Urrutia, editors, Handbook of Computational Geometry, p. 49-1 19. Elsevier, 2000. M. Atallah. A linear time algorithm for the Hausdorff distance between convex polygons. Info. Proc. Letters. 8:207-209, 1983. M. Atallah Checking similarity of planar figures. Internal. J. Comput. Inform. Sci., 13:279-290, 1984. M. Atallah. Algorithms and Theory of Computation Handbook. CRC, 1998. F. Aurenhammer. Voronoi diagrams: a survey of a fundamental data structure. ACM Computing Surveys, 23'345-405, 1991. A Barabasi. Linked: How Everything Is Connected to Everything Else and What It Means. Plume, 2003. V. Bafna, P. Berman, andT. Fujito. A 2-approximation algorithm for the undirected feedback vertex set problem. SIAM J Discrete Math , 12:289- 297, 1999. I. Bomze, M. Budinich. P. Pardalos, and M. Pelillo. The maximum clique problem. In D.-Z. Du and P.M. Pardalos, editors, Handbook of Combinatorial Optimization, vol. A sup., p. 1-74. Kluwer, 1999. D. Berque, R Cecchini, M Goldberg, and R. Rivenburgh. The SetPlayer system for symbolic computation on power sets. J. Symbolic Compulation, 14 645-662, 1992. J. Bojcr, P. Cortese, M. Patrignani, and G. Di Battista. Stop minding your p’s and q’s: Implementing a fast and simple DFS-based planarity testing and embedding algorithm. In Proc. Graph Drawing (GD 03), vol. 2912 LNCS, p. 25-36, 2004. T. Bell, J. Cleary, and I. Witten. Text Compression. Prentice Hall, Englewood Cliffs NJ, 1990. R. Bubley and M. Dyer. Faster random generation of linear extensions. Disc. Math., 201:81-88, 1999. C. Barber, D. Dobkin, and H. Huhdanpaa. The Quickhull algorithm for convex hulls ACM Trans, on Mathematical Software, 22:469—483, 1997. G. Bilardi, P. D’Alberto, and A. Nicolau. Fractal matrix multiplication: a case study on portability of cache performance. In Workshop on Algorithm Engineering (WAE), 2001.
678 Список литературы [BDY06] К. Been, Е. Daiclies, and С. Yap Dynamic map labeling. IEEE Trans. Visualization and Computer Graphics. 12:773-780, 2006. [Ве158] R. Bellman. On a routing problem Quarterly of Applied Mathematics, 16:87-90, 1958. [Веп75| J. Bentley. Multidimensional binary search trees used for associative searching. Communications of the ACM, 18:509-517, 1975. [Веп90] [Веп92а] J. Bentley. More Programming Pearls. Addison-Wesley, Reading MA, 1990. J. Bentley. Fast algorithms for geometric traveling salesman problems. ORSA J. Computing. 4 387—411. 1992. [Веп92Ь] J. Bentley. Software exploratorium: The trouble with qsort. UNIX Review, 10(2):85-93, February 1992. [Веп99] J. Bentley. Programming Pearls. Addison-Wesley, Reading MA, second edition edition, 1999. [Всг89] [ВсгО2] C. Berge Hypergraphs. North-Holland, Amsterdam, 1989. M. Bern. Adaptive mesh generation. In T. Barth and 11 Deconinck, editors, Error Estimation and Adaptive Discretization Methods in Computational Fluid Dynamics, p. 1-56. Springer-Verlag. 2002. [Вег04а] M. Bern. Triangulations and mesh generation. In J. Goodman and J. O’Rourke, editors. Handbook of Discrete and Computational Geometry, p. 563-582. CRC Press, 2004. [ВегО4Ь] D. Bernstein. Fast multiplication and its applications, http://cr.yp.to/arith.html, 2004. [ВЕТТ99] G. Di Battista, P. Eades, R. Tamassia, and I. Tollis. Graph Drawing: Algorithms for the Visualization of Graphs. Prentice-Hall, 1999. [BFOO] M. Bender and M. Farach. The LCA problem revisited. In Proc. 4th Latin American Svmp. on Theoretical Informatics, p. 88-94. Springer-Verlag LNCS vol. 1776, 2000. [BFP+72] M. Blum, R. Floyd, V. Pratt, R. Rivest, and R. Tarjan. Time bounds for selection. J. Computer and System Sciences, 7:448—461, 1972. [BFV07] G. Brodal, R. Fagerberg, and K. Vinther. Engineering a cache-oblivious sorting algorithm. ACM J. of Experimental Algorithm ics, 12, 2007. [BG95] J. Berry and M. Goldberg. Path optimization and near-greedy analysis for graph partitioning: An empirical study. In Proc. 6th ACM-SIAM Symposium on Discrete Algorithms, p. 223-232, 1995. [BGS95] M Bellare, O. Goldreich, and M. Sudan. Free bits, PCPs, and nonapproximability - towards tight results. In Proc. IEEE 36th Symp. Foundations of Computer Science, p. 422—431. 1995. [ВН90] F. Buckley and F. Harary. Distances in Graphs. Addison-Wesley, Redwood City, Calif., 1990. [BHOI] G. Barequct and S. Har-Peled. Efficiently approximating the minimumv. bounding box of a point set in three dimensions. J. Algorithms, 38:91-109, 2001. [BHROO] L. Bcrgroth, H. Hakonen, and T. Raita. A survey of longest common subsequence algorithms. In Proc. String Processing and Information Retreival (SPIRE), p. 39—48, 2000.
Список литературы 679 [BIK+04] Н. Bronnimann, J. lacono, J. Katajainen, P. Morin, J. Morrison, and G. Toussaint. Space-efficient planar convex hull algorithms, Theoretical Computer Science, 321:25-40, 2004. [BJL+94] A. Blum, T. Jiang, M. Li, J. Tromp, and M. Yanakakis. Linear approximation of shortest superstrings. J. ACM. 41:630-647, 1994. [BJL06] C. Buchheim. M. Junger, and S. Leipert. Drawing rooted trees in linear time. Software: Practice and Experience. 36:651-665, 2006. [BJLM83] J. Bentley. D. Johnson, F. Leighton, and C. McGeoch. An experimental study of bin packing. In Proc 21st Allerton Conf, on Communication. Control, and Computing, p. 51-60, 1983. [BK04] Y. Boykov and V Kolmogorov. An experimental comparison of mincut/ max-flow algorithms for energy minimization in vision. IEEE Trans. Pattern Analysis and Machine Intelligence (PAMI), 26:1124-1137, 2004. [BKRV00] A. Blum. G. Konjevod, R. Ravi, and S. Vempala. Semi-definite relaxations for minimum bandwidth and other vertex-ordering problems. Theoretical Computer Science. 235:25—42. 2000. [BL76] K. Booth and G Lueker. Testing for the consecutive ones property, interval graphs, and planarity using PQ-tree algorithms. J Computer System Sciences, 13:335-379, 1976. [BL77] В. P. Buckles and M. Lybanon. Generation of a vector from the lexicographical index. ACM Trans. Math Softw., 3(2): 180-182, June 1977. [BLS91] D. Bailey, K. Lee, and H. Simon. Using Strassen’s algorithm to accelerate the solution of linear systems. J. Supercomputing, 4:357-371, 1991 [Blu67] H. Blum. A transformation for extracting new descriptions of shape. In W. Wathen- Dunn, editor, Models for the Perception of Speech and Visual Form, p. 362-380. MIT Press. 1967. [BLW76] N. L. Biggs. E. K. Lloyd, and R. J. Wilson. Graph Theory 1736-1936. Clarendon Press, Oxford, 1976 [BM53] G. Birkhoff and S. MacLane. A survey of modern algebra. Macmillian, New York. 1953. [BM77] R. Boyer and J. Moore. A fast string-searching algorithm. Communications of the 4СЛЛ 20:762-772. 1977. [BM89] J. Boreddy and R. N. Mukherjee. An algorithm to find polygon similarity. Inform. Process. Lett., 33(4):205—206, 1989. [BM01] E. Bingham and H. Mannila. Random projection in dimensionality reduction: applications to image and text data. In Proc. ACM Conf Knowledge Discovery and Data Mining (KDD), p. 245 -250, 2001. [BM05] A. Broder and M. Mitzenmacher. Network applications of bloom filters: A survey. Internet Mathematics, 1:485-509, 2005. [BO79] J. Bentley and T. Ottmann. Algorithms for reporting and counting geometric intersections. IEEE Transactions on Computers, (-28:643-647, 1979. [BO83] M. Ben-Or. Lower bounds for algebraic computation trees In Proc. Fifteenth ACM Symp. on Theory of Computing, p. 80 -86, 1983. [BolOl] B. Bollobas. Random Graphs. Cambridge Univ. Press, second edition, 2001.
680 Список литературы [ВР76] Е. Balas and М. Padbcrg. Set partitioning - — a survey. 57/1Л7 Review, 18:710-760. 1976. [BR80] I. Barrodale and F. D. K. Roberts. Solution of the constrained L, linear approximation problem. ACM Trans. Math. Soflw.. 6(2^:231-235, June 1980. [BR95J A. Binstock and J. Rex. Practiced Algorithms for Progi ammers. Addison-Wesley, Reading MA, 1995. |Bra99j R. Bracewell. The Fourier Transform and its Applications. McGraw-Hill, third edition, 1999. [Bre73] R. Brent. Algorithms for minimization without derivatives. Prentice-Hall, Englewood Cliffs NJ, 1973. [Bre74] R. P. Brent. A Gaussian pseudo-random number generator. Comm. ACM, 17(12):704- 706, December 1974. [Bre79] D. Brelaz. New methods to color the vertices of a graph. Comm ACM. 22:25 1-256, 1979. [Bri88J E. Brigham. The Fast Fourier Transform. Prentice Hall, Englewood Cliffs NJ, facimile edition, 1988. [Bro95| F. Brooks. The Mythical Man-Month. Addison-Wesley, Reading MA, 20th anniversary edition, 1995. [Bru()7] P. Brucker. Scheduling Algorithms. Springer-Verlag, fifth edition, 2007. [Brz64] J. Brzozowski. Derivatives of regular expressions.ACM, 11:481—494, 1964. [BS76] J. Bentley and M. Btamos. Divide-and-conquer in higher-dimensional space. In Proc. Eighth ACMSymp. Theory of Computing, p. 220-230, 1976. [BS86] G. Berry and R. Sethi. From regular expressions to deterministic automata. Theoretical Computer Science, 48:1 17-126, 1986. [BS96] E Bach and J. Shallit. Algorithmic Number Theory: Efficient Algorithms, vol. 1. MIT Press, Cambridge MA, 1996. [ BS97] R. Bradley and S. Skiena. Fabricating arrays of strings. In Proc. First Int. Conf Computational Molecular Biology' (RECOMB ’97), p. 57-66. 1997. [BS07] A. Barvinok and A. Samorodnitsky. Random weighting, asymptotic counting and inverse isoperimetry. Israel Journal of Mathematics, 158:159-191,2007. [BT92J J. Buchanan and P. Turner. Numerical methods and analysis. McGraw -Hill, New York, 1992. [Buc94] A. G. Buckley. A Fortran 90 code for unconstrained nonlinear minimization. ACM Trans. Math. Sojtw., 20(3):354-372, September 1994. [BvG99] S. Baase and A. van Gelder. Computer Algorithms. Addison-Wesley, Reading MA. third edition, 1999. [BW91] G. Brightwell and P. Winkler. Counting linear extensions. Order, 3:225-242, 1991. [BW94] M. Burrow's and D. Wheeler. A block sorting lossless data compression algorithm. Technical Report 124, Digital Equipment Corporation, 1994. [BW00] R. Borndorfer and R. Weismantel. Set packing relaxations of some integer programs. Math. Programming A, 88:425—450, 2000. [Can87] J. Canny. The complexity of robot motion planning. MIT Press, Cambridge MA, 1987.
Список литературы 681 [Cas95] G. Cash. A last computer algorithm for finding the permanent of adjacency matrices J. Mathematical Chemistry, 18:115-119, 1995. [CB04J C. Cong and D. Bader. The Euler tour technique and parallel rooted spanning tree. In hit. Conf. Parallel Processing (1CPP), p. 448^157, 2004. [CC92] S. Carlsson and J. Chen. The complexity of heaps. In Proc. Third ACMSIAM Svmp. on Discrete Algorithms, p. 393-402. 1992. [CC97] W. Cook and W. Cunningham. Combinatorial Optimization. Wiley, 1997. [CC05] S. Chapra and R. Canale. Numerical Methods for Engineers. McGraw-Hill, fifth edition, 2005. [CCDG82J P. Chinn, J. Chvatolva, A. K. Dewdney, and N. E. Gibbs. The bandwidth problem for graphs and matrices - a survey. J. Graph Theory, 6:223-254, 1982. [CCPS98| W. Cook, W. Cunningham, W. Pullcyblank, and A. Schrijver. Combinatorial Optimization. Wiley, 1998. [CD85] B. Chazelle and D. Dobkin. Optimal convex decompositions. In G. Toussaint, editor. Computational Geometry, p. 63-133. North-Holland, Amsterdam, 1985. [CDL86| B. Chazelle, R. Drysdale, and D. Lee. Computing the largest empty rectangle. SIAM J. Computing, 15:300-315, 1986. [CDT95] G. Carpento, M. Dell’Amico, and P. Toth. CDT: A subroutine for the exact solution of large-scale, asymmetric traveling salesman problems. ACM Trans. Math. Sojtw., 21(4):410-415, December 1995. [CE92] B. Chazelle and H. Edelsbrunner. An optimal algorithm for intersecting line segments. J. 4CW, 39:1-54, 1992. [CFC94] C. Cheng, B. Feiring, and T. Cheng. The cutting stock problem — a survey. Int. J. Production Economics, 36:291-305, 1994. [CFR06] D. Coppersmith, L. Fleischer, and A. Rudrea. Ordering by weighted number of wins gives a good ranking for weighted tournaments. In Proc. 17th ACMSIAM Symp. Discrete Algorithms (SODA), p. 776-782, 2006. [CFT99] A. Caprara, M. Fischctti, and P. Toth. A heuristic method for the set covering problem. Operations Research, 47:730-743, 1999. [CFTOOJ A. Caprara, M. Fischctti, and P. Toth. Algorithms for the set covering problem. Annals of Operations Research, 98:353-371, 2000. [CG94| B. Cherkassky and A. Goldberg. On implementing push-relabel method for the maximum flow problem. Technical Report 94-1523, Department of Computer Science, Stanford University, 1994. [CGJ96] E. G. Coffman, M. R. Garey, and D. S. Johnson. Approximation algorithms for bin packing: a survey. In D. Hochbaum, editor, Approximation algorithms. PWS Publishing. 1996. [CGJ98] C.R. Coullard, A.B. Gamble, and P.C. Jones. Matching problems in selective assembly operations. Annals of Operations Research, 76:95-107, 1998. [CGK.+97] C. Chekuri, A. Goldberg, D. Karger, M. Levine, and C. Stein. Experimental study of minimum cut algorithms. In Proc. Symp. on Discrete Algorithms (SODA), p. 324-333, 1997. [CGL85] B. Chazelle, L. Guibas, and D. T. Lee. The power of geometric duality. BIT. 25:76-90, 1985.
682 Список литературы [CGMt98| В. Cherkassky. A. Goldberg, Р. Martin. J. Setubal, and J. Stolfi. Augment or push: a computational study of bipartite matching and unit-capacity flow algorithms. J. Experimental Algorithmics,!, 1998. [CGPS76] ILL. Crane Jr., N. F. Gibbs, W. G. Poole Jr., and P. K. Stockmeycr. Matrix bandwidth and profile reduction. ACM Trans. Math. Softw., 2(4):375 377, December 1976 [CGR99] B. Cherkassky, A. Goldberg, and T Radzik. Shortest paths algorithms: theory and experimental evaluation. Math Prog., 10:129-174, 1999. [CGS99] B. Cherkassky, A. Goldberg, and C. Silverstein. Buckets, heaps, lists, and monotone priority queues. SIAM J. Computing, 28:1326-1346, 1999. [CH06] D. Cook and L. Holder. .Mining Graph Data. Wiley, 2006. [Cha91 ] B. Chazelle. Triangulating a simple polygon in linear time. Discrete and Computational Geometry, 6:4X5-524, 1991. [ChaOO] B. Chazelle. A minimum spanning tree algorithm with inverse-Ackerman type complexity. J. ACM, 47:1028 -1047, 2000. [ChaOl] T. Chan. Dynamic planar convex hull operations in near-logarithmic amortized time. J. A CM, 48:1 12.2001. [Che85] L. P. Chew. Planing the shortest path fora disc in O(nAgn) time. In Proc. First ACM Symp. Computational Geometry, p. 214-220, 1985. [CHL07| M. Crochemore, C. Hancart, and T. Lecroq. Algorithms on Strings. Cambridge University Press, 2007. [Chr76] N Ghristofides. Worst-case analysis of a new heuristic for the traveling salesman problem Technical report. Graduate School of Industrial Administration, Carnegie- Mellon University, Pittsburgh PA, 1976. [Chu97] F. Chung. Spectral Graph Theory. AMS, Providence RI, 1997. [Chv83] V. Chvatal. Linear Programming. Freeman, San Francisco, 1983. [CIPR011 M Crochemore, C. Iliopolous, Y. Pinzon, and J. Reid. A fast and practical bit-vector algorithm for the longest common subsequence problem. Info. Processing Letters. 80:279-285, 2001. [CK.94] A Chetverin and F. Kramer. Oligonucleotide arrays: New concepts and possibilities. Bio/Technology, 12:1093-1099. 1994 [CK07] W. Cheney and D. Kincaid. Numerical Mathematics and Computing. Brooks/Cole, Monterey CA, sixth edition, 2007. [CKSU05] H. Cohn, R. Kleinberg, B. Szegedy, and C. Umans. Group-theoretic algorithms for matrix multiplication, in Proc. 46th Symp. Foundations of Computer Science, p. 379-388, 2005. [CL98] M. Crochemore and T. Lecroq. Text data compression algorithms. In M. J. Atallah, editor, Algorithms and Theory of Computation Handbook, p. 12.1 12.23. CRC Press Inc., Boca Raton, FL, 1998. [Cla92] K. L. Clarkson. Safe and effective determinant evaluation. In Proc. 31st IEEE Symposium on Foundations of Computer Science, p. 387-395, Pittsburgh. PA, 1992. [CLRS011 T. Cormen, C. Leiserson, R. Rivest, and C. Stein. Introduction to Algorithms. MIT Press. Cambridge MA, second edition, 2001.
Список литературы 683 [СМ69] Е. Cuthill and J. McKee. Reducing the bandwidth of sparse symmetric matrices. In Proc. 24th Nat. Conf. ACM, p. 157-172, 1969. [CM96] J. Chcriyan and K. Mehlhorn. Algorithms for dense graphs and networks on the random access computer. Algorithmica, 15:521-549, 1996. [CM99| G. Del Corso and G. Manzini. Finding exact solutions to the bandwidth minimization problem. Computing. 62:189-203, 1999. [Coh94| E. Cohen. Estimating the size of the transitive closure in linear time. In 35th Annual Symposium on Foundations of Computer Science, p. 190 200. IEEE, 1994. [Con711 J. 11. Conway. Regular Algebra and Finite Machines. Chapman and Hall, London, 1971. [Coo71 J S. Cook. The complexity of theorem proving procedures. In Proc Third ACM Symp. Theory of Computing, p. 151-158, 1971. [CP90] R. Carraghan and I’. Pardalos. An exact algorithm for the maximum clique problem. In Operations Research Letters, vol. 9, p. 375-382, 1990. [CP05] R. Crandall and C. Pomerance. Prime Numbers: A Computational Perspective. Springer, second edition, 2005. [CPW98J B. Chen, C. Potts, and G. Woeginger. A review of machine scheduling: Complexity, algorithms and approximability. In D.-Z. Du and P. Pardalos, editors. Handbook of Combinatorial Optimization, vol. 3, p. 21-169. Kluwer, 1998. [CR76] .1. Cohen and M. Roth. On the implementation of Strassen’s fast multiplication algorithm. Acta Information. 6:341-355, 1976. [CR99] W. Cook and A. Rohe. Computing minimum-weight perfect matchings. INFORMS Journal on Computing, 11:138-148, 1999. [CR01] G. Del Corso and F. Romani. Heuristic spectral techniques for the reduction of bandwidth and work-bound of sparse matrices. Numerical Algorithms, 281 17-136. 2001. [CR03J M. Crochemore and W. Rytter. Jewels pfSrringologv.WoM Scientific, 2003. [CS93] J. Conway and N. Sloane. Sphere packings, lattices, and groups. Springer-Verlag, New York, 1993. [CSG05] A. Caprara and J. Salazar-Gonzalez. Laying out sparse graphs with provably minimum bandwidth. INFORMS J. Computing, 17:356-373,2005. [CT65] J. Cooley and J. Tukey. An algorithm for the machine calculation of complex Fourier scries. Mathematics of Computation, 19:297-301, 1965. [CT92] Y. Chiang and R. Tamassia. Dynamic algorithms in computational geometry. Proc. IEEE, 80:1412-1434, 1992. [CW90] D. Coppersmith and S. Winograd. Matrix multiplication via arithmetic progressions. J. Symbolic Computation, p. 251-280, 1990. [Dan63] G. Dantzig. Linear programming and extensions. Princeton University Press, Princeton NJ, 1963. [Dan94] V. Dancik. Expected length of longest common subsequences. PhD. thesis, Univ, of Warwick, 1994. [DB74] G. Dahlquist and A. Bjorck. Numerical Methods. Prentice-Hall, Entilewood Cliffs N.I, 1974.
684 Список литературы [DB86] [dBDK+98] [dBvKOSOO] [DEKM98] [Den05] [Dey06] [DF79] [DFJ54] [dFPI’90] [DGFD02] [DGKK79] [DH92] [DHSOOj [Die04] [Dij59] [DJ92] [DjiOO] [DJP04] [DL76] [DLR79] G. Davies and S. Bowsher. Algorithms for pattern matching. Software Practice ami Experience, 16:575-601, 1986. M. de Berg, O. Devillers, M. Kreveld, O. Schwarzkopf, and M. Teillaud. Computing the maximum overlap of two convex polygons under translations. Theoretical Computer Science. 31:613-628, 1998. M de Berg, M. van Kreveld, M. Ovcrmars, and O. Schwarzkopf. Computational Geometry: Algorithms ami Applications. Springer-Verlag, Berlin second edition, 2000. R. Durbin, S. Eddy, A. Krough, and G. Mitchison. Biological Sequence Analysis. Cambridge University Press, 1998. L. Y. Deng. Efficient and portable multiple recursive generators of large order. ACM Trans, on Modeling and Computer Simulation, 15:1-13, 2005. T. Dey. Curve and Surface Reconstruction: Algorithms with Mathematical Analysis. Cambridge Univ. Press, 2006. E. Denardo and B. Fox. Shortest-route methods: I. reaching, pruning, and buckets. Operations Research, 27:161 186, 1979. G. Dantzig, D. Fulkerson, and S. Johnson. Solution of a large-scale travelingsalesman problem. Operations Research, 2:393^410, 1954. H. de Fraysseix, J. Pach, and R. Pollack. How to draw a planar graph on a grid. Combinatorica, 10:41-51, 1990. E. Dantsin, A. Goerdt, E. Hirsch, R. Kannan, J. Kleinberg, C. Papadimilriou, P. Raghavan, and U. Schoning. A deterministic (2- 2/(k+\f)n algorithm for k-SAT based on local search Theoretical Computer Science, 289:69-83, 2002. R. Dial, F. Glover, D. Karney, and D. Klingman. A computational analysis of alternative algorithms and labeling techniques for finding shortest path trees. Networks, 9:215-248, 1979. D. Du and F. Hwang. A proof of Gilbert and Pollak’s conjecture on the Steiner ratio. Algorithmica, 7:121—135, 1992. R. Duda, P. Hart, and D. Stork. Pattern Classification. Wiley-Interscicnce, New York, second edition, 2000. M Dietzfelbinger. Primality Testing tn Polynomial Time From Randomized Algorithms to "PRIMES Is in P Springer, 2004. E. W. Dijkstra. A note on two problems in connection with graphs. Numerische Mathematik, 1:269-271, 1959. G. Das and D. Joseph. Minimum vertex hulls for polyhedral domains. Theoret. Comput. Sci., 103:107-135, 1992. H. Djidjev. Computing the girth of a planar graph. In Proc. 27th Int. Colloquium on Automata. Languages and Programming (ICALP), p. 821-831, 2000. E Demaine, T. Jones, and M. Patrascu. Interpolation search for nonindependent data. In Proc. /5th ACM-SIAMSymp. Discrete Algorithms (SODA), p 522-523,2004. D. Dobkin and R. Lipton. Multidimensional searching problems. SIAM J. Computing, 5:181-186, 1976. D. Dobkin, R. Lipton, and S. Reiss. Linear programming is log-space hard for P. Info. Processing Letters, 8:96-97, 1979.
Список литературы 685 [DM 80] D. Dobkin and J. 1. Munro. Determining the mode. Theoretical Computer Science, 12:255-263. 1980. [DM97] K. Daniels and V. Milenkovic Multiple translational containment, part 1: an approximation algorithm.. Ugorithmica, 19:148-182, 1997. [DMBS79] J. Dongarra, C. Moler, J. Bunch, and G. Stewart. UNPACK User 's Guide. SIAM Publications, Philadelphia. 1979. [DMR97] K. Daniels, V. Milenkovic, and D. Roth. Finding the largest area axisparallel rectangle in a polygon. Computational Geometry: Theory and Applications, 7:125-148,1997/ [DN07] P. D'Alberto and A Nicolau. Adaptive Strassen’s matrix multiplication. In Proc. 21st hit. Conf, on Supercomputing, p. 284-292, 2007. [DP73] D. H. Douglas and T. K. Peucker. Algorithms for the reduction of the number of points required to represent a digitized line or its caricature. Canadian Cartographer, 10(2): I12-122, December 1973. [DPS02] J. Diaz, J. Petit, and M. Serna. A survey of graph layout problems.. \CM Computing Surveys, 34:3 13-356, 2002. [DR90J N. Dershowitz and E. Reingold. Calendrical calculations. Software Practice and Experience, 20:899-928, 1990. [DR02] N. Dershowitz and E. Reingold. Calendrical Tabulations: 1900-2200. Cambridge University Press, New York, 2002. [DRR^95] S. Dawson. C. R. Ramakrishnaii. 1. V. Ramakrishnan. K. Sagonas, S. Skiena, T. Swift, and D. S. Warren. Unification factoring for efficient execution of logic programs. In 22nd ACM Symposium on Principles of Programming Languages (POPL 95), p. 247-258, 1995. [DSR00] [DT04] [dVS82] D. Du, J. Smith, and J. Rubinstein.. Idvances in Steiner Trees. Kluwer. 2000. M. Dorigo and T.Stutzle. Ant Colony Optimization. MIT Press. Cambridge MA. 2004. G. de \ Smit. A comparison of three string matching algorithms. Software Practice and Experience, 12:57-66, 1982. [dVV03] S. de Vries and R. Vohra. Combinatorial auctions: A survey. Informs J. Computing, 15:284-309, 2003. [DY94] Y. Deng and C. Yang. Waring’s problem for pyramidal numbers. Science in China (Series A), 37:377-383, 1994. [DZ99] D. Dor and U.Zwick. Selecting the median. SI. IM J Computing, p. 1722-1758, 1999. [DZ01] D. Dor and U. Zwick. Median selection requires (2+е)п comparisons. SIAM J. Discrete Math , 14:312-325, 2001. [Ebe88| [ECW92] .1. Ebert. Computing Eulerian trails. Info. Proc. Leiters. 28:93-97, 1988. V. Estivill-Castro and D. Wood. A survey of adaptive sorting algorithms. ACM Computing Surveys, 24:441-476, 1992. [Ede87] H. Edelsbrunner Ugorithms for Combinatorial Geometry. Springer-Verlag, Berlin, 1987. [Ede06] H. Edelsbrunner. Geometry and Topology for Mesh Generation. Cambridge Univ. Press, 2006. [Edm65] J. Edmonds. Paths, trees, and flowers. Canadian J. Math., 17:449-467, 1965.
686 Список литературы [Edm7l] J. Edmonds. Matroids and the greedy algorithm. Mathematical Programming, 1:126 136. 1971. [EE99] D. Fppstein and J. Erickson. Raising roofs, crashing cycles, and playing pool: applications of a data structure for finding pairwise interactions. Disc Comp. Geometry. 22:569—592, 1999 [EG60] P. Erd'os and T. Gallai. Graphs with prescribed degrees of vertices. Mat. Lapok (Hungarian), 1 1:264-274. 1960. [EG89] Fl. Edelsbrunner and L. Guibas. Topologically sweeping an arrangement J. Computer and System Sciences, 38:165-194, 1989. [EG91] 11. Edelsbrunner and L. Guibas. Corrigendum: Topologically sweeping an arrangement. J. Computer and System Sciences. 42:249-251. 1991. [EGIN92] D. F.ppstein, Z. Galil. G. F. Italiano, and A. Nissenzweig. Sparsification: A technique for speeding up dynamic graph algorithms. In Proc. 33rd IEEE Symp. on Foundations of Computer Science (1 ОС St. p. 60-69. 1992. [EGS86] H. Edelsbrunner. L. Guibas. and J. Stolfi. Optimal point location in a monotone subdivision. SIAM./ Computing, 15:317—340, 1986. [EJ73] J. Edmonds and E. Johnson. Matching, Euler tours, and the Chinese postman. Math Programming, 5:88-124, 1973. [EK72J J. Edmonds and R. Karp. Theoretical improvements in the algorithmic efficiency for network flow problems. J. ACM, 19:248 -264. 1972. [EKA84] M. 1. Edahiro, I. Kokubo, and T. Asano. A new point location algorithm and its practical efficiency - comparison with existing algorithms.. tCM Trans. Graphics, 3:86-109, 1984. [EKS83] 11. Edelsbrunner, D. Kirkpatrick, and R. Seidel. On the shape of a set of points in the plane. IEEE Trans, on Information Theory. IT-29:55 1-559. 1983. [ELOI] S. Ehmann and M. Lin. Accurate and fast proximity queries between polyhedra using convex surface decomposition. Comp. Graphics Forum. 20:500- 510. 2001. [EM94] H. Edelsbrunner and E. Mucke. Three-dimensional alpha shapes. tCM Transactions on Graphics. 13:43-72, 1994. [ENSS98] G. Even, J. Naor. B. Schieber, and M. Sudan. Approximating minimum feedback sets and multi-cuts in directed graphs. Algorithmica, 20:151-174, 1998. [Epp98] D. Eppstein. Finding the k shortest paths. SIAM./. Computing, 28:652-673, 1948. [ES86] H Edelsbrunner and R. Seidel. Voronoi diagrams and arrangements. Discrete and ('omputationai Geometry, 1:25—44, 1986. [ESS93| H. Edelsbrunner, R. Seidel, and M. Sharir. On the zone theorem for hyperplane arrangements. SI IMJ. Computing, 22:418 429, 1993. [ESV96] F. Evans, S. Skiena, and A. Varshney. Optimizing triangle strips for fast rendering. In Proc. IEEE I tsualizalion '96, p. 319—326, 1996. [Eul36] L. Euler. Solutio problematis ad geometriam situs pertinentis. Commentarii Icademiae Scientiarum Pen opolilanae, 8:128-140, 1736. [Eve79a] S. Even. Graph .Ugorilhm.s. Computer Science Press, Rockville MD, 1979. [Eve79b] G. Everstine. A comparison of three rescquencing algorithms for the reduction of matrix profile and wave-front. Int. J. Numerical Methods in Engr.. 14:837-863, 1979.
Список литературы 687 [F48] 1. Fary. On straight line representation of planar graphs. ,k7«. Sci. Malli. Szeged. 11:229-233, 1948. [Fei98] U. Fcige. A threshold of In n for approximating set cover../. ACM. 45:634— 652. 1998? [FF62J L. Ford and D. R. Fulkerson. Flows in Networks. Princeton University Press, Princeton NJ, 1962. [FG95] U. Feige and M. Goemans. Approximating the value of two prover proof systems, with applications to max 2sat and max dicut. In Proc 3rd Israel Symp on Theory of Computing and Systems, p. 182-189, 1995. [FH06] E. Fogel and D. Halperin. Exact and efficient construction of Minkowski sums for convex pohhedra with applications. In Proc 6th Workshop on . \lgorithm Engineering and Experiments (. ILE.MEX), 2006. [FHW07] F Fogel, D. Halperin, and C. Weibel. On the exact maximum complexity of minkowski sums of convex polyhedra. In Proc. 23rd Symp. Computational Geometry p. 319-326, 2007 [FJ05] M. Frigo and S. Johnson. The design and implementation of FFTW3. Proc. IEEE, 93:216-231.2005. [F.IMO93] M. Fredman, D. Johnson. L. McGcoch, and G. Ostheimer. Data structures for traveling salesmen, in Proc. 4th 7 th Symp Discrete Algorithms (SODA), p. 145-154, 1993. [Fle74] H. Fleischner. The square of every two-connected graph is Hamiltonian../ Combinatorial Theory, B, 16:29-34, 1974. [FleSO] R. Fletcher. Practical Methods of Optimization Unconstrained Optimization, vol. 1 John Wiley, Chichester. 1980. [Flo62] [Flo64] [FLPR99] R. Floyd. Algorithm 97 (shortest path). Communications of the ACM, 7:345, 1962. R. Floyd. Algorithm 245 (treesort) Communications of the tCM. 18:701, 1964. M. Frigo. C. Leiserson. H. Prokop, and S. Ramachandran. Cache-oblivious algorithms. In Proc. 40th Symp. Foundations of Computer Science. 1999. [FM71] M. Fischer and A. Meyer. Boolean matrix multiplication and transitive closure. In IEEE ITlhSymp. on Switching and Automata Theory, p. 129- 131, 1971. [FM82] C. Fiduccia and R. Mattheyses. A linear time heuristic for improving network partitions. In Proc 19th IEEE Design Automation Conf., p. 175- 181, 1982. [FN04] K. Fredriksson and G. Navarro. Average-optimal single and multiple approximate siring matching. ACM J. of Experimental Algorithmic*, 9, 2004. [For87] S. Fortune. A sweepline algorithm for Voronoi diagrams. Algorithmica, 2:153-174, 1987. [For04] S. Fortune. Voronoi diagrams and Delauney triangulations. In J. Goodman and J. O’Rourke, editors. Handbook of Discrete and Computational Geometry, p. 513-528. CRC Press, 2004. [FPR99J P. Festa, P. Pardalos, and M. Resende. Feedback set problems. In D.-Z. Du and P.M. Pardalos, editors, Handbook of Combinatorial Optimization, vol. A. Kluwer. 1999. [FPRO1] P. Festa, P. Pardalos, and M. Resende. Algorithm 815: Fortran subroutines for computing approximate solution to feedback set problems using GRASP. ACM Transactions on Mathematical Software, 27:456-464, 2001.
688 Список литературы [FR75] R. Floyd and R. Rivest. Expected time bounds for selection. Communications of the ACM, 18:165-172, 1975. [FR94] M. Furer and B. Raghavachari. Approximating the minimum-degree Steiner tree to within one of optimal. J. Algorithms, 17:409-423. 1994. [Fra79] D. Fraser. An optimized mass storage FFT. ACM Trans Math. Softw., 5(4):500-517. December 1979. [Fre62] | Fre76] E. Fredkin. Trie memory. Communications of the ACM, 3.490 499, 1962. M. Fredman. How good is the information theory bound in sorting? Theoretical Computer Science. 1:355-361, 1976. [FS03] [FSV01] N. Ferguson and B. Schneier. Practical Cryptography. Wiley, 2003. P. Foggia. C. Sansone, and M. Vento. A performance comparison of five algorithms for graph isomorphism. In 3rd 1APR ГС-15 Workshop on Graphbased Representations in Pattern Recognition, 2001. [FT87] M. Fredman and R. Tarjan. Fibonacci heaps and their uses in improved network optimization algorithms.,/. ACM. 34:596-615, 1987. [FvW9.3] S. Fortune and C. van Wyk. Efficient exact arithmetic for computational geometry. In Proc. 9th ACM Symp. Computational Geometry. \>. 163- 172, 1993. [FW77] S. Fiorini and R. Wilson. Edge-colourings of graphs. Research Notes in Mathematics 16, Pitman, London. 1977. [FW93] M. Fredman and D. Willard. Surpassing the information theoretic bound with fusion trees. J. Computer and System Sci., 47:424—436, 1993. [FWH04] E. Folgel. R Wein, and D. Halperin Code flexibility and program efficiency by genericity: Improving CGA L’s arrangements. In Proc. 12th European Symposium on . Ugorithms (ESA 04), p. 664 676, 2004. [Gab76] H. Gabow. An efficient implementation of Edmond’s algorithm for maximum matching on graphs../. . ICM, 23:221-234, 1976. [Gab77] H. Gabow. Two algorithms for generating weighted spanning trees in order. SI I MJ. Computing, 6:139— 150, 1977. [Gal86] Z. Galil. Efficient algorithms for finding maximum matchings in graphs. ACM Computing Surveys. 18 23-38. 1986. [Gal90] K. Gallivan. Parallel Algorithms for Matrix Computations. SIAM. Philadelphia, 1990. [Gas03] [GBDS80] S. Gass. Linear Programming: Methods and Applications. Dover, fifth edition, 2003. B. Golden. L. Bodin, T. Doyle, and W. Stewart. Approximate traveling salesman algorithms. Operations Research, 28:694-71 1, 1980. [GBY91| G. Gonnet and R. Baeza-Yates. Handbook of Algorithms and Data Structures. Addison-Wesley, Wokingham, England, second edition, 1991. [Gen04] J. Gentle. Random Slumber Generation and Monte Carlo Methods. Springer, second edition, 2004. [GGJ77] M. Garey, R. Graham, and D. Johnson. The complexity of computing Steiner minimal trees. SIAM J Appl Math., 32:835-859, 1977. [GG.IK78] M. Garey, R. Graham, D. Johnson, and D. Knuth. Complexity results for bandwidth minimization. SIAM./. Appl. Math., 34:477-495, 1978.
Список литературы 689 [СЗН85] R. Graham and P. Hell. On the history of the minimum spanning tree problem, Inna/s of the History of Computing. 7:43-57. 1985. [GH06] P. Galinierand A. Hertz. A survey of local search methods for graph coloring. ('omputers and Operations Research, 33:2547-2562. 2006. [GHMS93] L. J. Guibas, J. E. Hershberger, J. S. B. Mitchell, and J. S. Snoeyink. Approximating polygons and subdivisions with minimum link paths. Internal. ./. Comput Geom. Appl., 3(41:383-415, December 1993. [GHR95] R. Greenlaw. J. Hoover, and W. Ruzzo. Limits to Parallel Computation; P-completeness theory. Oxford University Press, New York, 1995. |G189| D. Gusfield and R. Irving. The Stable Marriage Problem: structure and algorithms. MIT Press. Cambridge MA, 1989. [GI911 Z. Galil and G Italiano. Data structures and algorithms for disjoint set union problems. ACM Computing Surveys, 23:319-344, 1991. [Gib76] N. E. Gibbs. A hybrid profile reduction algorithm. ACM Trans Math. Softw., 2(41:378-387. December 1976. [Gib85] [GJ77] A. Gibbons. Algorithmic Graph Theory. Cambridge Univ. Press, 1985. M. Garey and D. Johnson. The rectilinear Steiner tree problem is NPcomplete. SIAM J. tppl Math , 32:826-834. 1977. [GJ79] M. R. Garey and D. S. Johnson. Computers and Intractability: A Guide to the theory of NP-completeness. W. H. Freeman, San Francisco, 1979. [GJM02] M. Goldwasser, D. Johnson, and C. McGeoch, editors. Data Structures, Near Neighbor Searches, and Methodology: Eifth and Sixth DIMACS Implementation ( hallenges. vol. 59. AMS, Providence Rl, 2002. [GJPT78] M. Garey. D. Johnson, F. Preparata, and R. Tarjan. Triangulating a simple polygon. Info. Proc. Leiters. 7:175-180, 1978. [GK951 A. Goldberg and R. Kennedy. An efficient cost scaling algorithm for the assignment problem. Math. Programming, 1\ :E53—177, 1995. |GK98] S. Guha and S. Khuller. Approximation algorithms for connected dominating sets. Algorithmica, 20:374-387, 1998 [GKK74] F. Glover, D. Karney, and D. Klingman. Implementation and computational comparisons of primal-dual computer codes for minimum-cost network flow problems. Networks. 4:191-212, 1974. [GKP89] R. Graham, D. Knuth, and O. Patashnik. Concrete Mathematics. Addison-Wesley, Reading MA, 1989. [GKS95] S. Gupta, J. Kececioglu, and A. Schaffer. Improving the practical space and time efficiency of the shortest-paths approach to sum-of-pairs multiple sequence alignment. J Computational Biology. 2:459—472, 1995. [GKT05] D. Gibson, R. Kumar, and A. Tomkins Discovering large dense subgraphs in massive graphs. In Proc. 31st Int. Conf on Very Large Data Bases, p. 721-732. 2005. [GKW06] A. Goldberg, H. Kaplan, and R. Werneck. Reach for A*: Efficient point-topoint shortest path algorithms In Proc. 8lh Workshop on Algorithm Engineering and Experimentation (ALENEX), 2006. [GKW07] A. Goldberg, H. Kaplan, and R.Werneck. Better landmarks within reach. In Proc 9th Workshop on Algorithm Engineering and Experimentation ( ILENEX), p. 38-51, 2007. 2.3 Зак 3741
690 Список литературы [GL96] G. Golub and C. Van Loan. Matrix Computations. Johns Mopkins University Press, third edition, 1996. [Glo90] [GM86] [GM91] F. Glover Tabu search: A tutorial. Interfaces, 20 (4):74—94, 1990. G. Gonnet and J. 1. Munro. Heaps on heaps. SIAM J. Computing, 15:964- 971, 1986. S. Ghosh and D. Mount. An output-sensitive algorithm for computing visibility graphs. SIAM J Computing, 1991. [GMPV06] F. Goines, C. Meneses, P. Pardalos. and G. Viana. Experimental analysis of approximation algorithms for the vertex cover and set covering problems. Computers and Operations Research. 33:3520-3534, 2006. [GO04] J. Goodman and J O'Rourke, editors. Handbook of Discrete and Computational Geometry. CRC Press, second edition, 2004. [Goe97] M. Goemans. Semidefinite programming in combinatorial optimization. Mathematical Programming, 79:143-161, 1997. [Gol93] L. Goldberg. Efficient Algorithms for Listing Combinatorial Structures. Cambridge University Press, 1993. [Gol97] A. Goldberg. An efficient implementation of a scaling minimum-cost flow algorithm. J Ugorithms, 22:1-29, 1997. [GolOl] A. Goldberg. Shortest path algorithms: Engineering aspects. In 12th Internationa! Symposium on . Ugorithms and Computation, number 2223 in LNCS, p. 502-513. Springer, 2001. [Gol04] M. Golumbic. Algorithmic Graph Theory and Perfect Graphs, vol. 57 of Annals of Discrete Mathematics. North Holland, second edition. 2004. [Gon07] T. Gonzalez. Handbook of Approximation Algorithms and Metaheurislics. Chapman- Hall I CRC, 2007. [GP68] E. Gilbert and H. Pollak. Steiner minimal trees. SIAM J. Applied Math., 16:1-29, 1968. [GP79J B. Gates and C. Papadimitriou. Bounds for sorting by prefix reversals. Discrete Mathematics, 27:47-57, 1979. [GP07] G. Gutin and A. Punnen. The Traveling Salesman Problem and Ils V ariations. Springer, 2007. [GPS76] N. Gibbs, W. Poole, and P. Stockmeyer. A comparison of several bandwidth and profile reduction algorithms. ACM Trans. Math. Software, 2:322 330, 1976. [Gra53] [Gra72] F. Gray. Pulse code communication. US Patent 2632058, March 17, 1953. R. Graham. An efficient algorithm for determining the convex hull of a finite planar point set. Info. Proc. Letters. 1:132-133, 1972. [Gri89] [GS62] D. Gries. The Science of Programming. Springer-Verlag, 1989. D. Gale and L. Shapely. College admissions and the stability of marriages.. Imerican Math. Monthly, (O:9-\A, 1962. [GS02] R. Giugno and D. Shasha. Graphgrep : A fast and universal method for querying graphs. In International Conference on Pattern Recognition (1CPR), vol. 2, p. 112-115,2002. [GT88] A. Goldberg and R. Tarjan. A new approach to the maximum flow problem. J. ACM, p. 921-940, 1988.
Список литературы 691 [GT94] [GT05] T. Genscn and B. Toft. Graph Coloring Problems. Wiley, 1994. M. Goodrich and R. Tamassia. Data Structures and Algorithms in Jara. Wiley, fourth edition, 2005. [GTV05] M. Goodrich. R. Tamassia, and L. Vismara. Data structures in JDSL. In D. Mehta and S. Sahni, editors. Handbook of Data Structures and Applications, p. 43:1—43:22. Chapman and Hall' CRC, 2005. [Gup66] R. P. Gupta. The chromatic index and the degree of a graph. Notices of the Amer. Math. Soc., 13:719, I966. [Gus94] D. Gusfield. Faster implementation of a shortest superstring approximation. Info. Processing Letters, 51:271-274, 1994. [Gus97] D. Gusfield. Algorithms on Strings, frees, and Sequences: Computer Science and Computational Biology. Cambridge University Press, 1997. [GW95] M. Goemans and D. Williamson, ,878-approximation algorithms for MAX CUT and MAX2SAT.J. ACM. 42.1115-1145, 1995. [GW96] 1. Goldberg and D. Wagner. Randomness and the Netscape browser. Dr. Dobb's Journal, p. 66-70. 1996. [GW97] T. Grossman and A. Wool. Computational experience with approximation algorithms for the set covering problem. European J. Operational Research, 101, 1997. [Hai94] E. Haines. Point in polygon strategies. In P. Heckbert. editor. Graphics Genies II', p. 24—46. Academic Press, 1994. [Hal04] D. Halperin. Arrangements. In .1. Goodman and J. O’Rourke, editors. Handbook of Discrete and Computational Geometry, chapter 24, p. 529-562. CRC Press, Boca Raton, FL, 2004. [Ham87] R. Hamming. Numerical Methods for Scientists and Engineers. Dover, second edition. 1987 [Has82] H. Hastad. Clique is hard to approximate within n e. Acta Mathematica, 182:105-142, 182. [Has97] J. Hastad. Some optimal inapproximability results. In Proc. 29th ACM Symp. Theory of Comp., p. 1-10, 1997. [HD80] P. Hall and G. Dowling. Approximate string matching. ACM Computing Survevs, 12:381—402, 1980. [HDD03] M. Hilgemeier, N Drechsler, and R. Drchsler. Fast heuristics for the edge coloring of large graphs. In Proc. Euromicro Symp. on Digital Systems Design, p. 230-239, 2003. [HdITOI] J. Holm, K. de lichtenberg, and M. Thorup. Poly-logarithmic deterministic fully- dynamic algorithms for connectivity, minimum spanning tree, 2-edge, and biconnectivity. J. ACM, 48:723-760, 2001. [Held] M. Held. VRONI: An engineering approach to the reliable and efficient computation of Voronoi diagrams of points and line segments. Computational Geometry: Theory and. ipplications, 18:95-123, 2001. [HFN05] H. Hyyro, K. Fredriksson, and G. Navarro. Increased bit-parallelism for approximate and multiple string matching. ACM J. of Experimental Algorithmies, 10, 2005. [HG97] P. Heckbert and M. Garland. Survey of polygonal surface simplification algorithms. SIGGRAPH 97 Course Notes, 1997.
692 Список литературы [ННОО] 1. Hanniel and D. Halperin. Two-dimensional arrangements in CGAL and adaptive point location for parametric curves. In Proc. 4th International Workshop on llgonihm Engineering (WAE). LNCSv. /96'2. p. 171 182.2000. [HHS98] T. Haynes, S. Hedetniemi, and P. Slater. Fundamentals of Domination in Graphs. CRC Press. Boca Raton, 1998. |Hir75] D. Hirschberg. A linear-space algorithm for computing maximum common subsequences. Communications of the ACM, 18:341 343, 1975. [HK73J J. Hopcroft and R. Karp. An n ’ algorithm for maximum matchings in bipartite graphs. .87 Lt/./. Computing, 2:225 -231. 1973. [НК90] D. P. Huttenlocher and K. Kedem. Computing the minimum Hausdorff distance for point sets under translation. In Proc 6th Anna. ACMSympos. Comput. Geom.. p. 340-349. 1990. [HLD04] W. Hermann, J. Leydokl, and G. Derfinger. Automatic Nonuniform Random 1 ariale Generation. Springer, 2004. [НМ83| S Hertel and k. Mehlhorn. Fast triangulation of simple polygons. In Proc. 4th Internal. Conf Found. Comput Theory, p. 207-218. Lecture Notes in Computer Science, Vol. 158, 1983. [НМ99] X. Huang and A Madan. Cap3: A DNA sequence assembly program. Genome Research. 9:868-877, 1999. [HMS03] J. Hershberger, M. Maxel, and S. Suri. Finding the к shortest simple paths: A new algorithm and its implementation. In Proc. 5th Workshop on Algorithm Engineering and Experimentation (Al.EN EX), 2003. [HMU06| J. Hopcroft, R. Motwani, and .1. Ullman. Introduction to Automata Theory. / angttages. and Compulation Addison-Wesley, third edition. 2006 [Ноа6|] C. A. R. Hoare. Algorithm 63 (partition) and algorithm 65 (find). Communications of the \CM, 4:321 322, 1961. [Ноа62] [Нос96] C A. R. Hoare. Quicksort. Computer Journal. 5:10-15, 1962. D. Hochbaum, editor. Approximation Algorithms for NP-hard Problems. PWS Publishing, Boston, 1996. [Hof82] С. M. Hoffmann. Group-theoretic algorithms and graph isomorphism, vol. 136 of Lecture Noles in Computer Science. Springer-Verlag Inc.. New York, 1982. [Но175] J H. Holland.. idaptation in Natural and trlijicial Systems. University of Michigan Press. Ann Arbor, 1975. [Но181] 1. Holyer. The N P-completeness of edge colorings. SIAM J. Computing. 10:718-720, 1981. [Но192] [Нор71 ] J. H. Holland. Genetic algorithms. Scientific American, 267( 1 ):66-72, July 1992. J. Hopcroft. An «logn algorithm for minimizing the states in a finite automaton. In Z. Kohavi, editor. The theory of machines and computations, p. 189-196. Academic Press. New York, 1971. [Hor80J R. N. Horspool. Practical fast searching in strings. Software Practice and Experience, 10.501 -506, 1980. [НР73] [HPS+05] F. Harary and E. Palmer. Graphical enumeration. Academic Press, New York. 1973. M. Holzer, G. Prasinos. F. Schulz, D.Wagner. and C. Zaroliagis. Engineering planar separator algorithms. In Proc. /3th European Svmp. on Algorithms (ES 1), p. 628-637, 2005.
Список литературы 693 [HRW92| R. Hwang, D. Richards, and Р.Winter. The Steiner Tree Problem, vol 53 of Annuls of Discrete Mathematics. North Holland, Amsterdam, 1992. [HS77] J. Hunt and T. Szymanski. A fast algorithm for computing longest common subsequences. Communications of the ACM, 20:350—353. 1977. [HS94] J. Hershberger and J. Snoeyink. An O(n log n} implementation of the Douglas- Peucker algorithm tor line simplification. In Proc Kith Annii l( MSympos. Сотри! Geom p. 383-384 1994. [HS98] J. Hershberger and .1. Snoeyink. Cartographic line simplification and polygon CSG formulae in O(n\og*n) time. Computational Geometry: Theorv and Applications, 11.175-185.1998. [HS99] J. Hershberger and S. Suri. An optimal algorithm for Euclidean shortest paths in the plane. SIAM.J. Computing, 28:22\5-2256. 1999. [HSS87J J. Hopcroft, J. Schwartz, and M. Sharir. Planning, geometry, and complexity of robot motion. Ablex Publishing, Norwood N.I, 1987. [HSS07] R. Hardin, N. Sloane, and W. Smith. Maximum volume spherical codes. http://www.research.ait.cwm/~njas/iniixvolumes/, 2007. [HSWW05] M. Holzer, F. Schultz, D. Wagner, and T. Willhalm. Combining speed-up techniques for shortest-path computations. ACM J. of Experimental Algorithmic*. 10, 2005. [НТ73а] J Hopcroft and R Tarjan. Dividing a graph into triconnected components. SIAM J Computing, 2:135-158, 1973. [НГ73Ь] J. Hopcroft and R. Tarjan. Efficient algorithms for graph manipulation. Communications of the ACM. 16'372-378. 1973 [НТ74] [НТ84] .1 Hopcroft and R. Tarjan. Efficient planarity testing. J. ACM, 21 549 568, 1974. D. Harel and R. E. Tarjan. Fast algorithms for finding nearest common ancestors. SIAM J. Сотри!., 13:338-355. 1984. [НиЬОб] M. Huber. Fast perfect sampling from linear extensions. Disc. Math., 306 420 428. 2006 [Huf52] D. Huffman. A method for the construction of minimum-redundancy codes. Proc, of the IRE, 40:1098-1101, 1952. [HUW02] E. Haunschmid, C. Ueberhuber, and P. Wurzinger Cache oblivious high performance algorithms for matrix multiplication. 2002. [HW74] J. E. Hopcroft and J. K. Wong. Linear time algorithm for isomorphism of planar graphs. In Proc. Sixth Annual ACM Symposium on Theorv of Computing. p. 172-184. 1974. [HWA+03] X. Huang. J. Wang, S. Aluru, S. Yang, and L Hillier. PCAP' A wholegenome assembly program. Genome Research, 13:2164-2170, 2003. [HWK94] T. He. S.Wang, and A. Kaufman.Wavelet-based volume morphing. In Proc IEEE 1 ’isualEation '94. p. 85-92, 1994. [IK75] O. Ibarra and C. Kim. Fast approximation algorithms for knapsack and sum of subset problems. J ACM, 22:463-468, 1975. [IM04] P. Indyk and J. Matousek. Low-distortion embeddings of finite metric spaces. In J. Goodman and J. O’Rourke, editors, Handbook of Discrete and Computational Geometry. CRC Press, 2004.
694 Список литературы [lnd98] P Indyk. Faster algorithms for string matching problems: matching the convolution bound. In Proc. 39th Symp. Foundations of Computer Science, 1998. [lnd04] P. Indyk. Nearest neighbors in high-dimensional spaces. In J. Goodman and J. O’Rourke, editors. Handbook of Discrete and Computational Geometry, p. 877—892. CRC Press. 2004. [1R78] A. Ilai and M. Rodeh. Finding a minimum circuit in a graph. SIAM ) Computing, 7.413 -423. 1978. [lta78] Р92] [Jac89] A. ilai. Two commodity flow.,/. ACM, 25:596-611, 1978. J. JaJa. An Introduction to Parallel Algorithms. Addison-Wesley, 1992. G. Jacobson. Space-efficient static trees and graphs. In Proc. Symp. Foundations of Computer Science (FOCS), p 549-554, 1989. [JAMS91] D. Johnson, C. Aragon, C. McGeoch, and D. Schevon. Optimization by simulated annealing: an experimental evaluation; part 11. graph coloring and number partitioning. In Operations Research, vol. 39, p. 378-406, 1991. [Jar73] R. A. Jarvis. On the identification of the convex hull of a finite set of points in the plane. Info. Proc. Letters, 2:18-21, 1973. [JD88] A. Jain and R. Dubes. Algorithms for Clustering Data. Prentice-Hall, Englewood Cliffs NJ, 1988. [JLROOJ [JM93] S. Janson, T. Luczak, and A. Rucinski. Random Graphs. Wiley, 2000. D. Johnson and C. McGeoch, editors. Network Flows and Matching: First DIMACS Implementation Challenge, vol. 12. American Mathematics Society, Providence Rl. 1993. [JM03] [Joh63] M. Jungerand P. Mutzel. Graph Drawing Software. Springer-Verlag, 2003. S. M. Johnson. Generation of permutations by adjacent transpositions. Math. Computation. 17:282-285, 1963. [Joh74] D. Johnson. Approximation algorithms for combinatorial problems.,/. Computer and System Sciences. 9:256-278. 1974. [Joh90] D S. Johnson. A catalog of complexity classes. In J. van Leeuwen, editor, Handbook of Theoretical Computer Science: Algorithms and Complexity, vol. A, p. 67-162. MIT Press, 1990. [Jon86] D. W. Jones. An empirical comparison of priority-queue and event-set implementations. Communications oj the ACM, 29:300-311. 1986. [Jos99] N. Josuttis. The C++ Standard Library: A tutorial and reference. Addison- Wesley. 1999. [JR93] T. Jiang and B. Ravikumar. Minimal NFA problems are hard. SIAM J. Computing. 22 1117-1141, 1993. [JSOI] A. Jagotaand L. Sanchis. Adaptive, restart, randomized greedy heuristics for maximum clique. J. Heuristics, 7:1381-1231,2001. [JSVOI] M. Jerrum, A. Sinclair, and E. Vigoda. A polynomial-time approximation algorithm for the permanent of a matrix with non-negative entries. In Proc 33rd ACM Symp Theory of Computing, p. 712-721, 2001. [JT96] D Johnson and M Trick. Cliques. Coloring, and Satisfiability: Second DIM ACS Implementation Challenge, vol. 26. AMS, Providence Rl, 1996.
Список литературы 695 [КАОЗ] P. Ko and S. Alum. Space-efficient linear time construction of suffix arrays,. In Proc. 14th Symp. on Combinatorial Pattern Matching (CPM), p. 200- 210. Springer-Verlag LNCS. 2003. [Ка1167] D. Kahn. The Code breakers: the story of secret writing. Macmillan, New York, 1967. [Каг72] R. M. Karp. Reducibility among combinatorial problems. In R. Miller and J. Thatcher, editors. Complexity of Computer Computations, p. 85—103 Plenum Press, 1972. [Каг84] N. Karmarkar. A new polynomial-time algorithm for linear programming. Combinatorica, 4:373-395, 1984. [Каг96] H. Karloff. How good is the Goemans-VVilliamson MAX CUT algorithm? In Proc. Twentv-Eighth Annual ACM Symposium on Theory of Computing, p. 427—434, 1996. [КагОО] [KeiOO] D. Karger. Minimum cuts in near-linear time. J. ACM, 47:46-76. 200. M. Keil. Polygon decomposition. In J.R. Sack and J. Urrutia, editors, Handbook of Computational Geometry, p. 491-5 18. Elsevier, 2000. [KGV83] S. Kirkpatrick. C. D. Gelatt, Jr., and M. P. Vecchi. Optimization by simulated annealing. Science, 220:671-680, 1983. [Kha79] L. Khachian. A polynomial algorithm in linear programming. Soviet Math. DokL, 20:191-194. 1979. [Kir79] D. Kirkpatrick. Efficient computation of continuous skeletons. In Proc. 20th IEEE Symp. Foundations of Computing, p. 28-35, 1979. [Kir83] D. Kirkpatrick. Optimal search in planar subdivisions. SIAM J Computing, 12:28-35, 1983. [KKT95] D. Karger, P. Klein, and R. Tarjan. A randomized linear-time algorithm to find minimum spanning trees. J. .46717,42:321-328. 1995 [KL70] B. W. Kernighan and S. Lin. An efficient heuristic procedure for partitioning graphs. The Bell System Technical Journal, p. 291- 307, 1970. [KM72] V. Klee and G. Minty. How good is the simplex algorithm. In Inequalities HI. p 159-172, New York, 1972. Academic Press. [KM95] J. D. Kececioglu and E. W. Myers. Combinatorial algorithms for DNA sequence assembly. Algorithmica, 13( 1/2):7—51, January 1995. [KMP77] D. Knuth, J. Morris, and V. Pratt. Fast pattern matching in strings. SIAM J Computing, 6:323-350, 1977. [KMP+04] L. Kettner, K. Mehlhorn, S. Pion, S. Schirra, and C Yap. Classroom examples of robustness problems in geometric computations. In Proc. 12th European Symp. on Algorithms (ESA ’04), p. 702-713. www.mpiinf. mpg.de/~mehlhorn/ftp/ClassRoomExamples.ps, 2004. [KMS96] J. Komlos, Y. Ma, and E. Szemeredi. Matching nuts and bolts in o(" log «) time. In Proc 7th Symp Discrete Algorithms (SODA), p 232-241, 1996. [KMS97] S. Khanna, M. Muthukrishnan, and S. Skiena. Efficiently partitioning arrays. In Proc. 1CALP '97, vol. 1256, p. 616-626. Springer-Verlag LNCS, 1997. [Knu94] D. Knuth. The Stanford GraphBase: a platform for combinatorial computing. ACM Press, New York, 1994.
696 Список литературы [Knu97a] D. Knuth, /'he Irt of Computer Programming, 1 I: Fundamental Ugorithms. Addison-Wesley. Reading MA, third edition, 1997. [Knu97b] D. Knuth. The Art of Computer Programming. Г. 2: Seminumerical Algorithms. Addison-Wesley, Reading MA, third edition. 1997. [Knu98] D. Knuth. Hie lit of Computer Programming. J< 3 Sorting and Searching. Addison- Wesley. Reading MA. second edition, 1998. [KnuO5a] D. Knuth. The Art of Computer Programming, 1' 4 Fascicle 2: Generating All Tuples and Permutations. Addison Wesley, 2005. [Knu05b] D. Knuth. The \rt of Computer Programming. Г 4 Fascicle 3 ; Generating All Combinations and Partitions. Addison Wesley. 2005. [Knu06] D. Knuth. The Art of Computer Programming. 1'. 4 Fascicle 4: Generating . Ill Trees; History pfCombinationalorial Generation. Addison Wesley. 2006. [KO63] A. Karatsuba and Yu. Ofman. Multiplication of multi-digit numbers on automata. ,$ov. Pins. DokL, 7:595-596. 1963. | Koe05] H Koehler. A contraction algorithm for finding minimal feedback sets. In Proc. 28th tustralasian Computer Science Conference (ACSC), p. 165- 174, 2005. [KOS91] A. Kaul. M. A. O'Connor, and V. Srinivasan. Computing Minkowski sums of regular polygons. In Proc. 3rdCanad. Conf. Comput. Geom.. p. 74 77, 199] [KP98] J. Kececioglu and J. Pecqueur. Computing maximum-cardinality matchings in sparse general graphs. In Proc. 2nd Workshop on Algorithm Engineering, p. 121 132, 1998. [KPP04] [KR87] H. Kellerer, U. Pferschy, and P. Pisinger. Knapsack Problems. Springer, 2004. R. Karp and M. Rabin. Efficient randomized pattern-matching algorithms. HIM .1 Research and Development. 3 1:249-260, 1987. [KR91] A. Kanevsky and V. Ramachandran. Improved algorithms for graph fourconnectivity. ./ Comp. Sys. Sci., 42:288-306, 1991. [Kru56] .1.13 Kruskal. On the shortest spanning subtree of a graph and the traveling salesman problem. Proc, of the American Mathematical Society, 7:48- 50, 1956. [KS74] D.E. Knuth and J.L. Szwarcfiter. A structured program to generate all topological sorting arrangements. Information Processing l etters, 2:153- 157 1974. [KS85] M. Keil and J. R. Sack. Computational Geometry; Minimum decomposition of geometric objects, p. 197-216. North-Holland. 1985. [KS86] D. Kirkpatrick and R. Siedel. The ultimate planar convex hull algorithm? SIAM J. Computing, 15:287-299, 1986. [KS90] K. Kedem and M. Sharir. An efficient motion planning algorithm for a convex rigid polygonal object in 2-dimensional polygonal space. Discrete andC omputational Geometry. 5:43—75, 1990. [KS99] D. Kreher and D. Stinson. Combinatorial Algorithms; Generation, Enumeration, and Search. CRC Press, 1999. [KS02] M. Keil and J. Snoeyink. On the time bound for convex decomposition of simple polygons. Int. J. Comput Geometry Appl., 12:181-192, 2002. [KS05a] H. Kaplan and N. Shafrir. The greedy algorithm for shortest superstrings. Info. Proc Letters, 93:13-17, 2005.
Список литературы 697 [KS05b] J Reiner and D. Spielman. A randomized polynomial-time simplex algorithm for linear programming. Electronic CoUoqtiim on Computational ( omplexity, 156:17, 2005. [KS07] H Kautz and B. Selman. The state of SAT. Disc. Applied Math., 155:1514- 1524, 2007. [KSB05] J. Karkkainen. P. Sanders, and S. Burkhardt. Linear work suffix array construction. J..K W, 2005. [KSBD07] H. Kautz, B. Selman, R. Brachman. and T. Dietterich. Satisfiability Testing. Morgan and Claypool, 2007. [KSPP03] D Kim, J. Sim, H. Park, and K. Park. Linear-time construction of suffix arrays. In Proc. 14th Symp Combinatorial Pattern Matching (CPM), p. 186-199, 2003. [KST93] J. Kobler, U. Schoning, and J. Turan. I he Graph Isomorphism Problem: its structural complexity. Birhauser, Boston, 1993. [KSV97] D. Keyes, A. Sameh, and V. Venkatarishnan. Parallel Numerical Ugorithms. Springer, 1997. [KT06] J. Klcinberg and E. Tardos.. llgorilhm Design. Addison Wesley, 2006. [Kuh75] H. V Kuhn. Steiner’s problem revisited. In G. Dantzigand B. Eaves, editors, Studies in Optimization, p. 53-70. Mathematical Association of America, 1975. [Kur30] K. Kuratowski. Sur le probl'eme des courbes gauches en topologie. Fund. Math., 15:217-283, 1930. | К W01 ] M. Kaufmann and D. Wagner. Draw ing Graphs: Methods and Models. Springer- Verlag. 2001. [Kwa62] M. Kwan. Graphic programming using odd and even points. Chinese Math., 1:273-277, 1962. [LA04] J. Leung and J. Anderson, editors. Handbook of Scheduling- Algorithms, Models, and Performance Analysis. CRC/Chapman-Hall, 2004. [LA06] J. Lien and N. Amato. Approximate convex decomposition of polygons. Computational Geometry: Theory and Applications, 35:100—123, 2006. [Lam92] J.-L. Lambert. Sorting the sums (xi +yj) in o(n2) comparisons. Theoretical Computer Science. 103:137-147, 1992 [Lat9!] J.-C. Latombe. Robot Motion Planning. Kluwer Academic Publishers, Boston, 1991. [Lau98] J. Laumond. Robot Motion Planning and Control. Springer-Verlag. Lectures Notes in Control and Information Sciences 229, 1998. [LaV06] S. LaValle. Planning Algorithms. Cambridge University Press, 2006. [Law76] E. Lawler. Combinatorial Optimization: Networks and Matroids. Holt, Rinehart, and Winston, Fort Worth TX, 1976. [LD03] R. Laycock and A. Day. Automatically generating roof models from building footprints. In Proc. 11 th hit. Conf. Computer Graphics. I'isnalLotion and Computer I ision (WSCG), 2003. [Lec95] T. Lecroq. Experimental results on string matching algorithms. Software Practice and Experience, 25:727-765, 1995. [Lee82] D. T. Lee. Medial axis transformation of a planar shape. IEEE Trans. Pattern Analysis and Machine Intelligence, PAM 1-4:363-369, 1982.
698 Список литературы [Len87a] T. Lengauer. Efficient algorithms for finding minimum spanning forests of hierarchically defined graphs. .7. Algorithms, 8, 1987. [Len87b] H. W. Lenstra. Factoring integers with elliptic curves. Annals of Mathematics. 126:649-673, 1987. [Len89] T. Lengauer. Hierarchical planarity testing algorithms. J. ACM. 36(3):474- 509. July 1989. ~ [Len90] T. Lengauer. Combinatorial Algorithms for Integrated Circuit Layout. Wiley, Chichester, England, 1990. [Lev92] J. L Leva. A normal random number generator. ACM Trans. Math. Soft™., 18(4):454—455, December 1992. [Lew82] J. G. Lewis. The Gibbs-Poole-Stockmeyer and Gibbs-King algorithms for reordering sparse matrices. ACM Trans. Math Sofhv., 8(2): 190-194, June 1982. [LL96] A. LaMarca and R. Ladner. The influence of caches on the performance of heaps. ACM J Experimental Algorithmics, 1, 1996. [LL99] A. LaMarca and R. Ladner. The influence of caches on the performance of sorting. .1 Algorithms, 3\:66-IO4, 1999. [LLK83] J. К Lenstra. E. L. Lawler, and A. Rinnooy Kan. Theory of Sequencing and Scheduling. Wiley, New York, 1983. [LLKS85] E. Lawler, J. Lenstra, A. Rinnooy Kan, and D. Shmoys. The Traveling Salesman Problem. John Wiley, 1985. [LLS92] L. Lam, S.-W. Lee, and C. Suen. Thinning methodologies — a comprehensive survey. IEEE Trans. Pattern Analysis and Machine Intelligence, 14:869 885, 1992. [LM04] M. Lin and D. Manocha. Collision and proximity queries. In J. Goodman and J. O’Rourke, editors, Handbook of Discrete and Computational Geometry, p. 787-807. CRC Press, 2004. [LMM02] A. Lodi, S. Martello, and M. Monaci. Two-dimensional packing problems: A survey. European J. Operations Research, 141:241-252, 2002. [LMS06] L. Lloyd, A. Mehler, and S. Skiena. Identifying со-referential names across large corpora. In Combinatorial Pattern Matching (CPM 2006), p. 12-23. Lecture Notes in Computer Science, v. 4009, 2006. [LP86] [LP02] [LP07] L. Lovasz and M. Plummer. Matching Theory. North-Holland, Amsterdam, 1986. W. Langdon and R. Poli. Foundations of Genetic Programming. Springer, 2002. A. Lodi and A Punnen. TSP software In G. Gutin and A. Punnen editors. The Traveling Salesman Problem and Its Variations, p. 737-749. Springer, 2007. [LPW79] T. Lozano-Perez and M. Wesley. An algorithm for planning collision-free paths among polygonal obstacles. Comm. ACM, 22:560-570, 1979. [LR93] K. Lang and S. Rao. Finding near-optimal cuts: An empirical evaluation. In Proc. 4th Annual ACM-SIAM Symposium on Discrete Algorithms (SODA '93), p. 212 221, 1993. [LS87] V. Lumelski and A. Stepanov. Path planning strategies for a point mobile automaton moving amidst unknown obstacles of arbitrary shape. Algorithmica, 3:403-430, 1987 [LS95] Y.-L. Lin and S. Skiena. Algorithms for square roots of graphs. SIAM J. Discrete Mathematics, 8:99-118, 1995.
Список литературы 699 [LSCK02] Р. L’Ecuyer, R. Simard, Е Chen, and W. D. Kelton. An object-oriented random- number package with many long streams and substreams. Operations Research, 50 1073-1075. 2002. [LT79] R. Lipton and R. Tarjan. A separator theorem for planar graphs. SIAM Journal on Applied Mathematics, 36:346—358, 1979. [LT80] R. Lipton and R. Tarjan. Applications of a planar separator theorem. SIAM J. Computing, 9:615-626, 1980. [Luc91] E. Lucas. Recreations Mathematupies. Gauthier-Villares, Paris, 1891. [Luk80] E. M. Luks. Isomorphism of bounded valence can be tested in polynomial time. In Proc, oj the 21 st Annual Symposium on Foundations of Computing, p. 42—49. IEEE, 1980 [LV88] G. Landau and U. Vishkin. Fast string matching with A differences. J. Comput. System Sci., 37:63 -78, 1988 [LV97] M. Li and P. Vitanyi. An introduction to Kolmogorov complexity and its applications. Springer-Verlag, New York, second edition, 1997. [LW77] D. T. Lee and С. K. Wong. Worst-case analysis for region and partial region searches in multidimensional binary search trees and balanced quad trees. Ada Informatica, 9:23-29, 1977. [LW88] T. Lengauer and E. Wanke. Efficient solution of connectivity problems on hierarchically defined graphs. SIAM J. Computing, 17:1063-1080, 1988. [Mah76] S. Maheshwari. Traversal marker placement problems are NP-complete. Technical Report CU-CS-09276, Department of Computer Science, University of Colorado, Boulder. 1976. [Mai78] D, Maier. The complexity of some problems on subsequences and supersequences. J ACM, 25:322-336, 1978. [Mak02] R. Mak. Java Number Cruncher: The Java Programmer's Guide to Numerical Computing. Prentice Hall, 2002. [Man89] U. Manber. Introduction to Algorithms. Addison-Wesley, Reading MA, 1989. [Mar83] S. Martello. An enumerative algorithm for finding Hamiltonian circuits in a directed graph. ACM Trans. Math Softw., 9(1): 131—138, March 1983. [Mat87j D. W. Matula. Determining edge connectivity in O(nm). In 28lh Ann. Symp. Foundations of Computer Science, p. 249-251. IEEE, 1987. [McC76] E. McCreight. A space-economical suffix tree construction algorithm. J. ACM, 23:262-272, 1976. [McK81] B. McKay. Practical graph isomorphism. Congressus Numeranlium, 30:45-87, 1981. [McN83] J. M. McNamee. A sparse matrix package - part 11: Special cases. ACM Irons. Math. Softw., 9(3):344-345, September 1983. [MDS01] D. Musser, G. Derge, and A. Saini. STL Tutorial and Reference Guide: C++ Programming with the Standard Template Library. Addison-Wesley Professional, second edition, 2001. [Meg83] N. Megiddo. Linear time algorithm for linear programming in r and related problems. SIAM J. Computing, 12:759-776, 1983. [Men27] K. Menger. Zur allgemeinen Kurventheorie. Fund. Math., 10:96-115, 1927.
700 Список литературы [MeyOl] S. Meyers. Effective STL: 50 Specific Ways to Improve Your Use of the Standard Template Library. Addison-Wesley Professional. 2001 [MFOO] Z. Michalewicz and D. Fogel, //on to Solve it: Modern Heuristics. Springer, Berlin, 2000. [MG92] J. Misra and D. Gries. A constructive proof of Vizing’s theorem. Info. Processing Letters, 41:131-133, 1992. [MG06] J. Matousek and B. Gartner. Understanding and Using Linear Programming. Springer, 2006. [MGH81] J. J. More, В S. Garbow, and К. E. Hillstrom. Fortran subroutines for testing unconstrained optimization software. ACM Trans. Math Softw., 7( 1): 136-140. March 1981. [MH78J R. Merkle and M. Hellman. Hiding and signatures in trapdoor knapsacks. IEEE Trans. Information Theory, 24:525-530. 1978. [Mie58] [Mi 176] W. Miehle. Link-minimization in networks. Operations Research. 6:232 -243. 1958. G. Miller. Riemann's hypothesis and tests for primality. J. Computer and System Sciences, 13:300-317, 1976. [Mil97] V. Milcnkovic.Multiple translational containment, part 11: exact algorithms. Algorithmica, 19:183-218, 1997. [Min78] H. Mine. Permanents, vol. 6 of Encyclopedia of Mathematics and its Ipphcations. Addison-Wesley, Reading MA, 1978 [Mit99] J. Mitchell. Guillotine subdivisions approximate polygonal subdivisions: A simple polynomial-time approximation scheme for geometric TSP, k-mst, and related problems. SIAM J. Computing, 28:1298-1309. 1999. [MKT07] E. Mardis, S. Kim, and H. Tang, editors. Advances in Genome Sequencing Technology and Algorithms. Artech House Publishers, 2007. [MM93] U. Manber and G. Myers. Suffix arrays: A new method for on-line string searches. SIAM.J Computing, p. 935-948, 1993. [MM96] K. Mehlhorn and P. MutzeL On the embedding phase of the Hopcroft and Tarjan planarity testing algorithm. Algorithmica, 16:233-242, 1996. [MM 172] D. Matula, G Marble, and J. Isaacson Graph coloring algorithms. In R C. Read, editor. Graph Theory and Computing, p. 109-122. Academic Press, 1972, [MMZ+01] M. Moskewicz, C. Madigan, Y. Zhao, L. Zhang, and S. Malik. Chaff: Engineering an efficient SAT solver. In 39lh Design Automation Conference (DAC). 2001. [MN98] M. Matsumoto and T. Nishimura. Mersenne twister: A 623-dimensionally equidistributed uniform pseudorandom number generator. iCM Trans on Modding and Computer Simulation, 8:3-30, 1998. [MN99] K. Mehlhorn and S. Naher. LEDA: A platform for combinatorial and geometric computing. Cambridge University Press, 1999. [MN07] V. Makinen and G. Navarro. Compressed full text indexes. ICM Computing Surveys. 39, 2007. [MO63] L. E. Moses and R, V. Oakford. Tables of Random Permutations. Stanford University Press. Stanford, Calif.. 1963. [Moe90] S. Moen. Drawing dynamic trees. IEEE Software, 7-4:21-28, 1990.
Список литературы 701 [Моо59] Е. F. Moore. The shortest path in a maze. In Proc International Svmp Switching Theory, p. 285 292. Harvard University Press, 1959. [MOS06] K. Mehlhorn, R. Osbild, and M. Sagraloff. Reliable and efficient computational geometry via controlled perturbation. In Proc. Int. ( oil. on Automata. Languages, anti Programming (1C iLP), vol. 4051, p. 299 310. Springer Verlag, Lecture Notes in Computer Science, 2006. [Mou04] D. Mount. Geometric intersection. In J. Goodman and J. O’Rourke, editors, Handbook of Discrete and Computational Geometry, p. 857—876. CRC Press, 2004. [MOV96] A. Menezes, P. Oorschot. and S. Vanstone. Handbook of Applied Cryptography. CRC Press. Boca Raton. 1996. [MP80J W. Masek and M. Paterson. A faster algorithm for computing string edit distances. J. ( omputer and System Sciences, 20:18—31, 1980. [MPC+06] S. Mueller, D. Papamichial, J.R. Coleman, S. Skiena, and E. Wimmer. Reduction of the rate of poliovirus protein synthesis through large scale codon deoptimization causes virus attenuation of viral virulence by lowering specific infectivity. ,/. of Tirolog) , 80:9687-96, 2006. [MPT99] S. Martello, D. Pisinger, and P. Toth. Dynamic programming and strong bounds for the 0-1 knapsack problem. Management Science, 45:414-424, 1999. [MPT00] S. Martello, D. Pisinger, and P. Toth. New trends in exact algorithms for the 0-1 knapsack problem. European Journal of Operational Research, 123:325- 332, 2000. [MR95] R. Motwani and P. Raghavan. Randomized Algorithms. Cambridge University Press, New York, 1995. [MR01J W. Myrvold and F. Ruskey. Ranking and unranking permutations in linear time. Info. Processing Letters, 79:281—284, 2001. [MR06] W. Mulzer and G. Rote. Minimum weight triangulation is NP-hard. In Proc. 22nd ACM Symp. on Computational Geometry, p. 1 10, 2006. [MRRT53] N. Metropolis, A.W. Rosenbluth, M. N. Rosenbluth, and A. H. Teller. Equation of state calculations by fast computing machines. Journal of Chemical Physics, 21(6): 1087-1092. June 1953. [MS91] B. Moret and H. Shapiro. Algorithm from P to \'P: Design and Efficiency. Benjamin/Cummings, Redwood City, CA, 1991 [MS93] M. Murphy and S. Skiena. Ranger: A tool for nearest neighbor search in high dimensions. In Proc. Ninth ACM Symposium on Computational Geometry, p. 403-404, 1993. [MS95a] D. Margaritis and S. Skiena. Reconstructing strings from substrings in rounds. Proc. 36th IEEE Symp. Foundations of Computer Science (FOCS), 1995. [MS95b] J. S. B. Mitchell and S. Suri. Separation and approximation of polyhedral objects. Comput Geom Theory Appl, 5:95-\\ 4, 1995. [MS00] M. Mascagni and A. Srinivasan. Algorithm 806: Sprng: A scalable library for pseudorandom number generation. ACM Trans. Mathematical Software. 26:436 461, 2000. [MS05] D. Mehta and S. Sahni. Handbook of Data Structures and Applications. Chapman and Hall / CRC, Boca Raton, FL, 2005. [MT85] S. Martello and P. Toth. A program for the 0-1 multiple knapsack problem. ACM Trans. Math. Softw., 11(2): 135-140, June 1985.
702 Список литературы [МТ87] [МТ90а] [МТ90Ь] [MU05] [Ми194] [MutO5] [MV80] [MV99] [MY07] [Муе86] [Муе99а] [Мус99Ь] [NavOla] [NavOlb] [Nel96] [Neu63] [NI92] [NMB05] [NOI94] [Not02] [NROO] S. Martello and P. Toth. Algorithms for knapsack problems. In S. Martello, editor. Surveys in Combinatorial Optimization, vol 31 of. Innals of Discrete Mathematics, p. 213-258. North-Holland, 1987. S. Martello and P. Toth. Knapsack problems: algorithms and computer implementations. Wiley, New York, 1990. K. Mehlhorn and A. Tsakalidis. Data structures. In .1. van Leeuwen, editor, Handbook of Theoretical Computer Science: Algorithms and Complexity, vol. A. p. 301-341. MIT Press, 1990. M. Mitzenmacher and E. Upfal. robabdity and Computing: Randomized. Ugorithms and Probabilistic Analysis. Cambridge University Press, 2005. K. Mulmuley. Computational Geometry: an introduction through randomized algorithms. Prentice-Hall, New York. 1994. S. Mulhukrishnan. Data Streams: Algorithms and Applications. Now Publishers, 2005. S. Micali and V. Vazirani. An o( д/l I | |el) algorithm for finding maximum matchings in general graphs. In Proc 21 st. Symp. Foundations of Computing, p. 17-27, 1980. B. McCullough and H. Vinod. The numerical reliability of econometical software. J. Economic Literature, 37:633-665, 1999. K. Mehlhorn and C. Yap. Robust Geometric Computation, manuscript, http://cs.nyu.edu/yap/bookycgc/, 2007. E. Myers. An O(nd) difference algorithm and its variations. Algorithmica, 1:514-534, 1986. E. Myers. Whole-genome DNA sequencing. IEEE Computational Engineering and Science, 3:33—43, 1999. G. Myers. A fast bit-vector algorithm for approximate string matching based on dynamic progamming. J. ACM. 46:395-415, 1999. G. Navarro. A guided tour to approximate string matching. ACM Computing Surveys, 33:31-88, 2001. G Navarro. Nr-grep: a fast and flexible pattern matching tool. Software Practice and Experience. 31:1265-1312, 2001. M. Nelson. Fast searching with suffix trees. Dr. Dobbs Journal, August 1996. J. Von Neumann. Various techniques used in connection with random digits. In A. H. Traub, editor, John von Neumann. Collected Works, vol. 5. Macmillan, 1963. H. Nagamouchi and T. Ibaraki. Computing edge-connectivity in multigraphs and capacitated graphs. SI. \M J Disc. Math, 5:54-55, 1992. W. Nooy, A. Mrvar, and V. Batagelj. Exploratory Social Network Analysis with Pajek. Cambridge University Press, 2005. H. Nagamouchi, T. Ono, and T. Ibaraki. Implementing an efficient minimum capacity cut algorithm. Math. Prog., 67:297-324, 1994. C. Notredame. Recent progress in multiple sequence alignment: a survey. Pharmacogenomics. 3:131-144, 2002. G. Navarro and M. Raffmot. Fast and flexible string matching by combining bit-parallelism and suffix automata. ACM J. of Experimental Algorithmics, 5, 2000.
Список литературы 703 [NR04] [NR07] T. Nishizeki and S. Rahman Planar Graph Drawing. World Scientific, 2004. G. Navarro and M. Raffinot. Flexible Pattern Matching in Strings: Practical On-Line Search Algorithms for Texts and liiological Sequences. Cambridge University Press, 2007 [NS07] G. Narasimhan and M. Sinid. Geometric Spanner Seiworks. Cambridge Univ. Press, 2007. [Nuu95] E. Nuutila. Efficient transitive closure computation in large digraphs. hitp://www .cs.hut.fi/~enu/thesis.htinl, 1995. [NW78] A. Nijenhuis and H. Wilf. Combinatorial Algorithms for Computers and Calculators. Academic Press, Orlando FL, second edition, 1978. [NZ80] 1. Niven and H. Zuckerman. An Introduction to the Theory of Numbers. Wiley, New York, fourth edition, 1980. [NZ02] S. Naher and O. Ziotowski. Design and implementation of efficient data types for static graphs. In European Symposium on Algorithms (ESA), p. 748-759, 2002. [OBSCOO] A. Okabe, B. Boots, K. Sugihara, and S. Chiu. Spatial Tessellations: Concepts and Applications of Voronoi Diagrams. Wiley, 2000. [Ogn93] R. Ogniewicz. Discrete Voronoi Skeletons. Hartung-Gorre Veriag, Konstanz, Germany, 1993. [O’R85] J. O'Rourke. Finding minimal enclosing boxes. Int. J. Computer and Information Sciences, 14:183—199, 1985 [O'R87] J. O’Rourke. Art Gallery Theorems and Algorithms. Oxford University Press, Oxford, 1987. [O’ROl] .1. O'Rourke. Computational Geometry in C. Cambridge University Press, New York, second edition, 2001. [Ort88] J. Ortega. Introduction to Parallel and Vector Solution of Linear Systems. Plenum, New York, 1988. [OS04] J. O’Rourke and S. Suri. Polygons. In J. Goodman and J. O’Rourke, editors, Handbook of Discrete and Computational Geometry, p. 583—606. CRC Press, 2004. [OvL8i] M. Overmars and .1. van Leeuwen. Maintenance of configurations in the plane. J. Computer and System Sciences, 23:166-204. 1981. [OW85] J. O'Rourke and R. Washington. Curve similarity via signatures. In G. T. Toussaint, editor. Computational Geometry, p. 295-317. North-Holland, Amsterdam, Netherlands, 1985. [P57] G. Polya. How to Solve It. Princeton University Press, Princeton NJ, second edition, 1957. [РапОб] [Pap76a] [Pap76b] R. Panigrahy. Hashing, Searching, Sketching. PhD thesis, Stanford University, 2006. C. Papadimitriou. The complexity of edge traversing. J. ACM, 23:544-554, 1976. C. Papadimitriou. The NP-completeness of the bandwidth minimization problem. Computing, 16:263—270, 1976 [Par90] [Pas97] G. Parker. A better phonetic search. C Gazette. 5-4, June/July 1990. V. Paschos. A survey of approximately optimal solutions to some covering and packing problems. Computing Surveys, 171-209:171-209, 1997. [Pas03] V. Paschos. Polynomial approximation and graph-coloring. Computing, 70:41-86, 2003.
704 Список литературы [Pav82] Т. Pavlidis. Algorithms for Graphics and linage Processing. Computer Science Press. Rockville MD, 1982. [Рес04] M. Peczarski. New results in minimum-comparison sorting. Algorithmica. 40:133-145.2004. [Рес07] M. Peczarski. The Ford-Johnson algorithm still unbeaten for less than 47 elements. Info. Processing Letters, 101:126-128,2007. [PetO.3] J. Petit. Experiments on the minimum linear arrangement problem. ACM J. of Experimental Algorithmics, 8, 2003. [PFTV07] W. Press, B. Flannery, S. Teukolsky, and W. T. Vettcrling. Numerical Recipes: the art of scientific computing. Cambridge University Press, third edition, 2007. [РН80] M. Padberg and S. Hong. On the symmetric traveling salesman problem: a computational study. Math. Programming Studies. 12:78-107, 1980. [Р1А78] Y. Perl. A. Itai, and H. Avni. Interpolation search - a log log n search. Comm ACM, 21:550-554, 1978. [Pin02] M Pinedo. Scheduling Theory. Algorithms, and Systems. Prentice Hall, second edition, 2002. [PL94] P. A. Pevzner and R. J. Lipshutz. Towards DNA sequencing chips. In 19th Int. Conf Mathematical Foundations of Computer Science, vol. 841. p. 143 158, Lecture Notes in Computer Science, 1994. [PLM06] F. Panneton, P. L'Ecuyer, and M. Matsumoto. Improved long-period generators based on linear recurrences modulo 2. ACM hans. Mathematical Software, 32:1-16, 2006. [РМ88] S. Park and K. Miller. Random number generators: Good ones are hard to find. Communications ofthe ACM. 31 1192 1201, 1988 [PN04] Shortest Paths and Networks. J. Mitchell. In J. Goodman and J. O'Rourke, editors, I landbook of Discrete and Computational Geometry, p. 607-641. CRC Press. 2004. [Pom84] C. Pomerance. The quadratic sieve factoring algorithm. In T. Beth, N. Cot, and 1. Ingemarrson, editors, Advances in Cryptology, vol. 209, p. 169-182. Lecture Notes in Computer Science, Spnnger-Verlag, 1984. [РР06] M. Penner and V. Prasanna. Cache-friendly implementations of transitive closure. ACM J. of Experimental Algorithmics, 11,2006. [PR86] G. Pruesse and F. Ruskey. Generating linear extensions fast. SIAM J. Computing. 23.1994, 373-386. [PR02] S. Pettie and V. Ramachandran. An optimal minimum spanning tree algorithm. ACM, 49:16-34, 2002. [Рга75] V. Pratt. Every prime has a succinct certificate. SIAM J Computing, 4:214- 220, 1975. [Pri57] R. C. Prim. Shortest connection networks and some generalizations. Bell System Technical Journal, 36:1389-1401, 1957. [Prul8] H. Prufer. Neuer Beweis eines Satzes uber Permutationen. Arch Math. Pins.. 27:742-744, 1918. [PS85] F. Preparata and M. Shamos. Computational Geometry. Springer-Verlag, New York. 1985. [PS98] C. Papadimitriou and K. Steiglitz. Combinatorial Optimization Algorithms and Complexity. Dover Publications, 1998
Список литературы 705 [PS02] Н. Promel and A. Steger. The Steiner Tree Problem: a tour through graphs, algorithms, and complexity. Friedrick Vieweg and Son. 2002. [PS03J S. Pemmaraju and S. Skiena. Computational Discrete Mathematics: Combinatorics and Graph Theory with Mathematica. Cambridge University Press, New York, 2003. [PSL90] A. Pothen, H. Simon, and K. Liou. Partitioning sparse matrices with eigenvectors of graphs. SIAM J. Matrix Analysis, 11:430 452, 1990. [PSS07] F. Putze, P. Sanders, and J. Singler. Cache-, hash-, and space-efficient bloom filters. In Proc. 6th Workshop on Experimental Algorithms (WEA). LNCS 4525, p. 108-121, 2007. [PST07] S. Puglisi, W. Smyth, and A. Turpin. A taxonomy of suffix array construction algorithms. ACM Computing Surveys, 39, 2007. [PSW92] T. Pavlides, J. Swartz, and Y. Wang. Information encoding with twodimensional bar- codes. IEEE Computer, 25:18-28, 1992. [PT05] A. Pothen and S. Toledo. Cache-oblivious data structures. In D. Mehta and S. Sahni, editors. Handbook of Data Structures and Applications, p. 59:1-59:29. Chapman and Hall / CRC, 2005. [Pug86] G. Allen Pugh. Partitioning for selective assembly. Computers and Industrial Engineering, 1 1:175—179, 1986. [PV96] M. Pocchiola and G. Vegter. Topologically sweeping visibility complexes via pseudo- triangulations. Discrete and Computational Geometry, 16:419-543, 1996. [Rab80] M. Rabin. Probabilistic algorithm fortesting primality. J. Number Theory, 12:128 138, 1980. [Rab95] F. M. Rabinowitz, A stochastic algorithm for global optimization with constraints. ACM Trans. Math. Softw., 21 (2): 194-213, June 1995. [Ram()5] R. Raman Data structures for sets. In D. Mehta and S. Sahni. editors. Handbook of Data Structures and Applications, p. 33:1-33:22. Chapman and Hall / CRC, 2005. [Raw92] G. Rawlins. Compared to What? Computer Science Press, New York, 1992. [RBT04] H. Romero, C. Brizuela, and A. Tchernykh. An experimental comparison of approximation algorithms for the shortest common superstring problem. In Proc. Fifth Mexican Int Conf in Computer Science (ENC’04), p. 27-34, 2004 [RC55] Rand-Corporation. 4 million random digits with 100.000 normal deviates. The Free Press, Glencoe, IL, 1955. [RD01] E. Reingold and N. Dershowitz. Calendrical Calculations: The Millennium Edition Cambridge University Press, New' York, 2001. [RDC93] E. Reingold, N. Dershowitz, and S. Clamen. Calendrical calculations II: Three historical calendars. Software - Practice and Experience, 22:383—404, 1993. [Rei72] E. Reingold. On the optimality of some set algorithms. J. ACM, 19:649-659, 1972 [Rei91] G. Reinelt. TSPLIB - atraveling salesman problem library. ORSA ./. Computing, 3:376-384, 1991. [Rei94] G. Reinelt. The traveling salesman problem: Computational solutions for TSP applications. In Lecture Notes in Computer Science 840, p. 172- 186. Spnnger- Verlag, Berlin, 1994. [RF06] S. Roger and T. Finley. JFLAP: An Interactive Formal Languages and Automata Package. Jones and Bartlett. 2006.
706 Список литературы [RFS98] [RHG07J [RHS89] [Riv92] [RR99] [RS96] [RSA78J [RSL77] [RSN+O I] [RSS02] [RSST96] [RT81] [Rus03] [Ryt85] [RZ05] [SA95] [SahO5] [SalO6] [Sam05] M Resende, T. Feo, and S. Smith. Algorithm 787: Fortran subroutines for approximate solution of maximum independent set problems using GRASP. i('M Transactions on Mathematical Software. 24:386-394, 1998. S. Richter, M. Helert. and C. Gretton. A stochastic local search approach to vertex cover. In Proc. 30lh German Conf, on Irtijicial Intelligence (А/ 201)7). 2007. A. Robison, B. Hafner, and S. Skiena. Eight pieces cannot cover a chessboard. Computer Journal, 32:567-570, 1989. R. Rivest. The MD5 message digest algorithm. RFC 1321, 1992. C.C Ribeiro and M.G.C. Resende. Algorithm 797: Fortran subroutines for approximate solution of graph planarization problems using GRASP. ACM Transactions on Mathematical Software, 25:341-352, 1999. H. Rau and S. Skiena. Dialing for documents: an experiment in information theory. Journal of Visual Languages and Computing, p. 79-95, 1996. R Rivest, A. Shamir, and L. Adleman. A method for obtaining digital signatures and public-key cryptosystems. Communications of the ACM, 21:120-126, 1978. D. Rosenkrantz. R. Stearns, and P. M. Lewis. An analysis of several heuristics for the traveling salesman problem. SIAM J. Computing, 6:563-58\, 1977. A. Rukihin, J. Soto, J. Nechvatal, M. Smid. E Barker, S. Leigh, M Levenson, M Vangel, D. Banks, A. Heckert, J. Dray, and S. Vo. A statistical test suite for the validation of random number generators and pseudo random number generators for cryptographic applications. Technical Report Special Publication 800-22, NIST, 2001. E. Rafalin. D. Souvame, and 1. Streinu. Topological sweep in degenerate cases. In Proc 4th IVorkshop on Algorithm Engineering and Experiments (ALENEX), p. 273-295, 2002. N. Robertson, D. Sanders, P. Seymour, and R. Thomas. Efficiently fourcoloring planar graphs. In Proc 28th ACM Symp. Theory of Computing, p 571—575,1996. E Reingold and J Tilford. Tidier drawings of trees. IEEE Trans Software Engineering, 7:223-228, 1981. F. Ruskey. Combinatorial Generation. Manuscript in preparation. Draft available at http://www.lstworks.com/rcf/RuskcyCoiiibGcn.pdf, 2003. W. Rytter. Fast recognition of pushdown automata and context-free languages. Information and Control, 67:12-22, 1985. G. Robins and A. Zelikovsky. Improved Steiner tree approximation in graphs. Tighter Bounds for Graph Steiner Tree Approximation, p. 122-134, 2005. M. Sharirand P. Agarwal. Davenport-Schinze! sequences and their geometric applications. Cambridge University Press, New York, 1995. S. Sahni. Double-ended priority queues. In D. Mehta and S. Sahni, editors. Handbook of Data Structures and Applications, p. 8:1-8:23. Chapman and Hall / CRC, 2005. D. Salomon. Data Compression The Complete Reference. Springer-Verlag, fourth edition, 2006. H. Samet. Multidimensional spatial data structures. In D. Mehta and S. Sahni, editors. Handbook of Data Structures and Applications, p. 16:1- 16:29. Chapman and Hall / CRC, 2005.
Список литературы 707 [Sam06] Н. Samet. Foundations of Multidimensional and Metric Data Structures. Morgan Kaufmann, 2006. [SanOO] P. Sanders. Fast priority queues for cached memory.. ICM Journal of Experimental llgorithmics, 5. 2000. [Sav97] C. Savage. A survey of combinatorial gray codes. SIAM Review, 39:605 -629, 1997. [Sax80] J B. Saxe. Dynamic programming algorithms for recognizing smallbandwidth graphs in polynomial time. SIAM J. Algebraic and Discrete Methods, 1:363-369, 1980. [Say05] K. Sayood. Introduction to Data Compression. Morgan Kaufmann, third edition, 2005. [SB01] A. Samorodnitsky and A. Barvinok. The distance approach to approximate combinatorial counting. Geometric and Functional Analysis, 1 1:871-899, 2001. [Sch96] B. Schneier. Applied Cryptography: Protocols. Algorithms, and Source Code in C. Wiley, New York, second edition, 1996. [Sch98] A. Schrijver. Bipartite edge-coloring in (9(r5m) time. SIAM J. Computing, 28:841-846, 1998. [SD75] M. Syslo and J. Dzikiewicz. Computational experiences with some transitive closure algorithms. Computing, 15:33-39, 1975. [SD76] D. C. Schmidt and L. E. Druffel. A fast backtracking algorithm to test directed graphs for isomorphism using distance matrices. .7. ACM, 23:433- 445, 1976. [SDK83] M. Syslo, N. Deo. and J. Kowalik. Discrete Optimization Algorithms with Pascal Programs. Prentice Hall, Englewood Cliffs NJ, 1083. [Sed77] R. Sedgewick. Permutation generation methods. ComputingSurvevs, 9:137-164, 1977. [Sed78] R. Sedgewick. Implementing quicksort programs. Communications of the ACM, 21:847-857, 1978. [Sed98] R. Sedgewick. Algorithms in C+ Parts 1-4 Fundamentals. Data Structures. Sorting. Searching, and Graph Algorithms. Addison-Wesley, Reading MA, third edition, 1998. [Sei04] R. Seidel. Convex hull computations. In J. Goodman and J. O’Rourke, editors, Handbool, of Discrete and C omputational Geometry, p. 495-512. CRC Press, 2004. [SF92] T Schlick and A. Fogelson. TNPACK - a truncated Newton minimization package for large-scale problems: I. algorithm and usage. ACM Trans. Math. Softw.. 18( I ):46-70, March 1992. [SFG82] M. Shore. L. Foulds, and P. Gibbons. An algorithm for the Steiner problem in graphs Networks. 12:323-333, 1982. [SH75] M. Shamos and D. Hoey. Closest point problems. In Proc. Sixteenth IEEESymp. Foundations of Computer Science, p. 151-162. 1975. [SH99] W. Shih and W. Hsu. A new planarity test. Theoretical Computer Science, 223(1-2): 179-191, 1999. [Sha87] M. Shanr. Efficient algorithms for planning purely translational colhsionfree motion in two and three dimensions. In Proc. IEEE Internal. Conf. Robot Autom., p. 1326-1331, 1987. [Sha04] M. Sharir. Algorithmic motion planning. In J. Goodman and J. O'Rourke, editors, Handbook of Discrete and C omputational Geometry, p. 1037- 1064. CRC Press, 2004.
708 Список литературы [She97] J. R. Shewchuk. Robust adaptive floating-point geometric predicates. Disc. ( omputational Geometry, 18:305—363, 1997. [Sho05] V. Shoup.. I ( omputational Introduction to Number Theory and Algebra. Cambridge University Press, 2005. [Sip05] M. Sipser. Introduction to the Theory of Computation. Course Technology, second edition, 2005. [SK86] T. Saaty and P. Kainen. The Four-Color Problem. Dover, New York, 1986. [SK99] D. Sankoff and J. Kruskal Time Warps, Siring Edits, and Macromolecules: the theory and practice of sequence comparison. CSLI Publications, Stanford University. 1999* [SK00] R. Skeel and J. Keiper. Elementary Numerical computing with Mathematica. Stipes Pub Lie., 2000. [Ski88] S. Skiena. Encroaching lists as a measure of presortedness. HIT, 28:775-784, 1988. [Ski90] S. Skiena. Implementing Discrete Mathematics. Addison-Wesley, Redwood City, CA. 1990. [Ski99] S. Skiena. Who is interested in algorithms and why?: lessons from the stony brook algorithms repository. ACM SlGAC T News. p. 65-74. September 1999. [SL07] M. Singh and L. Lau. Approximating minimum bounded degree spanning tree to within one of optimal. In Proc 39th Symp. Theory Computing (STOC). p. 661-670, 2007. [SLL02] .1 Siek, L. Lee. and A. Lumsdaine. The Boost Graph Library: user guide and reference manual. Addison Wesley, Boston, 2002. [SM73] L. Stockmeyer and A. Meyer. Word problems requiring exponential time. In Proc. Fifth ACM Symp. Theory of Computing, p. 1—9, 1973. [Smi91 ] D. M. Smith. A Fortran package for floating-point multiple-precision arithmetic. 4CM Trans. Math. Softw., 17(2):273-283, June 1991. [Sno04] J. Snoeyink. Point location. In J. Goodman and J. O’Rourke, editors, Handbook of Discrete and Computational Geometry, p. 767-785. CRC Press, 2004. [SR83] K. Supowit and E. Reingold. The complexity of drawing trees nicely. Ada Informatica, 18:377-392, 1983. [SR03] S. Skiena and M. Revilla. Programming Challenges: The Programming Contest Training Manual. Springer-Verlag, 2003. [SS71] A. Schonhage and V. Strassen. Schnelle Multiplikation grosser Zahlen. Computing, 7:281-292, 1971. [SS02] R. Sedgewick and M. Schidlowsky. Algorithms in Java, Parts 1-4. Fundamentals. Data Structures, Sorting. Searching, and Graph Algorithms. Addison-Wesley Professional, third edition. 2002. [SS07] K. Schurmann and J. Stoye. An incomplex algorithm for fast suffix array construction. Software: Practice and Experience, 37:309-329, 2007. [ST04] D. Spielman and S. Teng. Smoothed analysis: Why the simplex algorithm usually takes polynomial time. J ACM, 51:385—463, 2004. [Sta06] W. Stallings. Cryptography and Network Security: Principles and Practice. Prentice Hall, fourth edition, 2006.
Список литературы 709 |Str69| V. Strassen. Gaussian elimination is not optimal. Numerische Mathematik, 14:354-356, 1969. [SV87] J. Stasko and J. Vitter. Pairing heaps: Experiments and analysis. Communications of the AC W, 30(3):234—249, 1987 [SV88] B. Schieber and U. Vishkin. On finding lowest common ancestors: simplification and parallelization. SL IM J. Comput., 17(6): 1253-1262, December 1988. [SW86a] D. Stanton and D. White. Constructive Combinatorics. Springer-Verlag, New York, 1986 [SW86bJ Q. Stout and B.Warren. Tree rebalancing in optimal time and space. Comm ACM, 29:902-908, 1986. [SWA03] S. Schlieimer, D. Wilkerson, and A. Aiken. Winnowing: Local algorithms for document fingerprinting. In Proc. ACM SIG MOD int. Conf, on Management of data. p. 76-85, 2003. [Swe99] Z. Sweedyk. A 2.5-approximation algorithm for shortest superstring. SIAM J Computing, 29:954-986, 1999. [SWM95] J. Shallit, H. Williams, and F Moraine. Discovery of a lost factoring machine The Mathematical Intelligencer, 17-3:41—47, Summer 1995. [SzpO3] G. Szpiro. Kepler's Conjecture. How Some of the Greatest Minds in History Helped Solve One of the Oldest Math Problems in the World. Wiley, 2003. [TamO8] R. Tamassia. Handbook of Graph Drawing and Visualization. Chapman-Hall / CRC, 2008. [Tar95] [Tar72] G. Tarry. Le probleme de labyrmthes. Nouvelles Ann. de Math , 14:187, 1895. R. Tarjan. Depth-first search and linear graph algorithms. SIAM J. Computing, 1:146-160, 1972. [Tar75] R. Tarjan. Efficiency of a good but not linear set union algorithm.1C V/, 22:215-225, 1975. [Tar79] R. Tarjan. A class of algorithms which require non-linear time to maintain disjoint sets. J. ( omputer and System Sciences, 18:110-127, 1979. [Tar83] R. Tarjan. Data Structures and Network Algorithms. Society for Industrial and Applied Mathematics, Philadelphia, 1983. [TH03] R. Tam and W. Heidrich. Shape simplification based on the medial axis transform. In Proc 14th IEEE Visualization (VIS-03), p. 481—488, 2003. [THG94] J. Thompson, D. Higgins, and T. Gibson. CLUSTAL W: improving the sensitivity of progressive multiple sequence alignment through sequence weighting, position- specific gap penalties and weight matrix choice. Nucleic Acids Research, 22:4673-80, 1994. [ThiO3] H. Thimbleby. The directed Chinese postman problem. Software Practice and Experience., 33:1081-1096, 2003. [Tho68] K. Thompson. Regular expression search algorithm. Communications of the ACM, 11:419-422, 1968. [Tin90] [TNX08] G. Tinhofer. Generating graphs uniformly at random. Computing, 7:235- 255, 1990. K. Thulasiraman, T. Nishizeki, and G. Xue. 1 he Handbook of Graph. Ilgorithms and Applications, vol. 1: Theory and Optimization. Chapman- Hall/CRC, 2008.
710 Список литературы [Тго62] [Tur88] [TVO I] [TW88] [Ukk92J [Val79] [Val02] [Van98| [Vaz04] [VB96] [vEBKZ77] [VitO I] [Viz64] [vL90a] [vL90b] [VL05] [Vos92] [Wal99] [War62] [Wat03] [Wat04] [WBCS77] [WC04a] H F. Trotter. Perm (algorithm 115). ( omm. AC M, 5:434—435, 1962. J. Turner. Almost all A-colorable graphs are easy to color. J. Algorithms. 9-63-82, 1988. R. Tamassia and L. Vismara. A case study in algorithm engineering for geometric computing, hit J. Computational Geometry and, Ipplications, 11 (1): 15-70, 2001. R. Tarjan and C Van Wyk. An O(n 1g 1g n) algorithm for triangulating a simple polygon. SIAM J. Computing, 17:143-178, 1988. E. Ukkonen. Constructing suffix trees on-line in linear time. In Intern. Federation of Information Processing (1FIP '92), p. 484^192, 1992. L. Valiant. The complexity of computing the permanent. Theoretical Computer Science, 8:189-201, 1979. G. Valiente. ilgorithms on Trees and Graphs. Springer, 2002. B. Vandegriend. Finding hamiltonian cycles: Algorithms, graphs and performance. M.S. Thesis, Dept, of Computer Science, Univ, of Alberta, 1998. V. Vazirani. Approximation Algorithms. Springer, 2004. L. Vandenberghe and S. Boyd. Semidefinite programming. SIAM Rei iew, 38:49-95, 1996. P. van Emde Boas. R. Kaas. and E. Zulstra. Design and implementation of an efficient priority queue. Math. Systems Theory, 10:99-127, 1977. J. Vitter. External memory algorithms and data structures: Dealing with massive data ACM Computing Surveys, 33:209 271,2001. V. Vizing. On an estimate of the chromatic class of a p-graph (in Russian). Diskret. Analiz, 3:23-30. 1964. .1. van Leeuwen. Graph algorithms. In .1. van Leeuwen, editor. Handbook of Theoretical Computer Science: Algorithms and Complexity, vol. A, p. 525-631. MIT Press. 1990. J. van Leeuwen, editor. Handbook of Theoretical Computer Science: Algorithms and Complexity, vol. A. MIT Press, 1990. F. Vigcr and M. Latapy. Efficient and simple generation of random simple connected graphs with prescribed degree sequence. In Proc. 11 th Conf, on Computing and Combinatorics (COCOON), p. 440 449, 2005. S. Voss. Steiner’s problem in graphs: heuristic methods. Discrete Applied Mathematics, 40 45 - 72, 1992. J. Walker. A Primer on Wavelets and Their Scientific Applications. CRC Press, 1999. S. WarshalL A theorem on boolean matrices. J. ACM, 9:11—12, 1962. B. Watson. A new algorithm for the construction of minimal acyclic DFAs. Science of Computer Programming, 48:81-97, 2003. D. Watts. Six Degrees: The Science of a Connected Ige. W.W. Norton, 2004. J.Weglarz, J. Blazewicz, W. Cellaiy, and R. Slowinski An automatic revised simplex method for constrained resource network scheduling. ACM Trans. Math. Softyv., 3(3):295-300, September 1977. B. Watson and L. Cleophas. Spare parts: a C++ toolkit for string pattern recognition. Soflyi are Practice and Experience., 34:697-710, 2004
Список литературы 711 [WC04b] В. Wu and К Chao. Spanning Trees and Optimization Problems. Chapman-Hall / CRC, 2004. [Wei73] P. Weiner. Linear pattern-matching algorithms. In Proc. 14th IEEE Symp. on S» itching and Automata Theory, p. 1-i 1, 1973. [Wei06] M. Weiss. Data Structures and Algorithm Analysis in.Java. Addison Wesley, second edition, 2006. [Wel84] T. Welch. A technique for high-performance data compression. IEEE Computer. 17-6:8-19, 1984. [Wes83] D H. West. Approximate solution of the quadratic assignment problem.. ICM Trans, '.lath. Softw., 9(4) 461—466, December 1983. [WesOO] D. West. Introduction to Graph Theory. Prentice-Hall, Englewood Cliffs NJ, second edition, 2000. [WF74] R. A. Wagner and M. J. Fischer. The string-to-string correction problem. J. It I/, 21:168-173, 1974. [Whi32] H. Whitney. Congruent graphs and the connectivity of graphs. American J. Mathematics, 54:150-168, 1932. [Wig83] A. Wigerson. Improving the performance guarantee for approximate graph coloring. J. AC It, 30:729-735, 1983. [Wil64] J. W. J. Williams. Algorithm 232 (heapsort). Communications of the ACM, 7:347-348, 1964. [Wil84| H. Wilf. Backtrack: An 0(1) expected time algorithm for graph coloring. Info. Proc. Letters, 18:119-121, 1984. [Wil85] D. E. Willard. New data structures for orthogonal range queries. SIAM J. Computing, 14:232-253, 1985. [Wil89] H. Wilf. Combinatorial Algorithms- an update. SIAM, Philadelphia PA, 1989, [Win68] S. Winograd. A new algorithm for inner product. IEEE Trans Computers, C-17:693-694, 1968. [Win80] S. Winograd. Arithmetic Complexity of Computations. SIAM. Philadelphia. 1980. [WM92a] S. Wu and U. Manber. Agrep — a fast approximate pattern-matching tool. In Llsenix Winter 1992 Technical Conference, p. 153—162, 1992. [WM92b] S. Wu and U. Manber. Fast text searching allowing errors. Comm ACM. 35:83-91, 1992. [Woe03] G. Woeginger. Exact algorithms for NP-hard problems: A survey. In Combinatorial Optimization - Eureka! ion shrink!, vol. 2570 Springer-Verlae LNCS, p. 185-207, 2003. [Wol79] T. Wolfe. The Right Stuff. Bantam Books, Toronto, 1979. [WW95] F. Wagner and A. Wolff. Map labeling heuristics: provably good and practically useful. In Proc. 11th ACM Symp Computational Geometry, p. 109-118, 1995. [WWZ00] D. Warme, P. Winter, and M. Zachariasen. Exact algorithms for plane Steiner tree problems: A computational study. In D. Du, J. Smith, and J. Rubinstein, editors. Advances in Steiner Trees, p. 81-116. Kluwer, 2000. [WY05] X. Wang and H. Yu. How to break MD5 and other hash functions. In El 'ROCRi PT. LS'CS vol. 3494, p. 19-35, 2005.
712 Список литературы [УапОЗ] S. Yan. Primality Testing and Integer Factorization in Public-Key Cryptography. Springer, 2003. [Yao81] [Yap04] A C. Yao. A lower bound to finding convex hulls. J. ACM, 28:780-787. 1981. C. Yap. Robust geometric computation. In J. Goodman and .1. O’Rourke, editors. Handbook of Discrete and Computational Geometry, p. 607 -641. CRC Press, 2004. [YLCZO5] R Yeung, S-Y. Li, N. Cai, and Z. Zhang. Network Coding Theory. http://www.uowpublisliers.coni/. Now Publishers. 2005. [You67] D. Younger. Recognition and parsing of context-free languages in time O(n ). Information and Control, 10:189-208, 1967. [YS96] F. Younas and S. Skiena. Randomized algorithms for identifying minimal lottery ticket sets. Journal of Undergraduate Research, 2-2:88-97, 1996. [YZ99] E. Yang and Z. Zhang The shortest common superstring problem: Average case analysis for both exact and approximate matching. IEEE Trans Information Theory, 45:1867-1886, 1999. [Zar02] C. Zaroliagis. Implementations and experimental studies of dynamic graph algorithms. In Experimental algorilhmics: from algorithm design to robust and efficient software, p. 229-278. Springer-Verlag LNCS, 2002. [ZL78] J, Ziv and A. Lempel A universal algorithm for sequential data compression. IEEE Trans. Information Theory, IT-23:337—343, 1978. [ZSO4] Z. Zaritsky and M. Sipper. The preservation of favored building blocks in the struggle for fitness: The puzzle algorithm. IEEE Trans. Evolutionary Computation, 8:443^155,2004. [ZwiOl] U. Zwick. Exact and approximate distances in graphs - a survey, in Proc. 9lh Euro. Symp. Algorithms (ESA), p. 33—48, 2001.
Предметный указатель А G agrep 645 ARPEC 447 Genetic Algorithm Utility Library 430 GEOMPACK615 GeoSteiner 571 В GLPK 434 GMP, библиотека 446 BioJava 401 Boost, библиотека 405, 668 BSP-дерево 412 В-дерево 393 GnuPG 655 GOBLIN, библиотека 523, 668 Grail+ 658 Graphiib 513 gzip 650 С 1 Calendrical 485 С ALGO 669 CGAL, библиотека 667 Chaco 557 Chaff 492 Cliquer 541 CLP, библиотека 434 Сосопе 612 Combinatorica 178. 670 Core, библиотека 580 COVER 546 CPAN 669 Crypto++, библиотека 655 ILOG CP 489 J JAMA, библиотека 426 JC, библиотека 394 JDSL, библиотека 394, 405 JFKAP658 JGraphEd 536 JGraphT, библиотека 405, 497 JOBSHOP 488 JOSTLE 557 JScience, библиотека 426 D JUNG, библиотека 405. 497 DSATUR 560 К F KDTREE2 413 kd-дерево 410 FFTPACK, библиотека 454 FFTW, библиотека 454 FIFO 90 FIRE Engine 658 FLUTE 571 FSM, библиотека 658 Kernel-Machine, библиотека 621 L LEDA, библиотека 666 LEKIN 488 L1BSVM, библиотека 621
714 Предметный указатель LiDl А, библиотека 442 LIFO 90 UNPACK 426 м Mathematica 178 METIS 557 MINCUTLIB 523 MiniSAT 492 MIRACL, библиотека 442 MPFUN90 447 N NEOS 430, 434 Netlib, библиотека 668 Nettle, библиотека 655 NTL, библиотека 442 P PARI 441 PAUP571 PHYLIP571 PicoSAT 492 PIGALE 536 Powercrust 612,618 Prolog, язык 325 Q Qhull 584, 588. 591 QSlim 618 R RAM 49 REDUCE 409 Roncorde 550 Rsat 492 R-дерево 412 s Scotch 557 Soundex, алгоритм 645 SourceForge 669 SPARE Parts 640 Spatial Index Demos 413 SPRNG, библиотека 439 Stanford GraphBase 669 STL, библиотека 394 strmat 402, 640 SYMPHONY 635,638 T TerraLib, библиотека 413 TRE, библиотека 645 TSPLIB, библиотека 550 V VFLib, библиотека 567 VRONI 612 A Алгоритм 21 0 Soundex 645 0 Ависа-Фукуды 584 0 б нижайшего соседа 23 0 ближайших пар 25 0 Борувки 502 0 генетический 286 0 Грэхема 583 0 Дейкстры 228, 506 0 Дугласа-Пекера 617 0 заворачивания подарка 582 0 Крускала 218, 501 0 Лемпеля-Зива 649 0 Прима 215,501 0 Укконена401 0 Флойда-Варшалла 232, 508 0 Хиршберга 644 Арифметическая прогрессия 36
Предметный указатель 715 Арифметические операции 445 Асимптотические обозначения 52 Б Библиотека О Boost 405. 668 0 CGAL 667 0 CLP 434 0 Core 580 0 Crypto++ 655 0 FFTPACK454 0 FFTW454 0 FSM658 0 GMP446 О GOBLIN 523, 6о8 О JAMA 426 О JC394 О JDSL 394. 405 О JGraphT 405. 497 О JScience 426 О JUNG 405, 497 О Kernel-Machine 621 О LEDA 666 О LIBSVM621 О LiDIA 442 О MIRACI442 О Netlib 668 О Nettle 655 О NTL442 О SPRNG439 О STL394 О TerraLib4l3 О TRE645 О TSPL1B550 О VFLib 567 Биномиальные коэффициенты 298 Ближайшая точка 592 Борувки, алгоритм 502 В Вершинная раскраска 557 Вершинное покрытие 347, 356, 368, 544 Возведение в степень 66 Вороного, диаграмма 589 Восстановление пути 304 Восхождение по выпуклой поверхности 272 Выпуклая оболочка 125. 344, 581 Г Гамильтонов цикл 345. 551 Гармоническое число 66 Генератор случайных чисел 436 Геометрическая прогрессия 36 Гиперграф 404 Грамматика контекстно-свободная 318 Граф 38, 168, 479 0 ациклический 171 0 бесконтурный подграф 371 0 вершинная раскраска 557 0 вершинное покрытие 209, 347, 356, 368, 544 0 взвешенный 169, 213 0 гамильтонов цикл 345, 551 0 гиперграф 404 0 двудольный 191,239 0 доминирующее множество 546 0 изоморфизм 564 0 квадрат 209 0 клика 350 0 контур 249 0 кратное ребро 170 0 кратчайший путь 505 О матрица инцидентности 404 О минимальное остовное дерево 500 0 мост 200 О независимое множество 209, 279, 348, 542 О обход 184 0 ориентированный 169 0 остовное дерево 214 0 паросочетание 239 0 перечисление путей 256 0 петля 170 0 планарный 404, 534 О плотный 170 0 полный 171 О помеченный 171 О путь 188 О разбиение 554 О разреженный 170
716 Предметный указатель Граф (пред.) О разрывающее множество 572 О рас крас ка верш ин 191 О реберная раскраска 561 О реберное покрытие 249, 546 О рисование 528 О связный I89, 197,494,521 О сильно связный 204 О список смежности 213 О степень вершины 173 О топологическая сортировка 171.202 О транзитивное замыкание 233 О турнир 211 О укладка 171 О цикл 196 О шарнир 196 О эйлеров подграф 376 О эйлеров цикл 376 Грея, код 473 Грэхема, алгоритм 583 д Двоичное дерево 64 Двоичный поиск 64, 154 Дейкстры, алгоритм 228, 506 Дерево 38, 64 0 BSP-дерево 412 0 В-дерево 393 0 kd-дерево 410 0 R-дерево 412 0 вставка элемента 98 О двоичное 64, 96 0 квадродерево 411 0 косое 393 0 минимальное остовное 500 0 наибольший элемент 98 0 наименьший элемент 97 0 обход 98 0 октадерево 411 0 остовное 214, 223 0 поиска 96 0 помеченное 481 0 рисование 532 0 сбалансированное 101 0 суффиксное 398 0 удаление элемента 99 0 Штейнера 223. 568 Диаграмма 0 Вороного 589 0 Ганта 489 Динамическое программирование 293 Дискретное преобразование Фурье 451 Доказательство сложности 358 Доминирование функции 56. 74 Доминирующее множество 546 Дугласа-Пекера. алгоритм 617 3 Задача 0 вершинной раскраски 557 0 выполнимости булевых формул 351, 489 0 выявления изоморфизма графов 564 0 выявления сходства фигур 619 0 календарного планирования 26, 348, 486 0 китайского почтальона 517 0 коммивояжера 26, 322, 547 0 линейного разбиения 315 0 максимального разреза 278 0 нечеткого сравнения строк 301, 641 0 о вершинном покрытии 347, 356, 368, 544 0 о доминирующем множестве 546 0 о клике 350 0 о назначениях 5 14 0 о независимом множестве 348, 542 0 о покрытии множества 63 1 0 о потоках в сетях 239, 524 0 о разрывающем множестве 572 0 о реберном покрытии 546 0 о рюкзаке 448 0 о сумме подмножества 449 0 оптимизации 428 0 планирования перемещений 621 0 поиска ближайшего соседа 592 0 поиска ближайшей пары 341 0 поиска в области 596 0 поиска гамильтонова цикла 345, 551 0 поиска компонент связности 494 0 поиска кратчайшего пути 505 0 поиска медианы 465 0 поиска эйлерова цикла 517 0 построения выпуклой оболочки 581
Предметный указатель 717 О построения дерева Штейнера 568 О преобразования к срединной оси 6I0 О разбиения графа 554 О разбиения множества целых чисел 449 О разложения по контейнерам 607 О реберной раскраски 561 О рисования графа 528 О сравнения строк 638 И Изоморфизм графов 564 Имитация отжига 274 К Календарное планирование 26. 348.486 Календарь 484 Квадратный корень 156 Квадродерево 411 Код 0 Г рея 473 0 Прюфера 48! О Хаффмана 649 Коллизия 108 Компонента связности 189, 494 Конечный автомат 656 Контейнер 90 Контекстно-свободная грамматика 3I8 Контрольная сумма 653 Конфигурации прямых 625 Конфликт имен файлов 245 Косое дерево 393 Кратчайший путь 505 Криптография 651 Критический путь 487 Крускала, алгоритм 218, 501 Кэширование 295 Л Лемпеля-Зива, алгоритм 649 Линейное программирование 43! Логарифм 64 0 двоичный 68 0 десятичный 68 0 натуральный 68 Локальный поиск 271 м Максимальная возрастающая подпоследовательность 342 Массив 85, 144 Матрица смежности 174. 402 Медиана 465 Местоположение точки 579, 599 Метод 0 "разделяй и властвуй" 156 0 внутренней точки 432 0 восхождения по выпуклой поверхности 272 0 имитации отжига 274 0 Монте-Карло 268 О опорных векторов 620 0 полос 60! 0 произвольной выборки 268 Минимальное остовное дерево 500 Минковского, сумма 628 Многоугольник 38, 613 Множество 406, 631 0 пересечение I26 0 разбиение 408 Морфинг 312 н Наибольший общий делитель 343 Наименьшее общее кратное 343 Независимое множество 542 Независимое множество вершин графа 348 Нормальная форма Хомского 319 О Обход графа 184 0 в глубину 192 0 в ширину 185 Октадерево 411 Оптимизация 428 Остовное дерево 214. 223 Открытая адресация 109 Отношение доминирования 56, 74 Отсечение вариантов 258 Очередь с приоритетами 102, 395
718 Предметный указатель п Параллельные алгоритмы 287 Паросочетание 239, 514 Перебор с возвратом 251 Пересечение О множеств 126 О отрезков 603 0 прямой и отрезка 579 0 прямых 577 Перестановки 37, 255. 468 Пирамида 129 0 вставка элемента 132 С создание 132 < удаление элемента 133 Пирамидальное число 70 Планарность 534 Планирование перемещений 621 Площадь треугольника 579 Подмножество 38, 472 Подстрока 400 Поиск 0 ближайшей пары 341 0 двоичный 96 0 локальный 271 0 подстроки 400 Поток в сети 239, 524 Преобразование к срединной оси 610 Прима, алгоритм 215, 501 Проверка числа на простоту 440 Произвольная выборка 268 Прюфера, код 481 Путь 188,227 Р Разбиение множества 220, 408 Разложение на множители 440 Размещение прямоугольников по корзинам 245 Разрывающее множество 572 Расстояние 0 хаусдорфово 620 0 Хемминга619 Реберная раскраска 561 Реберное покрытие 546 Рекуррентное соотношение 157 Рекурсивный объект 39 Решето числового поля 441 Рисование 0 графа 528 0 дерева 532 С Связность графа 521 Селективная сборка 280 Сжатие текста 647 Симплекс-метод 432 Синтаксический разбор 318 Система линейных уравнений 417 Словарь 91.389 Сложность алгоритма 50 Случайные числа 436 Сортировка 456 0 блочная 150 0 быстрая 143 0 вставками 21,60, 137 0 методом выбора 59, 129 0 пирамидальная 129 0 слиянием 141 0 топологическая 202, 497 Список 87 0 вставка элемента 88 0 поиск элемента 87 0 смежности 174, 213, 402 0 удаление элемента 88 Сравнение строк 61, 300, 306, 638, 641 Стек 90 Строка 38,631 Структуры данных 84 Судоку 259 Сумма Минковского 628 Суффиксное дерево 398 Сходство фигур 619 т Текст, сжатие 647 Топологическая сортировка 171,497 Точка 38 0 местоположение 579, 599 0 ближайшая 592 Транзитивное замыкание 233, 511 Триангуляция 104,321,585 У Указатель 86 Уменьшение ширины ленты 420
Предметный указатель 719 Умножение матриц 63, 422 Уоринга проблема 70 Упорядочивание последовательности 245 Ф Фибоначчи 294 Фильтр Блума 408 Флойда-Варшалла. алгоритм 232, 508 Фурье 451 X Хаусдорфово, расстояние 620 Хаффмана, код 649 Хемминга, расстояние 619 Хиршберга. алгоритм 644 Хэширование 112 Хэш-табпица 107 Хэш-функция 107 ц Целочисленное программирование 354 Цифровая подпись 654 ч Частотное распределение I24 Числа Фибоначчи 294 ш Шифр 0 AES652 0 DES651 0 RSA652 0 сдвиг Цезаря 651 Штейнера, дерево 223. 568 Штрих-код 328 э Эйлеров цикл 517
Стивен С. Скиена Алгоритмы. Руководство по разработке 2-е издание Перевод с английского Группа подготовки издания: Главный редактор Зам. главного редактора Зав. редакцией Перевод с английского Редактор Компьютерная верстка Корректор Оформление обложки Зав. производством Екатерина Кондукова Татьяна Лапина Григорий Добин Сергея Таранушенко Ирина Иноземцева Ольги Сергиенко Зинаида Дмитриева Елены Беляевой Николай Тверских Лицензия ИД № 02429 от 24.07.00. Подписано в печать 29.04.11. Формат 70х100'/)6. Печать офсетная. Усл. печ. л. 58 05 Тираж 1500 экз Заказ № 3741 "БХВ-Петербург”, 190005, Санкт-Петербург, Измайловский пр„ 29 Санитарно-эпидемиологическое заключение на продукцию № 77.99.60.953.Д.005770.05.09 от 26.05.2009 г. выдано Федеральной службой по надзору в сфере защиты прав потребителей и благополучия человека. Отпечатано с готовых диапозитивов в ГУП "Типография “Наука" 199034, Санкт-Петербург, 9 линия, 12
Наиболее полное руководство по разработке эффективных алгоритмов! Второе издание - Эта книга является настоящей сокровищницей алгоритмов, собрать которые в од- ном месте было работой не из легких. Каталог задач и обширная библиография делают книгу неоценимым подспорьем для любого, кто интересуется этой темой. Обозрение Ассоциации вычислительной техники (www.reviews.com) Книга является наиболее полным руководством по разработке эффективных алго- ритмов. Первая часть книги содержит практические рекомендации по разработке алгоритмов: приводятся основные понятия, дается анализ алгоритмов, рассматри- ваются типы структур данных, основные алгоритмы сортировки, операции обхода графов и алгоритмы для работы со взвешенными графами, примеры использования комбинаторного поиска, эвристических методов и динамического программирова- ния. Вторая часть книги содержит каталог наиболее распространенных алгоритми- ческих задач, для которых перечислены существующие программные реализации. Приведен обширный список литературы. • Большой объем обучающего материала и упражнений • Выделение основных понятий в конце каждой главы Уникальный каталог наиболее часто встречающихся на практике 75 алгоритмических задач • Ссылки на литературу и интернет-ресурсы по реализации алгоритмов на языках С, C++ и Java • Примеры задач для соискателей при приеме на работу в компании по разработке программного обеспечения Книгу можно использовать в качестве справочника по алгоритмам для программи- стов, исследователей и в качестве учебного пособия для студентов соответствую- щих специальностей. Об авторе: Стивен С. Скиена (Steven S. Skiena), профессор кафедры вычислитель- ной техники университета Стоуни — Брук, известный исследователь алгоритмов, лауреат премии института IEEE, автор популярной книги «Programming Challenges: The Programming Contest Training Manual» («Олимпиадные задачи по программиро- ванию. Руководство по подготовке к соревнованиям»). Полный набор слайдов лекций автора книги и другой дополнительный материал находится на веб-сайте www.algorist.com БХВ-Петероург 190005, Санкт-Петербург, Измайловский пр., 29 E-mail mail@bhv.ru Internet wwwbhv.ru Тел.:(812)251-42-44 Факс:(812)320-01-79
Предыдущие статьи: “Руководство по структурам данных и алгоритмам: введение и настройка среды”
Алгоритм — это пошаговая процедура, которая определяет набор выполняемых в том или ином порядке инструкций для получения желаемого результата. Алгоритмы обычно создаются независимо от базовых языков программирования, т. е. с возможностью реализации на нескольких языках.
С точки зрения структур данных, важны следующие категории алгоритмов:
- Алгоритм поиска элемента в структуре данных.
- Алгоритм сортировки элементов в определенном порядке.
- Алгоритм вставки элемента в структуру данных.
- Алгоритм изменения имеющегося в структуре данных элемента.
- Алгоритм удаления элемента из структуры данных.
Характеристики алгоритма
Не все процедуры можно назвать алгоритмом. Алгоритм должен обладать следующими характеристиками:
- Однозначность. Алгоритм должен быть четким и однозначным. Каждый из его шагов, а также данные в вводе/выводе должны быть четкими и приводить только к одному значению.
- Входные данные. В алгоритме должны быть четко определенные входные данные, но входные данные могут и отсутствовать.
- Выходные данные. Алгоритм должен иметь четко определенные выходные данные и соответствовать желаемому результату.
- Конечность. Алгоритмы должны завершаться после конечного числа шагов.
- Осуществимость. Должен быть осуществим при имеющихся ресурсах.
- Независимость. Алгоритм должен иметь пошаговые инструкции, независимые от любого программного кода.
Как написать алгоритм?
Это, скорее, зависит от задачи и ресурсов. Четко определенных стандартов написания алгоритмов не существует. Алгоритмы никогда не пишут для поддержки того или иного программного кода.
Как известно, у всех языков программирования общие базовые конструкции кода, например циклы (do, for, while), управление потоком (if-else) и т. д. Эти общие конструкции могут быть использованы для написания алгоритма.
Алгоритмы обычно пишут пошагово, но не всегда. Написание алгоритма — это процесс, который выполняется после четкого определения проблемной области. То есть надо знать проблемную область, для которой разрабатывается решение.
Пример
Попробуем научиться писать алгоритмы на примере.
Задача: разработать алгоритм сложения двух чисел и отображения результата:
Шаг 1 − НАЧАЛО ADD
Шаг 2 − объявляем три целых числа a, b и c
Шаг 3 − определяем значения a и b
Шаг 4 − добавляем значения a и b
Шаг 5 − сохраняем результат шага 4 в c
Шаг 6 − печатаем c
Шаг 7 − КОНЕЦ ADD
Алгоритмы сообщают программистам, как писать код программы. Алгоритм может быть написан и в таком виде:
Шаг 1 − НАЧАЛО ADD
Шаг 2 − получаем значения a и b
Шаг 3 − c ← a + b
Шаг 4 − отображение c
Шаг 5 − КОНЕЦ ADD
При разработке и анализе алгоритмов второй метод используется обычно для описания алгоритма. Это позволяет упростить его анализ, игнорируя все нежелательные определения. Аналитик может наблюдать, какие операции используются и как протекает процесс.
Писать номера шагов необязательно.
Алгоритм разрабатывается для получения решения задачи. А у задачи может быть несколько решений:
Поэтому для задачи могут быть разработаны алгоритмы с множеством решений. Следующий этап — анализ этих предложенных алгоритмов и реализация наиболее подходящего решения.
Анализ алгоритмов
Эффективность алгоритма можно проанализировать на двух разных этапах — до и после реализации. Вот эти этапы:
- Априорный анализ — это теоретический анализ алгоритма. Эффективность алгоритма измеряется при условии, что все остальные факторы, например скорость процессора, постоянны и не влияют на реализацию.
- Апостериорный анализ — это эмпирический анализ алгоритма. Выбранный алгоритм реализуется с использованием языка программирования, а затем выполняется на целевом компьютере. При проведении этого анализа собираются фактические статистические данные, такие как время выполнения и требующееся пространство.
Изучим априорный анализ алгоритмов. При анализе алгоритмов учитывается время выполнения или длительность различных задействованных операций. Время выполнения операции может быть определено как количество команд, выполняемых компьютером за одну операцию.
Сложность алгоритма
Пусть X — это алгоритм, n — размер входных данных, а используемые в алгоритме X время и пространство — это два основных фактора, которые определяют его эффективность.
- Фактор времени. Время измеряется путем подсчета количества ключевых операций, например сравнений в алгоритме сортировки.
- Фактор пространства. Пространство измеряется путем подсчета максимального объема памяти, требующегося алгоритму.
Сложностью алгоритма f(n) определяется время выполнения и/или дисковое пространство, требующиеся алгоритму, где n — это размер входных данных.
Пространственная сложность
Пространственная сложность алгоритма представляет собой объем памяти, требующийся алгоритму в течение его жизненного цикла. Этот объем состоит из двух частей:
- Фиксированная часть, то есть пространство, необходимое для хранения определенных данных и переменных, не зависящих от размера задачи. Например, используемые простые переменные и константы, размер программы и т. д.
- Изменяемая часть, то есть пространство, необходимое переменным, зависящим от размера задачи. Например, динамическое выделение памяти, пространство стека рекурсии и т. д.
Пространственная сложность S(P) любого алгоритма P определяется как S(P) = C + SP(I), где C — это фиксированная , а SP(I) — это изменяемая часть алгоритма, зависящая от характеристики экземпляра I. Вот простой пример, объясняющий эту концепцию:
Алгоритм: SUM(A, B)
Шаг 1 - НАЧАЛО
Шаг 2 - C ← A + B + 10
Шаг 3 - Конец
Здесь есть три переменные A, B, и C и одна константа. Поэтому пространственная сложность S(P) = 1 + 3. Пространство зависит от типов данных заданных переменных и типов констант и будет в соответствии с этим увеличиваться.
Временная сложность
Временная сложность алгоритма представляет собой количество времени, требующееся для его выполнения до завершения. Она может быть определена в виде численной функции T(n), где T(n) может измеряться количеством шагов при условии, что на каждый шаг расходуется постоянное время.
Например, сложение двух n-битных целых чисел проходит в n шагов. Значит, общее вычислительное время составит T(n) = c ∗ n, где c — это время, необходимое для сложения двух битов. Здесь имеет место линейный рост T(n) по мере увеличения размера входных данных.
Читайте также:
- Как алгоритм «случайный лес» вычисляет продавцов-мошенников на онлайн-рынке
- 6 техник, которые помогут вам учиться лучше
- Почему люди подсаживаются на TikTok? Алгоритм ИИ, который вас подловил
Читайте нас в Telegram, VK и Яндекс.Дзен