УДК 004.432.42 Дата подачи статьи: 27.03.2014
СИСТЕМА ТИПОВОГО КОНТРОЛЯ ПРОГРАММ НА ЯЗЫКЕ ФУНКЦИОНАЛЬНОГО ПРОГРАММИРОВАНИЯ FPTL
(Статья подготовлена в рамках выполнения проекта РФФИ № 13-07-00810)
И.А. Бочаров, магистрант; В.П. Кутепов, д.т.н., профессор; П.Н. Шамаль, аспирант (Национальный исследовательский университет «Московский энергетический институт», ул. Красноказарменная, 14, г. Москва, 111250, Россия, [email protected], [email protected], [email protected])
Описаны методы и алгоритм статического (до выполнения программы) контроля типов функциональных программ на созданном языке функционального параллельного программирования FPTL. Язык реализован на многоядерных компьютерах и успешно применяется на практике. FPTL - чисто функциональный язык, основанный на построении функций путем применения четырех операций композиции функций и рекурсивных определений. Три из этих операций являются параллельными, что позволило создать достаточно эффективные алгоритмы параллельного выполнения программы на многоядерных компьютерах, используя интерпретацию программы. Это принципиально отличает FPTL от других функциональных языков программирования, основанных на лямбда-нотации (Haskell, ML и др.). Система типов данных в языке состоит из встроенных типов (real, int, bool и др.) и определяемых абстрактных типов данных. Функции в языке являются в общем случае полиморфными. В теоретической части статьи определены условия правильной типизации FPTL-программ, на основе которых разработан алгоритм контроля типовой правильности программы до ее выполнения. В существующей реализации языка на многоядерных компьютерах типовой контроль осуществляется в процессе выполнения программы, что занимает около 30 % времени ее выполнения. Разработанная система типового контроля является статической и позволяет заметно сокращать время выполнения параллельных программ.
Ключевые слова: функциональные языки программирования, параллельные вычисления, типовой контроль.
TYPE CHECKING SYSTEM FOR FUNCTIONAL PROGRAMMING LANGUAGE FPTL
(The article is written as a part of the Russian Foundation for Basic Research project № 13-07-00810)
Bocharov I.A., Master Student; Kutepov V.P., Dr.Sc. (Engineering), Professor; Shamal P.N., Postgraduate Student (National Research University "MPEI", Krasnokazarmennaya St. 14, Moscow, 111250, Russian Federation, [email protected], [email protected], [email protected])
Received 27.03.2014
Abstract. The paper describes static type checking methods and algorithm for functional parallel programming language FPTL. FPTL is implemented on multicore systems and is widely used. FPTL is a pure functional language based on function compositions. By function composition we understand the application of four composition operations and recursive definitions. Three of composition operations are parallel. It made possible to implement rather effective algorithms of parallel program execution on multicore systems, using program interpretation. This is the key difference between FPTL and other functional languages, based on lambda-notation (such as Haskell, ML, etc.). Language type system consists of built-in types (int, real, bool, etc.) and user-defined abstract types. Functions in FPTL language are in general polymorphic. In theoretical section of this paper rules of correct typing and algorithm of type checking based on these rules are described. In the current language implementation on multicore system correctness of typing is checked while program is being executed. Type checking takes about 30 % of program execution time. Type checking system presented in the paper helps to significantly reduce time of program execution.
Keywords: functional programming languages, parallel computing, type checking.
Язык FPTL (functional parallel typified language) создавался с целью эффективного параллельного выполнения функциональных программ на компьютерных системах с общей памятью. В отличие от известных языков функционального программирования, таких как LISP [1], ML [2], Haskell [3, 4], которые в большей степени основаны на лямбда-исчислении [5], FPTL базируется на модели построения новых функций путем применения операций композиции функций и общей формы их задания в виде систем функциональных уравнений. Язык рекурсивных функций [6] - очевидный пример такого подхода к заданию вычислимых функций.
Кажущееся преимущество лямбда-основанных функциональных языков, обычно формулируемое как единообразный способ задания функций выс-
ших порядков, компенсируется в FPTL использованием в задании функций функциональных параметров. Этот традиционный способ повышения порядка функции допускает возможность в подставленной вместо функционального параметра функции снова использовать функциональные параметры, увеличивая таким образом функциональную иерархию в построении функций.
Однако FPTL имеет ряд преимуществ перед основанными на лямбда-исчислении функциональными языками. Наиболее важными на практике являются следующие его особенности:
- очевидная близость используемой нотации к традиционной математической форме задания функций;
- алгебраизированная форма задания функции, позволяющая извлекать ее композиционную
структуру (схему функции), что существенно улучшает проектирование, анализ и эквивалентные преобразования функций [7-9];
- возможность явного отражения и варьирования параллелизма в функциональной программе, поскольку три из четырех операций композиции являются параллельными;
- простой алгоритм обнаружения параллелизма при выполнении, что позволяет исключить необходимость явного указания с помощью соответствующих процессных примитивов (как, например, в Haskell [4]), какие фрагменты программы могут выполняться параллельно;
- простота в освоении; язык позволяет широко использовать библиотеки, как правило, оптимизированных функций, вводить абстрактные типы данных, использовать разработанные для него приложения для оценки сложности параллельного выполнения программы, приведения ее к оптимальной параллельной форме, анализа структурной сложности программ [10] и др.
Одна из последних реализаций языка FPTL [11] была недостаточно эффективной, так как выполнялась на компьютерах с распределенной памятью (многоядерные компьютеры в то время еще не были широко распространены). Реализация FPTL на многоядерных компьютерах, напротив, оказалась успешной. Это подтверждают приведенные в [12] экспериментальные данные ускорения FPTL-программ.
В данной реализации, которая базируется на принципе интерпретации программы и динамическом выявлении и регулировании зернистости параллелизма и выполнена с использованием ни-тевого процессного параллелизма, можно достигать заметного ускорения при выполнении FPTL-программ [12].
FPTL - строго типизированный язык, однако контроль типовой правильности (корректности применения функций к их аргументам), который осуществляется в данной реализации динамически, то есть в процессе выполнения программы, увеличивает время ее выполнения на величину, составляющую до 30 % от времени выполнения программы. Заметим, что продвинутые механизмы контроля типов присутствуют в компиляторах и интерпретаторах многих функциональных языков, включая компилятор языка Haskell GHC [13].
В настоящей статье описана разработанная система статического (до выполнения программы) определения ее типовой корректности, что позволит существенно уменьшить время выполнения FPTL-программы.
Язык FPTL
Язык FPTL создавался как прообраз общепринятой в математической практике композиционной формы задания функций в общем случае в ви-
де систем рекурсивных функциональных уравнений, в которой при построении новых функций используются операция подстановки функций вместо функциональных переменных и условный оператор.
В FPTL введены четыре простые бинарные композиции функций, которые позволяют более тонко отражать композиционное представление функций и явно задают параллелизм при вычислении их значений.
Функции в FPTL рассматриваются как типизированные (т, п)-арные соответствия (т>0, п>0) между кортежами данных, где т - длина кортежа на входе функции, п - на выходе. Функции арности (0, 1) рассматриваются в FPTL как константы. Для т и п, равных 0, имеем кортеж нулевой длины, обозначаемый X, со свойствами Ха=аХ=а, где а
- произвольный кортеж. Кортеж данных в языке представляется в виде последовательной записи его элементов.
В отличие от общепринятой формы задания функций с явным указанием ее аргументов (так называемая форма задания общего значения функции) в FPTL строго различаются собственно функция как отображение одного множества в другое, а также ее аппликация к конкретным данным. Роль переменных в задании функций в FPTL выполняют функции выбора необходимого элемента из кортежа данных. Формально функция выбора аргумента, обозначаемая 1(/, т) (в языке -просто [/]), при применении к кортежу произвольного типа данных длины т (т>0) выбирает его /-й элемент, 0</<т, т>0.
Для /=0 выбираемое значение есть X. Функции в FPTL являются в общем случае частичными, причем неопределенное значение функции может быть выражено либо как неограниченный процесс вычисления ее значения, либо как специальное вычисляемое неопределенное значение, обозначаемое ю, со свойствами юа=аю=а для любого кортежа а.
Формально функции определяются как системы функциональных уравнений ^,=т„ /= 1, п, где тi
- функциональные термы, построенные из заданных (базисных) функций и функциональных переменных Fi путем применения четырех операций композиции функций: +, *, •. Для функций и функциональных переменных задана их арность, а для базисных функций еще и их тип.
Пусть /1, / - заданные функции, / - определяемая новая функция; а, р, у - обозначение кортежей. Синтаксис и семантика операций композиции определяются ниже. Сначала задается синтаксическая форма операции композиции, а затем ее семантическое значение; символ ^ обозначает равенство по определению.
1. Последовательная композиция (•): "Ь //"'к) • /2(к,л);/а)ь/2(/1(а)).
2. Операция конкатенации кортежей значений функций (*):
/К.) ^ ^ > * /^ >;Ла)^Яа)Ма).
3. Операция условной композиции:
у (>». п) ^ у(т, к) ^ у(т, п) .
/(а)«
/2 (а), если / (а) отлично
от значения «ложь» и ю,
ю - иначе.
4. Операция объединения (графиков) ортогональных функций:
/(m, п) n) n)
^ J1 + J 2 ;
f (a), если значение f (a) определено, f (a) о < f2 (a), если значение f (a) определено, ю, если f (a) и f2 (a) не определены.
Напомним, что ортогональными называются такие функции f и f2, что для всякого кортежа данных а определена не более, чем одна из них. Например, общепринятая форма условного оператора if p(x) then f1(x) else f2(x) в FPTL представляется как (p^fi)+( p ^f2). Операция объединения ортогональных функций была введена с целью представления параллельных функций (известный пример - функция голосования в телефонии).
Все операции композиции обладают ассоциативностью, а операция объединения ортогональных функций является коммутативной. Приоритет операций композиции определяется следующим образом (в порядке возрастания): +, *, •. Введение приоритета операций позволяет опускать часть скобок в записи функций.
При вычислении значений функций только операция последовательной композиции предполагает строгий порядок вычисления значений функций, к которым она применена. Остальные три операции композиции параллельны, причем для операций *, ^ допустим произвольный порядок вычисления значений функций, к которым она применена. Для операции + только одновременное (или квазипараллельное) вычисление значений функций гарантирует получение результата.
Как уже было отмечено, термы в задании функций в виде системы функциональных уравнений F,=t„ i= 1, n , представляют собой функциональные выражения, построенные из базисных функций и функциональных переменных путем применения операций композиции. Предполагается, что арности терма и определяемой им функциональной переменной в системе функциональных уравнений одинаковы. Функциональные переменные выполняют двойную роль при задании функции (построении функциональной программы): одни из них появляются как необходимые элементы при задании рекурсивных функций, другие определяются далее (в следующем уравне-
нии уточняемой функции). Это позволяет просто реализовать пошаговую разработку функциональной программы по технологии проектирования «сверху-вниз», уточняя введенную функциональную переменную, определяя ее в следующем функциональном уравнении.
Определяемые абстрактные типы данных в FPTL задаются по аналогии с заданием функций в виде систем реляционных уравнений. В FPTL можно использовать встроенные типы bool, real, int, string и др. вместе с определенными в конкретной компьютерной системе операциями над ними.
Для определения новых типов данных используются конструкторы и деструкторы (обратные конструкторам функции). Для определения типов данных применяются те же операции композиции, которые применяются для задания функций, за исключением того факта, что операция + трактуется как операция объединения двух множеств данных. Используемые в определении абстрактного типа данных (системе реляционных уравнений) конструкторы и деструкторы выполняют роль базисных функций. В проведенных ранее исследованиях [9] доказано, что их достаточно для того, чтобы в FPTL можно было представить любую вычислимую над абстрактными типами данных функцию. При определении абстрактных типов данных можно использовать встроенные типы данных. В языке также имеются конструкторы констант, позволяющие создавать данные встроенных типов.
Приведем пример пользовательского типа. Определим тип «стек комплексных чисел»:
data StackOfComplex{
Complex=real*real.c_complex;
StackOfComplex=c_empty+(StackOfComplex*Complex).c_head;
}
В данном определении приводятся три конструктора: ccomplex, cempty и c_head, имеющие арности (2,1), (0,1) и (2,1) соответственно. В паре с определенными конструкторами определяются и деструкторы. Определения деструкторов извлекаются из определения типов и интерпретируются следующим образом:
~ c _ complex(x) = y, если x = c _ complex(y), ю - иначе,
~ с _ empty = X, если x = c _ empty, ю - иначе,
~ c _ head(x) = y, если x = c _ head(y), ю - иначе.
В язык FPTL добавлена возможность задания параметризованных типов и функций. Приведем пример определения параметризованного стека, используя параметр 't (в FPTL обозначение типового параметра предваряется апострофом):
data Stack['t]{
Stack=c_empty+(Stack['t] * 't).c_head;
}
Приведем пример программы, вычисляющей количество элементов стека:
data StackOfComplex{
Complex = real * real.c_complex;
StackOfComplex=c_empty+(StackOfComplex*
*Complex).c_head; }
scheme Count {
Count=~c_empty^0, ([1].~c_head.[2].Count * 1).add;
}
application
stack=(((c_empty*((0.1 *0.1).c_complex)).c_head)*((0.1 * *0.1). c_complex)).c_head;
%Count(stack);
FPTL также поддерживает импорт функций из внешних модулей. Для импорта функции из другой функциональной программы или библиотеки используется конструкция: from «имя модуля» import «имя функции».
Как уже отмечалось, язык FPTL имеет несколько реализаций и реализация FPTL на многоядерных компьютерах обладает рядом особенностей. Наиболее важной особенностью в контексте типового контроля является введение тернарной операции условной композиции взамен операций ^ и +, которые главным образом предназначены для описания условных конструкций. Новая тернарная операция - аналог условного оператора if-then-else в традиционных языках программирования; она представляется в FPTL как (p^f, f2), где p - предикат, f и f2 - функции, значение одной из которых будет использовано в зависимости от того, истинно или ложно значение p(x).
Язык FPTL спроектирован так, что программирование на нем позволяет при помощи минимального набора средств, привычного для математической функциональной нотации, не прибегая к введению дополнительных и обычно нарушающих общую архитектуру языка примитивов, отражать и регулировать параллелизм в программе.
Задание типов в FPTL
Напомним, что в общем случае функции в FPTL задаются в виде систем рекурсивных функциональных уравнений: F,=t„ i= 1, n , где Fi - определяемые функции (функциональные переменные); т - термы, построенные применением описанных выше операций композиции к множеству базисных функций Fbasis и функциональных переменных F={Fi | i= 1, n }.
Определение. Пусть T - множество константных типов, которое состоит из множества встроенных в FPTL типов (real, int, float, bool и др.) и абстрактных (определяемых пользователем в FPTL) типов; X - множество типовых переменных, таких, что каждая переменная xeX принимает в качестве значений типы из T. Пусть далее Mk=M\ xM2x... xMk - множество кортежей длины k, построенных из элементов множества M=TuX
Очевидно, что M0 состоит из единственного пустого кортежа, обозначаемого (как было указано выше) Я.
Кортеж типов длины k записывается следующим образом: z1 xz2x... xzk, z, eM, i=1, k .
Кортеж типов, не содержащий переменные, называется константным. Кортеж типов, содержащий переменные, интерпретируется как множество константных кортежей, полученных путем всевозможных подстановок константных типов из Т вместо вхождения в кортеж переменных типов.
Определение. Содержательно тип (m, п)-арной функции определяется как множество однозначных отображений {t1(l)xt2(l)x.xtm(l)^t1'(l)xt2'(l)x ...xt„'(,), i=1, 2, ...}, где справа и слева от символа — стоят константные кортежи. Множество однозначных отображений может быть как конечным, так и счетным.
Синтаксически тип функций задается в форме {zi(,)xz2(,)x.xzm(,)^zi'(,)xz2'(,)x.xz„'(,), i=1, 2, ..., k}. В интерпретации эта запись представляет множество константных отображений, полученных всевозможными подстановками константных типов в типовые кортежи слева и справа от символа —. Предполагается, что одинаковым переменным, входящим в кортежи, присваиваются одни и те же типовые константы при подстановке.
Требование однозначности в определении типа функции существенно, поскольку в случае его невыполнения мы получаем неопределенность. Про-
/m n)
' равен {intxint—real, intxint—int}. Представим ситуацию, когда на входе f имеется кортеж из двух целых чисел. Понятно, что мы не можем однозначно определить выходной типовой кортеж, поскольку неочевидно, какое число - целое или вещественное - нам ожидать на выходе. Однозначность не выполняется для функций, имеющих типы {x—^y} и {x—^t}, teT; x,yeX.
Определение. Функция называется полиморфной, если входной либо выходной кортеж хотя бы одного из ее типовых элементов содержит типовую переменную или если тип функции содержит более одного типового элемента.
Функция выбора i-го элемента из кортежа I(m, i) полиморфна и имеет тип {x1xx2x.xx,x. xxm—x,}. Базисная функция сложения add полиморфна, поскольку ее тип состоит более чем из одного типового элемента: {intxint—int, intxreal— —real, realxint—real, realxreal—real}.
Условия правильной типизации функций в FPTL
Пусть функции представлены в форме систем функциональных уравнений вида F,=t,, i= 1, п. Будем исходить из предположения, что типы всех
входящих в правые части т, базисных функций (встроенных функций, функций-конструкторов и деструкторов в определении абстрактных типов данных) заданы.
Введем отношение порядка на множестве типов, положив t<t, Xj<Xj, t<x V teT, V x, x,, x,eX. Отношение порядка на множестве кортежей вводится следующим образом: y1xy2x .xyk>z1xz2x .xzk, если yi>Zi, i= 1, n . Будем считать, что тип функции //"• "> меньше или равен типу функции /2("'"> (type(f1)<type(f2)), если выполнено условие: для каждого типового элемента у\' х.. .хyyJ ^ -— Zj(,) х z^' х.. .х z^, принадлежащего typef), существует типовой элемент у\' х у\' ...х y'J — ^ zf' х zf х.. .х z'N'', принадлежащий typef2), такой, что y(i) Xу? X...X< yf ху<° X Xym и z1(i) X zf X... X z® < zf X z'2) X ... X z'®. Положим, что typef)=typef2 )&typeft )<type(f2)Aiype(fl )> >type(f2).
Пусть a=yi xy2x . xym^zi xz2x . xz„ - типовой элемент. Тогда типовой кортеж y1 xy2x ... xym называется входным и обозначается IN(a). Аналогично кортеж z1xz2x...xz„ называется выходным и обозначается OUT(a).
Для типа функции p={y1(i)xy2(i)x...xym(i)^z1(i)x xz2(i)x... xz„(i), i=1, 2, ..., k} определим IN(P) и OUT(P) как объединение множеств входных и выходных кортежей каждого типового элемента соответственно: IN(P)={IN(a) |aep}, OUT(P)= ={OUT(a)|aeP}.
Определим в введенных выше обозначениях условия корректной типизации для операций композиции.
1. т=т -т2 - терм правильно типизирован, если
IN(type(T2))>OUT(type(TX)y,
type(Ti-T2)={(a^P) \a&IN(type(xl))Ae&OUT(type(x1)) Эр! 3p2 (p1<p2A(a^p1)etype(T1)A(p2^p)etype(T2)}.
К условию правильной типизации для функций, полученных путем операции последовательной композиции, добавляется дополнительное условие, связанное с использованием функций выбора аргумента I(m, i), m>0, i>0. Как было указано, эти функции полиморфны и выполняют тождественное преобразование между данными на i-м входе и выходе функции. Для функции f, имеющей только константные типы, тип функции I(m, i)f не является однозначным.
Поэтому особым образом определяется тип функций I(m, /)-т:
type(I(m, i)-T)={x1xx2x...xxm^-OUT(T) |x,eIN(T)}.
2. т=т1 *т2 - терм правильно типизирован, если
IN(type(Ti))=IN(type(T2));
type(T1*T2)={(a^P) | aeIN(type(T1))A3P1 ЭР2
(a^P1 e type (T1 )A(a^2 e type(T2))AP=P1 xP2)}.
3. т=т1+т2 - терм правильно типизирован, если type(%l)=type(%2).
4. X] —^т2 - терм правильно типизирован, если
1Щуре{т1))=Мфре{т2));
/;уре(т1—-т2)={(а—Р) | (а—Р)е(уе(т2))ла—• —Ьоо1е(уре(т1)}.
Определим условия строгой правильной типизации функций, заданных в виде системы функциональных уравнений. Эти условия вытекают из правил типизации функций, полученных применением операций композиции функций, и необходимости равенства типов функциональных переменных Fi и термов т„ /= 1, п .
Добавим к условиям правильной типизации, определенным выше для операций композиции, еще два условия правильной типизации для функциональных уравнений:
- типы термов т, и определяемых ими функциональных переменных Fi должны совпадать;
- все вхождения функциональной переменной Fi в функциональные уравнения должны иметь одинаковый тип.
Заметим, что функция может быть неправильно типизирована (в определенном выше смысле), однако при ее применении к данным конкретного типа она вычислит результат.
Перейдем к рассмотрению алгоритма типового контроля в FPTL.
Алгоритм типового контроля в FPTL
По-прежнему исходим из общего задания функций в FPTL в форме системы функциональных уравнений Fi=тi, 1= 1, п . Считаем, что типы всех базисных функций заданы.
Перед проведением типового контроля производится ряд преобразований исходной системы функциональных уравнений.
Сначала для каждой функциональной переменной Fi выполним для всех входящих в т, функциональных переменных подстановки т/ вместо каждого вхождения Fj в т,. Строим
=[т, /^|у = М,/ ФI]т,., /=1ТП .
Повторяем процесс, строя путем подстановок вместо всех новых функциональных переменных, входящих в ) (переменных, которые
появились в р> в результате осуществленных на первом шаге подстановок), соответствующих им термов.
Продолжаем этот процесс, пока для каждого Fi не будут получены ', такие, что в ' не существует вхождений функциональных переменных, для которых ранее не выполнялась подстановка.
Полученную систему функциональных уравнений ¥,= F¡i•k'', /-1, п , назовем приведенной. Полученная система функциональных уравнений эквивалентна исходной системе.
Затем представим каждый терм ' в эквивалентной форме: ' =х(,) +х'2,) + ... + хП'), где термы т® не содержат вхождений операции +. Для
этого используются следующие правила эквивалентности:
Л-(В+С)=(Л-В)+(Л-С), (А+В)-С=(А-С)+(В-С), Л*(В+С)=(Л*В)+(Л*С), (Л+В)*С=(Л*С)+(В*С), Л^(В+С)=(Л ^В)+(Л ^С), (Л+В)^С=(Л^С)+(В^С). Определение. Систему функциональных уравнений будем считать семантически корректной, если в представлении, полученном выше, в правой
части для каждого терма ' существует по
крайней мере один +-терм т®, не содержащий
вхождений функциональных переменных.
Требование семантической корректности на практике не является сильно ограничивающим, поскольку программы, которые не обладают этим свойством, могут выполняться бесконечно, не приводя к получению результата. Очевидно, что в случае, если программа семантически корректна, процесс подстановки, описанный выше, не может длиться бесконечно.
В дальнейшем будем рассматривать только семантически корректные системы функциональных уравнений.
После построения приведенной системы функциональных уравнений для семантически корректной программы в каждом уравнении ¥,=т„ /-1, п, имеется хотя бы один терм т, не содержащий функциональные переменные. Если в процессе построения приведенной системы уравнений получилось более одного слагаемого, не содержащего функциональные переменные, проверяем выполнение условий правильной типизации для операции +, то есть равенство типов слагаемых.
Если это условие не выполняется, считается, что типизация не является правильной. Если оно выполняется в каждом уравнении приведенной системы уравнений, полагаем тип ¥,, /=1, п, равным типу одного из его слагаемых и для каждого вхождения функциональной переменной ¥, j= 1, п, в соответствующее слагаемое ' определяем его тип. Если после этого все слагаемые в Fí:^k'',
/-1, п, имеют тип, равный типу ^, считаем, что типизация рассматриваемой системы функциональных уравнений правильна. В противном слу-
чае считается, что правильность типизации нарушена.
В реализации языка FPTL на многоядерных компьютерах вместо двух операций композиции функций ^ и + используется более эффективная с точки зрения параллельного выполнения программ тернарная операция, эквивалентная условному оператору в последовательных языках программирования. Эта операция записывается в форме т\^т2, т3, где т\ - терм-условие. Легко показать, что она эквивалентна представлению (т\^т2)+(т7^т3). Поэтому перед применением алгоритма контроля типовой правильности функциональной программы все вхождения в нее термов, полученных путем применения тернарной операции, приводятся к данной эквивалентной форме.
Покажем работу алгоритма типового контроля на примере следующей программы:
Ъ = / ■ Ъ + /г ■ Ъ + /,
Ъ = / • Ъ • / + Ъ • /.
Пусть
type(f\)={int^■int}, type(/2)={mt^■real},
type(f3)={int^■int}, type(f'4)={int^■real},
type(f5)={real^■int}, type(f6)={Ш^■real}.
Производим подстановку вместо ¥2 терма т2 в правой части уравнения ¥\=/\¥\+/г¥2+/3. Поручаем =/\¥\+/тУ4¥гЛ+¥\-/6)+Л=/\¥+ги¥?Л+
+/2-¥\-/6+/3. Все подстановки для ¥\ сделаны, заме-
(1)
тим, что в Ь{ имеется терм, не содержащий функциональные переменные. Получаем, что type(¥\)=type(f3)={int^■int}.
В правой части уравнения ¥2=/4 ■ ¥2 /5+¥\ /6 нет терма, не содержащего функциональные переменные. Выполняем подстановки для ¥2: =/ ¥2 /5+
+(f\ ¥+2 ■ ¥2+/з) /6=/4 ■ ¥2 /5+/ ¥\ /+2 ■ ¥2 /+/з / В результате подстановки появился терм /3 /6, не содержащий функциональные переменные. В соответствии с правилами определения типов для операции композиции получаем, что type(f3 /6)= ={Ш^геа1}. Из этого следует, что тип функции ¥2 равен {М^геа1}.
Проверяем выполнение условий типовой кор-
гЧ1) т-(1)
ректности для всех слагаемых в ' и Ь2 , содержащих вхождения функциональных переменных, считая их тип определенным ранее: type(f\ ■¥\)={int^int}. При проверке типовой корректности выражения /2 ■/4 ¥2 /5 встречаем ошибку типизации - тип функции/2 равен {т^геа1}, как и тип функции/4 равен {Ш^геа1}. Делаем вывод о неверной типизации.
Пример можно изменить так, чтобы получить верную типизацию. Если все типы функций /={М^М}, /е{1, 2, 3, 4, 5, 6}, то type(¥\)= ={ Ш^Ш}, ^е(¥2)={ Ш^Ш}.
В заключение следует отметить, что по предварительным экспериментальным данным разработанные метод и алгоритм статического типового контроля правильности программы позволяют уменьшать до 30 % время выполнения программы на многоядерных компьютерах.
Литература
1. McCarthy J. Recursive functions of symbolic expressions and their computation by machine. Cambridge, Mass.: MIT, 1960.
2. Milner R.G. The standard ML core language. Polymorphism. The ML/LCF/Hope Newsletter, 1985, vol. 2, no. 2.
3. Peyton Jones S.L. The implementation of functional programming languages. London: Prentice Hall, 1987.
4. Tim Harris, Simon Marlow, Simon Peyton Jones. Haskell on shared-memory multiprocessor. Proc. in Haskell workshop, 2005.
5. Church A. The calculi of lambda-conversion. Ann. of Math. Studies, Princeton, NJ, Princeton Univ. Press, 1941, vol. 6.
6. Клини С.К. Введение в метаматематику. М.: Иностранная литература, 1957.
7. Кутепов В.П. Исчисление функциональных схем и параллельные алгоритмы // Программирование. 1976. № 6.
8. Кутепов В.П., Фальк В.Н. Функциональные системы: теоретический и практический аспекты // Кибернетика. 1979. № 1.
9. Кутепов В.П., Фальк В.Н. Направленные отношения: теория и приложения // Изв. РАН: Техн. кибернетика. 1994. № 4, 5.
10. Бажанов C.E., Кутепов В.П., Шестаков Д.А. Структурный анализ и планирование процессов параллельного выполнения функциональных программ // Изв. РАН. ТиСУ, 2005. № 6; Structural analysis and planning of processes of parallel execution of functional programs, Journ. of CSSI, 2005, vol. 44, no. 6.
11. Бажанов C.E., Кутепов В.П., Шестаков Д.А. Язык функционального параллельного программирования и его реализация на кластерных системах // Программирование. 2005. № 5.
12. Кутепов В.П., Шамаль П.Н. Реализация языка функционального параллельного программирования FPTL на многоядерных вычислительных системах // Параллельные вычис-
ления и задачи управления: сб. докл. VI Междунар. конф. М., 2012.
13. The Glasgow Haskell Compiler. URL: http://www.haskell. org/ghc (дата обращения: 14.02.2014).
References
1. McCarthy J. Recursive functions of symbolic expressions and their computation by machine. Cambridge, Mass.: MIT Press, 1960.
2. Milner R.G. The standard ML core language. Polymorphism. The ML/LCF/Hope Newsletter. 1985, vol. 2, no. 2.
3. Peyton Jones S.L. The implementation of functional programming languages. London, Prentice Hall Publ., 1987.
4. Harris T., Marlow S., Jones S.P. Haskell on shared-memory multiprocessor. Proc. in Haskell workshop. 2005.
5. Church A. The calculi of lambda-conversion. Ann. of Math. Studies. Princeton, N.J., Princeton University Press, 1941, vol. 6.
6. Kleene S.C. Introduction to metamathematics. Amsterdam, Nord-Holland Publ. Co, 1952, 550 p. (Russ. ed.: Cleene C.K. Moscow, Inostrannaya literature Publ., 1957).
7. Kutepov V.P. Functional schemes computing and parallel algorithms. Programmirovanie [Programming]. 1976, no. 6.
8. Kutepov V.P., Falk V.N. Functional systems: theoretical and practical aspects. Kibernetika [Cubernetics]. 1979, no. 1.
9. Kutepov V.P., Falk V.N. Directed relations: theory and applications. Izv. RAN. Tekhn. kibernetika [Journ. of Computer and Systems Sciences International]. 1994, no. 4, 5.
10. Bazhanov C.E., Kutepov V.P., Shestakov D.A. Structural analysis and planning of processes of parallel execution of functional programs. Journ. of Computer and Systems Sciences International. 2005, vol. 44, no. 6, pp. 942-957.
11. Bazhanov C.E., Kutepov V.P., Shestakov D.A. Functional parallel programming language and its implementation on cluster systems. Programmirovanie [Programming]. 2005, no. 5, pp. 18-51.
12. Kutepov V.P, Shamal P.N. Multicore implementation of FPTL functional parallel programming language. Materialy konf. "Parallelnye vychisleniya i zadachi upravleniya " [Proc. VI Intern. Conf. "Parallel Computing and Control Problems"]. Moscow, 2012.
13. The Glasgow Haskell Compiler. Available at: http://www.haskell.org/ghc (accessed February 14, 2014).