подавляющее большиство фукций на пространстве последовательостей? |
Базисные схемы обработки информации
Основная цель данной главы — показать, как теория помогает практике. Красивые и эффективные алгоритмы решения многих непростых задач при правильном подходе к ним будут получаться как бы "сами собой". Процесс создания программ при этом в значительной мере автоматизируется, а вместе с решением мы получаем еще и доказательство правильности построенных программ.
В качестве дополнительного источника информации к данной главе можно порекомендовать книги [4] и [9].
Все задачи на написание программ можно разделить на следующие три группы.
1. Простейшие задачи. В этот класс задач мы отнесем линейные программы и программы с ветвлениями, которые не используют ни циклов, ни рекурсивных вызовов. Создание подобных программ обычно не вызывает серьезных проблем, а их правильность легко может быть проверена с помощью методов, изложенных в предыдущей главе.
2. Задачи на "программирование в малом". Большая часть задач, которые уже были нами рассмотрены, относятся именно к этой категории. Кроме совсем простых к ней относятся и более сложные, при решении которых оправдано применение различных методов разработки программ типа проектирования сверху вниз.
Характеристическим свойством задач на "программирование в малом" является их простая сводимость к совокупности таких задач на обработку информации, для решения которых достаточно применения итерации или рекурсии.
3. Большие проекты. Почти все практические задачи относятся именно к этому классу. Для них даже аккуратная формулировка постановки задачи является проблемой. Решение подобных задач требует применения специальных методов, большая часть из которых выходит за рамки нашего курса.
Объектно-ориентированное проектирование в значительной мере помогает справиться со сложностью подобных задач, позволяя свести их к решению многих значительно более простых. Проекты, которые мы рассмотрим в следующей главе, позволят проиллюстрировать это на практике.
Предложенное деление задач на группы является достаточно условным и некоторые из них могут быть отнесены как ко второй, так и к третьей. В данной главе мы сосредоточимся на методах решения наиболее простых задач из второй категории, программная реализация которых не превосходит 10-20 строк текста. Все подобные задачи можно рассматривать, как задачи на обработку информации, что и поясняет название классификации методов их решения — схемы обработки информации, графическая иллюстрация взаимосвязи которых приведена на рис. 7.1.
Рекурсия и итерация
Как определения этих двух базисных схем обработки информации, так и их взаимосвязь уже были рассмотрены нами ранее. Напомним самое главное.
Рекурсия — это такой способ организации обработки данных, при котором программа вызывает сама себя непосредственно, либо с помощью других программ.
Итерация — способ организации обработки данных, при котором определенные действия повторяются многократно, не приводя при этом к рекурсивным вызовам программ.
Любой алгоритм, реализованный в рекурсивной форме, может быть переписан в итерационном виде, и наоборот.
Некоторые особенности применения рекурсии уже были рассмотрены при решении задачи 5.1. Сейчас мы продолжим исследование данного вопроса, уделяя особое внимание процессу построения программы и доказательству ее правильности. Рекурсивная программа обычно возникает, как результат вычисления рекурсивно определенной функции на множестве программных переменных, а доказательство ее правильности требует исследования двух вопросов:
1) всегда ли и почему программа заканчивает работу?
2) почему после окончания работы программы будет получен требуемый результат?
Рассмотрим следующую задачу.
Задача 7.1. Напишите рекурсивную программу, перемножающую два целых числа, одно из которых неотрицательно, без использования операции умножения. Точные пред- и постусловия требуемой программы, временная сложность которой не должна превосходить , таковы: , . Числа и в программе изменять нельзя.
Решение Фиксировав , рассмотрим функцию , задаваемую следующим рекурсивным определением:
,
, когда положительно и четно,
, когда положительно и нечетно.
Программа, вычисляющая функцию , легко пишется в соответствии с ее определением. В целях эффективности деление пополам и умножение на два реализуются с помощью сдвигов.
Текст программы
public class MulR { static int mul(int x, int y) { // Xterm.print(" Вызов mul(" + x + "," + y + ")\n"); return (y == 0) ? 0 : ((y&1) == 0) ? mul(x,y >>> 1) << 1 : mul(x,y-1) + x; } public static void main(String[] args) throws Exception { int a = Xterm.inputInt("a -> "); int b = Xterm.inputInt("b -> "); Xterm.println("a * b = " + mul(a,b)); } }
Эта программа заканчивает работу, так как каждый следующий рекурсивный вызов уменьшает значение аргумента функции не менее, чем на единицу (нечетное число уменьшается ровно на один, а четное — делится пополам, что при приводит к уменьшению на один, а при больших значениях — к большему уменьшению). В результате конечного числа таких операций гарантированно станет нулем, что приведет к завершению цепочки рекурсивных вызовов.
Обратите внимание на тот факт, что хотя приведенные выше рассуждения не являются справедливыми при отрицательных значениях числа , это не означает неправильности программы, так как согласно спецификации ее поведение при может быть произвольным.
Доказательство правильности получаемого результата проведем с помощью метода математической индукции, показав, что при фиксированном и является тавтологией предикат ( данная программа печатает ).
Справедливость базы индукции следует непосредственно из текста программы, а индуктивный переход осуществляется следующим образом. Если предположить, что программа правильна при , то . Рассмотрим далее два возможных случая.
Пусть сначала число — четно. Тогда результатом работы программы будет
. Первое равенство здесь следует из текста программы, а второе — из предположения индукции. Если число является нечетным, то . Обоснование этих равенств проводится аналогично.Таким образом, нами проверена истинность индуктивного перехода , что и завершает доказательство.
Для того чтобы увидеть цепочку рекурсивных вызовов функции достаточно убрать символы комментария из первой строки этого метода. Это позволяет понять, почему программа имеет сложность порядка . Докажем аккуратно, что глубина рекурсии (количество рекурсивных вызовов) для нашей программы не превосходит (здесь — целая часть от ).
Каждый рекурсивный вызов функции уменьшает значение аргумента либо на единицу, либо вдвое. Два последовательных вызова этой функции всегда приводят к уменьшению величины более, чем в два раза, поэтому обратится в нуль после не более, чем после рекурсивных вызовов. Обратите внимание, что мы не учитываем в наших рассуждениях самый первый, нерекурсивный вызов функции .
В качестве дополнительного задания можете попробовать объяснить те результаты, которые получаются при выполнении этой программы для отрицательных значений .
Рассмотрим еще одну задачу.
Задача 7.2. Напишите программу, печатающую значение многочлена степени в заданной точке . Коэффициенты многочлена хранятся в массиве в порядке убывания степеней и являются целыми числами, также как и значение . Величины , и элементы массива изменять в программе нельзя.
Решение Заметим, что
, где . Воспользовавшись этим, определим функцию следующим образом:,
, при .
Теперь можно написать программу, реализующую вычисление этой функции.
Текст программы
public class PolVal { static int a[], x0; static int p(int k) { return (k == 0) ? a[0] : x0 * p(k-1) + a[k]; } public static void main(String[] args) throws Exception { int n = Xterm.inputInt("n -> "); a = new int[n+1]; for (int i=0; i<=n; i++) a[i] = Xterm.inputInt("a[" + i + "] -> "); x0 = Xterm.inputInt("x0 -> "); Xterm.println("P(" + x0 + ") = " + p(n)); } }
Завершение программы за конечное число шагов следует из уменьшения на единицу при каждом рекурсивном вызове аргумента функции , а ее правильность получается из доказательства по индукции следующего утверждения ( программа печатает значение многочлена степени в точке ).
База индукции проверяется непосредственно, ибо при нулевом значении аргумента функция возвращает , что совпадает с . Индуктивный переход удается осуществить благодаря формуле, приведенной в начале решения данной задачи, которая позволяет выразить значение через .
Рекурсивное решение задачи зачастую является менее эффективным, чем эквивалентное ему по другим параметрам итерационное решение. Обычным способом программной реализации итерации является цикл, и поэтому мы будем считать, что спецификация задачи на итерацию имеет вид .
Рассмотрим — множество значений всех программных переменных (прямое произведение множеств значений отдельных переменных), его подмножества и , определяемые предикатами и , и преобразование , соответствующее однократному выполнению тела цикла S. Тогда построение условия продолжения цикла e и его тела S сводится к нахождению предиката и преобразования таких, что .
Именно этот факт является основным результатом, вытекающим из определения слабейшего предусловия для оператора цикла (см. "Спецификация программ и преобразователь предикатов" ). Геометрическая интерпретация математической модели итерации, сводящейся к многократному применению преобразования , приведена на рис. 7.2.
К сожалению эта математическая модель итерации является слишком общей и трудно применимой на практике, так как она не дает конкретных рекомендаций по нахождению условия продолжения цикла e и тела цикла S по заданным и . К рассмотрению более частных и более полезных схем мы сейчас и перейдем.