Синтаксис и семантика языка программирования Пролог представляют собой набор rules, которые определяют, как пишется программа на Прологе и как она интерпретируется. Правила изложены в стандарте ISO ISO / IEC 13211, хотя есть различия в реализациях Prolog.
Prolog - это динамически типизированный. Он имеет единственный тип данных, термин, который имеет несколько подтипов: атомы, числа, переменные и составные термины.
атом - это имя общего назначения, не имеющее внутреннего значения. Он состоит из последовательности символов, которая анализируется средством чтения Prolog как единое целое. В коде Пролога атомы обычно представляют собой голые слова, написанные без специального синтаксиса. Однако атомы, содержащие пробелы или некоторые другие специальные символы, должны быть заключены в одинарные кавычки. Атомы, начинающиеся с заглавной буквы, также должны быть заключены в кавычки, чтобы отличать их от переменных. Пустой список, записанный , также является атомом. Другие примеры атомов включают
x
, синий
, 'Taco'
и 'некоторый атом'
.
. Числа могут быть с плавающей точкой или <30.>целые числа. Многие реализации Prolog также предоставляют неограниченные целые числа и рациональные числа.
Переменные обозначаются строкой, состоящей из букв, цифр и символов подчеркивания и начинающейся с заглавной буквы или символа подчеркивания. Переменные очень похожи на переменные по логике в том смысле, что они заменяют произвольные термины. Переменная может быть создана (привязана к определенному члену) с помощью унификации. Одиночный знак подчеркивания (_
) обозначает анонимную переменную и означает «любой термин». В отличие от других переменных, подчеркивание не означает одно и то же значение везде, где оно встречается в определении предиката.
A Составной терм состоит из атома, называемого «функтором», и ряда «аргументов», которые снова являются терминами. Составные термины обычно записываются как функтор, за которым следует список терминов аргументов, разделенных запятыми, который содержится в круглых скобках. Количество аргументов называется арностью терма. Атом можно рассматривать как составной термин с арностью ноль.
Примеры составных терминов: truck_year ('Mazda', 1986)
и 'Person_Friends' (zelda, [tom, jim])
. Составные термины с функторами, объявленные как операторы, могут быть записаны в префиксной или инфиксной нотации. Например, термины - (z)
, + (a, b)
и = (X, Y)
также могут быть записаны как -z
, a + b
и X = Y
соответственно. Пользователи могут объявлять произвольные функторы как операторы с разным приоритетом, что позволяет использовать нотации, специфичные для предметной области. Обозначение f / n обычно используется для обозначения термина с функтором f и арностью n.
Особые случаи составных терминов:
- это список. Составной терм с функтором .
(точка) и арностью 2, второй аргумент которого является списком, сам является списком. Для обозначения списков существует специальный синтаксис: . (A, B)
эквивалентен [A | B]
. Например, список . (1,. (2,. (3,)))
также можно записать как [1 | [2 | [3 | ]]]
, или даже более компактно, как [1,2,3]
.
Программы Prolog описывают отношения, определенные с помощью предложений. Чистый Пролог ограничен предложениями Хорна, полным по Тьюрингу подмножеством логики предикатов первого порядка. Есть два типа статей: факты и правила. Правило имеет вид
Голова: - Тело.
и читается как «Голова истинна, если тело истинно». Тело правила состоит из вызовов предикатов, которые называются целями правила. Встроенный предикат , / 2
(означает оператор 2-арности с именем ,
) обозначает соединение целей, а ; / 2
обозначает дизъюнкцию. Соединения и разъединения могут появляться только в теле, а не в голове правила.
Пункты с пустыми телами называются фактами . Пример факта:
кот (том).
, что эквивалентно правилу:
cat (tom): - true.
другой пример:
X равно 3 + 2.
и когда вы его запустите, результат будет
X = 5 Да.
. Встроенный предикат истина / 0
всегда истинен.
Выполнение программы на Прологе инициируется публикацией пользователем единственной цели, называемой запросом. Логично, что механизм Prolog пытается найти разрешение опровержения отрицательного запроса. Метод разрешения, используемый Prolog, называется разрешением SLD. Если отрицательный запрос может быть опровергнут, из этого следует, что запрос с соответствующими привязками переменных является логическим следствием программы. В этом случае пользователю сообщается обо всех сгенерированных привязках переменных, и считается, что запрос выполнен успешно. С функциональной точки зрения стратегию выполнения Пролога можно рассматривать как обобщение вызовов функций на других языках, с одним отличием в том, что несколько заголовков предложений могут соответствовать заданному вызову. В этом случае система создает точку выбора, объединяет цель с заголовком предложения первой альтернативы и продолжает цели этой первой альтернативы. Если какая-либо цель терпит неудачу в ходе выполнения программы, все привязки переменных, которые были выполнены с момента создания самой последней точки выбора, отменяются, и выполнение продолжается со следующей альтернативой этой точки выбора. Эта стратегия выполнения называется хронологическим отслеживанием с возвратом. Например:
мать-дитя (труде, салли). отец_ ребенок (том, салли). отец_ ребенок (том, эрика). Father_child (Майк, Том). брат (X, Y): - parent_child (Z, X), parent_child (Z, Y). parent_child (X, Y): - Father_child (X, Y). parent_child (X, Y): - mother_child (X, Y).
В результате следующий запрос оценивается как истинный:
? - sibling (sally, erica). Да
Это получается следующим образом: Первоначально единственным совпадающим заголовком предложения для запроса sibling (sally, erica)
является первый, поэтому доказательство запроса эквивалентно доказательству тела этого запроса. предложение с соответствующими привязками переменных, т. е. конъюнкция (parent_child (Z, sally), parent_child (Z, erica))
. Следующая цель, которую нужно доказать, - это крайняя левая цель этого конъюнкции, то есть parent_child (Z, sally)
. Этой цели соответствуют две главы статьи. Система создает точку выбора и пробует первую альтернативу, тело которой - Father_child (Z, sally)
. Эта цель может быть доказана с помощью факта Father_child (tom, sally)
, поэтому создается привязка Z = tom
, и следующая цель, которую необходимо доказать, - это вторая часть приведенного выше соединение: parent_child (tom, erica)
. Опять же, это подтверждается соответствующим фактом. Поскольку все цели могут быть доказаны, запрос выполняется. Поскольку запрос не содержит переменных, пользователю не сообщается о привязках. Запрос с переменными, например:
? - Father_child (Отец, Дитя).
перечисляет все действительные ответы при возврате.
Обратите внимание, что с указанным выше кодом запрос ? - sibling (sally, sally).
также выполняется успешно. При желании можно было бы вставить дополнительные цели для описания соответствующих ограничений.
Итерационные алгоритмы могут быть реализованы с помощью рекурсивных предикатов. Системы Prolog обычно реализуют хорошо известный метод оптимизации, называемый хвостовой вызов оптимизация (TCO) для детерминированных предикатов, демонстрирующих хвостовую рекурсию или, в более общем плане, хвостовые вызовы: кадр стека предложения отбрасывается перед выполнение вызова в хвостовой позиции. Следовательно, детерминированные хвостовые рекурсивные предикаты выполняются с постоянным пространством стека, как циклы в других языках.
A cut (!
) внутри правила не позволит Prolog отследить любые предикаты за вырезом:
predicate (X): - one (X),!, Two (X).
завершится ошибкой, если первое найденное значение X
, для которого один (X)
истинно, приводит к тому, что два (X)
ложно.
Анонимные переменные _
никогда не привязаны к значению и могут использоваться несколько раз в предикате.
Например, поиск в списке по заданному значению:
содержит (V,): - false. содержит (V, [V | _]): - истина. содержит (V, [_ | T]): - содержит (V, T).
Встроенный предикат Prolog \ + / 1
обеспечивает отрицание как сбой, что позволяет немонотонно рассуждать. Цель \ + нелегально (X)
в правиле
допустимо (X): - \ + незаконно (X).
оценивается следующим образом: Пролог пытается доказать, что недопустимый (X)
. Если доказательство для этой цели может быть найдено, исходная цель (т. Е. \ + незаконный (X)
) терпит неудачу. Если не найти доказательств, первоначальная цель достигнута. Поэтому префиксный оператор \ + / 1
называется оператором «недоказуемости», поскольку запрос ? - \ + Goal.
выполняется успешно, если цель недоказуема. Этот вид отрицания является звуком, если его аргумент "земля" (т.е. не содержит переменных). Разумность теряется, если аргумент содержит переменные. В частности, запрос ? - legal (X).
теперь нельзя использовать для перечисления всех допустимых вещей.
При декларативном чтении порядок правил и целей внутри правил не имеет значения, поскольку логическая дизъюнкция и конъюнкция коммутативны. Однако с процедурной точки зрения часто важно учитывать стратегию выполнения Пролога либо по причинам эффективности, либо из-за семантики нечистых встроенных предикатов, для которых важен порядок оценки. Кроме того, поскольку интерпретаторы Пролога пытаются объединить предложения в том порядке, в котором они представлены, отсутствие правильного порядка может привести к бесконечной рекурсии, как в:
predicate1 (X): - predicate2 (X, X). predicate2 (X, Y): - predicate1 (X), X \ = Y.
При таком порядке любой запрос в форме
? - predicate1 (atom).
будет повторяться до тех пор, пока стек не будет исчерпан. Если, однако, последние 3 строки были изменены на:
predicate2 (X, Y): - X \ = Y, predicate1 (X).
тот же запрос приведет к неточному результату за очень короткое время.
Существует специальная нотация, называемая грамматиками с определенными предложениями (DCG ). Правило, определенное через ->/ 2
вместо : - / 2
, расширяется препроцессором (expand_term / 2
, средство, аналогичное макросам в других languages) в соответствии с несколькими простыми правилами переписывания, что приводит к обычным предложениям Пролога. В частности, при переписывании предикат снабжен двумя дополнительными аргументами, которые можно использовать для неявной обработки состояния, аналогично монадам в других языках. DCG часто используются для написания парсеров или генераторов списков, так как они также предоставляют удобный интерфейс для перечисления различий.
Более крупный пример покажет потенциал использования Prolog в синтаксическом анализе.
Учитывая предложение, выраженное в форме Бэкуса – Наура :
:: = :: = | :: = = ; :: = | :: = | :: = a | b :: = 0..9 :: = + | - | *
Это может быть записано на Прологе с использованием групп DCG, соответствующих предсказательному синтаксическому анализатору с упреждающим просмотром одного токена:
предложение (S) ->оператор (S0), предложение_r (S0, S). предложение_r (S, S) ->. предложение_r (S0, seq (S0, S)) ->утверждение (S1), предложение_r (S1, S). оператор (присваивать (Id, E)) ->id (Id), [=], выражение (E), [;]. выражение (E) ->термин (T), выражение_r (T, E). выражение_r (E, E) ->. выражение_r (E0, E) ->[+], термин (T), выражение_r (плюс (E0, T), E). выражение_r (E0, E) ->[-], термин (T), выражение_r (минус (E0, T), E). термин (T) ->фактор (F), term_r (F, T). термин_r (Т, Т) ->. term_r (T0, T) ->[*], factor (F), term_r (раз (T0, F), T). фактор (id (ID)) ->id (ID). фактор (цифра (D)) ->[D], {(число (D); var (D)), между (0, 9, D)}. id (а) ->[а]. id (b) ->[b].
Этот код определяет отношение между предложением (заданным в виде списка токенов) и его абстрактным синтаксическим деревом (AST). Пример запроса:
? - фраза (предложение (AST), [a, =, 1, +, 3, *, b,;, b, =, 0,;]). AST = seq (assign (a, plus (digit (1), times (digit (3), id (b)))), assign (b, digit (0)));
AST представлен с использованием терминов Пролога и может использоваться для применения оптимизаций, для компиляции таких выражений в машинный код или для прямой интерпретации таких операторов. Как типично для реляционной природы предикатов, эти определения могут использоваться как для синтаксического анализа и генерации предложений, так и для проверки того, соответствует ли данное дерево заданному списку токенов. Используя итеративное углубление для точного перечисления, каждое произвольное, но фиксированное предложение и соответствующее ему AST будут в конечном итоге сгенерированы:
? - длина (токены, _), фраза (предложение (AST), токены). Токены = [a, =, a, (;)], AST = assign (a, id (a)); Токены = [a, =, b, (;)], AST = assign (a, id (b)) и т. Д.