Лекции.Орг


Поиск:




Категории:

Астрономия
Биология
География
Другие языки
Интернет
Информатика
История
Культура
Литература
Логика
Математика
Медицина
Механика
Охрана труда
Педагогика
Политика
Право
Психология
Религия
Риторика
Социология
Спорт
Строительство
Технология
Транспорт
Физика
Философия
Финансы
Химия
Экология
Экономика
Электроника

 

 

 

 


Правило декларирования доменов для составных объектов




Пример 2. (слайд 8)

Рассмотрим пример, в котором в качестве базы знаний выступает информация о материальной собственности одного или нескольких лиц. Причем, материальная собственность может быть выражена обладанием денежными средствами, раритетами, квартирами, ювелирными ценностями и т.д. В частности, в программе могут встретиться следующие отношения:

vladeet(vladimir, schet (123456789, 1000000)).

vladeet(vladimir, kvartira("RF","Tula", "Pervomayskaya, 8- 814").

vladeet(vladimir, kvartira("Belarus","Minsk", "Pobedy, 13 - 21").

vladeet(vladimir, kvartira("US","New York", "17 Street, 9- 11").

vladeet(vladimir, boat).

vladeet(vladimir, dog("Lord")).

В том случае, если мы даем запрос в виде: vladeet(vladimir, X), переменная Х может оказаться связанной с различными объектами: счет, квартира, яхта, собака и т.д. Кроме того, мы не можем однозначно описать предикат vladeet, поскольку описание vladeet(symbol, symbol) не будет всегда верно. Вместо этого можно дать иное определение этого предиката: vladeet (name, sobstvennost), а домен sobstvennost в разделе DOMAINS можно описать так:

DOMAINS

sobstvennost =

schet(nomer, summa);

kvartira(strana, gorod, ulitsa_dom);

boat;

dog(name)

strana, gorod, ulitsa_dom, name = symbol

nomer, summa = unsigned

При этом boat – функтор без присоединенных к нему аргументов, а разделитель "; " обозначает логическую связку " или ".

DOMAINS

sobstvennost =

schet(nomer, summa);

kvartira(strana, gorod, ulitsa_dom);

boat;

dog (name)

strana, gorod, ulitsa_dom, name = symbol

nomer = unsigned

summa = real

PREDICATES

vladeet(name, sobstvennost)

CLAUSES

vladeet(vladimir,schet(123456789, 1000000)).

vladeet(vladimir,kvartira("RF","Tula","Pervomayskaya, 8- 814")).

vladeet(vladimir,kvartira("Belarus","Minsk","Pobedy, 13 - 21")).

vladeet(vladimir,kvartira("US","New York", "17 Street, 9- 11")).

vladeet(vladimir,boat).

vladeet(vladimir,dog("Lord")).

GOAL

vladeet(vladimir, X).

При этом выполнение будет иметь вид:

X = schet(123456789,1000000)

X = kvartira("RF","Tula","Pervomayskaya, 8- 814")

X = kvartira("Belarus","Minsk","Pobedy, 13 - 21")

X = kvartira("US","New York","17 Street, 9- 11")

X = boat

X = dog("Lord")

6 Solutions

 

После обсуждения примера мы можем приступить к формулировке правила объявления доменов для составных объектов в общем виде.

В разделе domains необходимо дать описание нового домена в виде:

nazvanie_domena = alternative1(D1 Name, D2 Name, …);

alternative2(D3 Name, D4 Name, …);

alternative# – допустимые, но различные функторы;

D#– стандартные, или объявленные в программе домены;

Name комментарии.

Альтернативы разделяются точкой с запятой, при этом каждая альтернатива состоит из функтора и списка доменов соответствующих аргументов.

Если функтор не имеет аргументов, то можно записать в программе alternativeN или alternativeN().

Повтор и рекурсия

В языке Пролог не существует прямого способа выражения повтора. Здесь возможны только два вида повторения – откат, с помощью которого осуществляется поиск многих решений в одном запросе, и рекурсия, в которой процедура вызывает сама себя.

Если мы рассмотрим пример 1 с позиций осуществления повтора, то очевидно, что поиску всего множества решений способствует предикат fail, инициализирующий неуспех и очередной откат на поиск решений.

Рекурсивная процедура – это процедура, которая вызывает сама себя. В рекурсивной процедуре нет проблемы запоминания результатов ее выполнения, потому что любые вычисленные значения можно передавать из одного вызова в другой как аргументы рекурсивно вызываемого предиката. Всякий раз, когда одна процедура вызывает другую, информация о выполнении вызывающей процедуры должна быть сохранена для того, чтобы она (вызывающая процедура), могла после выполнения вызванной процедуры, возобновить выполнение на том же месте, где остановилась. Это означает, что если процедура вызывает себя 100 раз, то 100 различных состояний должно быть записано одновременно. Стек на платформе DOS ограничен 64 Kb, на платформе Windows размеры стека не ограничены, но есть ряд других существенных проблем.

В ряде специальных случаев процедура может вызывать себя без сохранения информации о своем состоянии. Допустим, что процедура A вызывает процедуру B, а B – C в качестве своего последнего шага. Когда B вызывает C, B не должна больше ничего делать. Поэтому, вместо того, чтобы сохранить в стеке информацию о текущем состоянии C, можно переписать старую сохраненную информацию о состоянии B на текущую информацию о состоянии C. Когда C закончит выполнение, она будет считать, что она вызвана непосредственно процедурой A. С процедурной точки зрения данный процесс очень похож на всего лишь обновление управляющих переменных в цикле. Эта операция называется оптимизацией хвостовой рекурсии.

Пример 3. (слайд 9)

Составить программу, в которой вычисляется число Фибоначчи по его порядковому номеру.

DOMAINS

arg= integer

rez = unsigned

 

PREDICATES

fibonachchi (arg, rez)

 

clauses

fibonachchi(0,1).

fibonachchi(1,1).

fibonachchi(N,X):-

PR=N-1, PRR=N-2,

fibonachchi(PR,FPR),

fibonachchi(PRR,FPRR),

X=FPR+FPRR.

GOAL

write("Vvedite nomer chisla \t"),

readint(NOMER),

fibonachchi(NOMER, CHISLO),

writef("\n Nomer %u Chislo = %u",NOMER, CHISLO),

readchar(_).

Данный пример является яркой иллюстрацией обычной рекурсии. n

Пример 4 (слайд 10).

Составить программу вычисления наибольшего общего делителя двух натуральных чисел.

PREDICATES

nod(unsigned,unsigned,unsigned)

CLAUSES

nod(X,0,R):-R=X.

nod(X,Y,R):-

T=X mod Y,

nod(Y,T, R).

 

GOAL

nod(3345768,677424, D), write("nod=",D), readchar(_).

Этот пример служит демонстрацией хвостовой рекурсии. Обратите внимание на то, что рекурсивное обращение к nod(Y,T,R)является последней подцелью предложения nod(X,Y,R), а выше в этом предложении не было точек возврата. Эти два условия являются основными для распознавания хвостовой рекурсии. Пример 3 не является примером хвостовой рекурсии, поскольку в нем нарушены оба условия.

Выводы.

Оптимизация хвостовой рекурсии нарушается, если:

1. Рекурсивный вызов не является самым последним шагом процедуры;

2. Имеются в наличии непроверенными к моменту выполнения рекурсивного вызова некоторые возможные альтернативы;

3. К моменту выполнения рекурсивного вызова непроверенная альтернатива имеется в любом вызываемом предикате.





Поделиться с друзьями:


Дата добавления: 2017-03-18; Мы поможем в написании ваших работ!; просмотров: 261 | Нарушение авторских прав


Поиск на сайте:

Лучшие изречения:

Два самых важных дня в твоей жизни: день, когда ты появился на свет, и день, когда понял, зачем. © Марк Твен
==> читать все изречения...

2290 - | 2115 -


© 2015-2025 lektsii.org - Контакты - Последнее добавление

Ген: 0.011 с.