Основы функционального программирования/Трансформация программ
Основы функционального программирования
- Вводная лекция
- Структуры данных и базисные операции:
Под программой на некотором языке понимается некоторый текст на . В случае функционального языка под программой понимается набор клозов. Семантика же языка определяется, если задан интерпретатор этого языка. Интерпретатор определяется формулой:
где — программа;
Невозможно разобрать выражение (SVG (MathML можно включить с помощью плагина для браузера): Недопустимый ответ («Math extension cannot connect to Restbase.») от сервера «http://localhost:6011/ru.wikibooks.org/v1/»:): {\displaystyle d} — исходные данные;
— выходные данные.
Если интерпретатор представлен в виде каррированной функции такой, что , тогда определение , а точнее функционал называется денотационной семантикой языка . В этом случае имеет смысл -выражение: . При этом частичную функцию можно рассматривать как функцию одного аргумента, которая есть функция, реализующая программу . Как было показано в предыдущей лекции, можно построить рекурсивное определение вида: . Такой вид изначально имеют все программы на функциональном языке. Но эту запись можно понимать двояко:
- Это определение можно понимать просто как строку символов, подаваемую на вход интерпретатора. Функция, вычисляемая интерпретатором по тексту , обозначается как .
- — есть чистое математическое определение функции . Решение этого уравнения обозначается как .
Резонный вопрос: каково соотношение этих двух функций — и ? Надо полагать, что корректный интерпретатор как раз вычисляет .
Определение: говорят, что функция менее определена, чем функция (обозначается как ), если . Если для двух функций одновременно выполняется и , то имеет место тождество функций.
Как правило, — это происходит потому, что обычно интерпретатор реализует аппликативную стратегию редукции. Однако в дальнейших рассуждениях будет полагаться тождество функций и , поэтому тексты программ будут рассматриваться как математическое определение функций. Тогда эквивалентное преобразование функциональных программ есть просто эквивалентные преобразования специального вида математических определений функций.
Трансформация программ — это просто синтаксические преобразования, во время которых совершенно не затрагивается семантика программ, так как программа воспринимается просто как набор символов. Обозначение того факта, что один текст получаем при помощи синтаксических трансформаций другого текста , выглядит следующим образом: .
Говорят, что преобразование корректно, если .
Говорят, что преобразование эквивалентно, если .
Далее вводится еще несколько специальных обозначений:
- Имеется исходный набор клозов (то есть объектов преобразования), и этот набор будет обозначаться либо DEF, либо SPEC.
- Клозы, описывающие функцию, которая содержит отображения из исходных клозов, обозначается как INV.
- Некоторые равенства, выражающие свойства внутренних (зарезервированных) функций, обозначаются как LAW.
Определение: пусть — некоторое выражение (равенство), в котором выделены все свободные вхождения терма . Тогда называется примером . При этом считается, что — это некоторое выражение.
Виды преобразований
править1°. Конкретизация (instantiation) — INS.
2°. Преобразование — без названия :)
3°. Развёртка (unfolding) — UNF.
4°. Свёртка (folding) — FLD.
5°. Закон (law) — LAW.
6°. Абстракция и аппликация (abstraction & application) — ABS.
Пример 30. Преобразование функции length
.
length [] = 0 1 (DEF)
length (H:T) = 1 + length T 2 (DEF)
length_2 L1 L2 = length L1 + length L2 3 (SPEC)
length_2 [] L = length [] + length L 4 (INS 3)
= 0 + length L 5 (UNF 1)
= length L 6 (LAW +) (*)
length_2 (H:T) L = length (H:T) + length L 7 (INS 3)
= (1 + length T) + length L 8 (UNF 2)
= 1 + (length T + length L) 9 (LAW +)
= 1 + length_2 T L 10 (FLD 3) (**)
Теперь остаётся взять два полученных клоза (*) и (**) и составить из них новое рекурсивное определение новой функции, не использующее вызова старой функции:
length_2 [] L = length L
length_2 (H:T) L = 1 + length_2 T L
Однако следует отметить, что выбор новых клозов для формирования нового определения требует дополнительных исследований, а не выполняется автоматически.
Подобная трансформация определений функций часто будет приводить к уменьшению сложности определения функций. Например, для функции, вычисляющей -ое число Фиббоначи можно построить определение, сложность вычисления которого линейно зависит от , а не по закону Фиббоначи, как это есть для обычного определения.
Но трансформации следует делать обдуманно, так как можно придти к бесконечным циклам шагов FLD и UNF. Чтобы при трансформации не придти к абсурду, необходимо следить за тем, чтобы в процессе преобразования общность получаемых выражений не увеличивалась. То есть трансформацию надо осуществлять от общего к частному.
Второй закон информатики
править- Существуют неразрешимые задачи.
- Не существует эффективной реализации для декларативных языков, если они универсальны.
Концепция трансформационного синтеза: позволить программисту писать определения функций, не заботясь об их эффективности.
Однако было доказано, что по языку спецификаций невозможно выработать (то есть трансформировать исходный текст) вариант функции, работающий эффективно. Если в качестве языка спецификаций рассматривать функциональный язык, то становится понятно, что программист сам должен заботиться об эффективности своих программ — концепция трансформационного синтеза здесь не пройдёт.
Частичные вычисления
правитьПусть и — два языка, работающих со строками символов (это не нарушает общности рассуждений), а и — множества синтаксически правильных программ на соответствующих языках. — домен всевозможных символьных последовательностей.
Если — программа из , то:
, и
В последнем равенстве переменными обозначены неизвестные данные. Для программы эти переменных представляют остаточный код.
Частичным вычислителем MIX называется программа из (хотя, частичный вычислитель может быть реализован на любом языке) такая, что:
.
То есть MIX — это программа, которая, получив некоторую исходную программу и данные для её известных параметров, выдаёт остаточную программу для исходной.
Интерпретатором языка называется программа такая, что:
.
Компилятором языка называется программа такая, что:
.
Или, что то же:
.
Компилятором компиляторов языка называется программа такая, что:
.
Строго говоря, эти три утверждения являются теоремами, которые, однако, вполне легко доказываются при помощи определений, данных выше.