8.1.1. Следующая рекурсивная процедура вычисляет числа сочетаний
(биномиальные коэффициенты). Написать эквивалентную
нерекурсивную программу.
function C(n,k: integer):integer; | {n >= 0; 0 <= k <=n} begin | if (k = 0) or (k = n) then begin | | C:=1; | end else begin {0<k<n} | | C:= C(n-1,k-1)+C(n-1,k) | end; end;Замечание. Cnk -- число k -элементных подмножеств n -элементного множества. Соотношение Cnk = Cn-1k-1+Cn-1k получится, если мы фиксируем некоторый элемент n -элементного множества и отдельно подсчитаем k -элементные подмножества, включающие и не включающие этот элемент. Таблица значений Cnk называется треугольником Паскаля (того самого). В нем каждый элемент, кроме крайних единиц, равен сумме двух стоящих над ним.
Решение. Можно воспользоваться формулой Мы, однако, не будем этого делать, так как хотим продемонстрировать более общие приемы устранения рекурсии. Вместо этого составим таблицу значений функции , заполняя ее для , пока не дойдем до интересующего нас элемента.
8.1.2. Что можно сказать о времени работы рекурсивной и
нерекурсивной версий в предыдущей задаче? Тот же вопрос о
памяти.
Кардинальный выигрыш во времени при переходе от рекурсивной версии к нерекурсивной связан с тем, что в рекурсивном варианте одни и те же вычисления происходят много раз. Например, вызов C(5,3) в конечном счете порождает два вызова C(3,2):
Заполняя таблицу, мы каждую клетку заполняем только однажды -- отсюда и экономия. Этот прием называется динамическим программированием, и применим в тех случаях, когда объем хранимой в таблице информации оказывается не слишком большим.
8.1.3. Порассуждать на ту же тему на примере рекурсивной и
(простейшей) нерекурсивной программ для вычисления чисел
Фибоначчи, заданных соотношением
8.1.4. Дан выпуклый n -угольник (заданный координатами своих
вершин в порядке обхода). Его разрезают на треугольники
диагоналями, для чего необходимо n-2 диагонали (докажите
индукцией по n ). Стоимостью разрезания назовем сумму длин всех
использованных диагоналей. Найти минимальную стоимость
разрезания. Число действий должно быть ограничено некоторым
многочленом от n . (Перебор не подходит, так как число вариантов
не ограничено многочленом.)
Составив таблицу для a(k,l) и заполняя ее в порядке возрастания числа вершин (равного l-k+1 ), мы получаем программу, использующую память порядка n2 и время порядка n3 (однократное применение рекуррентной формулы требует выбора минимума из не более чем n чисел).
8.1.5. Матрицей размера называется прямоугольная
таблица из m строк и n столбцов, заполненная числами.
Матрицу размера можно умножить на матрицу размера
(ширина левого сомножителя должна равняться высоте
правого), и получается матрица размером . Ценой
такого умножения будем считать произведение mnk (таково число
умножений, которые нужно выполнить при стандартном способе
умножения -- но сейчас это нам не важно). Умножение матриц
ассоциативно, поэтому произведение s матриц можно вычислять в
разном порядке. Для каждого порядка подсчитаем суммарную цену
всех матричных умножений. Найти минимальную цену вычисления
произведения, если известны размеры всех матриц. Число действий
должно быть ограничено многочленом от числа матриц.
Решение. Представим себе, что первая матрица написана на отрезке [0,1] , вторая -- на отрезке [1,2] ,..., s -ая -- на отрезке [s-1,s] . Матрицы на отрезках [i-1,i] и [i,i+1] имеют общий размер, позволяющий их перемножить. Обозначим его через d[i] . Таким образом, исходным данным в задаче является массив .
Через a(i,j) обозначим минимальную цену вычисления произведения матриц на участке [i,j] (при ). Искомая величина равна a(0,s) . Величины a(i,i+1) равны нулю (матрица одна и перемножать ничего не надо). Рекуррентная формула будет такой: где минимум берется по всем возможных местам последнего умножения, то есть по всем . В самом деле, произведение матриц на отрезке [i,k] есть матрица размера d[i]d[k] , произведение матриц на отрезке [k,j] имеет размер d[k]d[j] , и цена вычисления их произведения равна d[i]d[k]d[j] .
Замечание. Две последние задачи похожи. Это сходство станет яснее, если написать матрицы-множители на сторонах 1 -2 , 2 -3 ,...,(s-1) -s многоугольника, а на каждой хорде i -j написать произведение всех матриц, стягиваемых этой хордой.
8.1.6. Железная дорога с односторонним движением имеет n
станций. Известны цены билетов от i -ой станции до j -ой (при
-- в обратную сторону проезда нет). Найти минимальную
стоимость проезда от начала до конца (с учетом возможной
экономии за счет пересадок).
Мы видели, что замена рекурсивной программы на заполнение таблицы значений иногда позволяет уменьшить число действий. Примерно того же эффекта можно добиться иначе: оставить программу рекурсивной, но в ходе вычислений запоминать уже вычисленные значения, а перед очередным вычислением проверять, нет ли уже готового значения.
8.1.7. Задано конечное множество с бинарной операцией (вообще
говоря, не коммутативной и даже не ассоциативной). Имеется n
элементов этого множества и еще один элемент
x . Проверить, можно ли так расставить скобки в произведении
, чтобы в результате получился x . Число
операций должно не превосходить Cn3 для некоторой константы
C (зависящей от числа элементов в выбранном конечном
множестве).
По существу этот же прием применяется в полиномиальном алгоритме проверки принадлежности слова произвольному контекстно-свободному языку (см. главу 13).
Следующая задача (задача о рюкзаке) уже упоминалась в главе 3.
8.1.8. Имеется n положительных целых чисел и
число N . Выяснить, можно ли получить N , складывая некоторые
из чисел . Число действий должно быть порядка
Nn .