Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Л.К. №10.docx
Скачиваний:
0
Добавлен:
01.04.2025
Размер:
25.03 Кб
Скачать

10.5 Знаходження зразу всіх розв’язків.

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

Але іноді виникає проблема, яку важко вирішити і за допомогою рекурсії. Нехай вам потрібно знайти зразу всі розв’язки задоволення цілі, як частини єдиної складної структури даних. Пролог вирішує дану проблему за допомогою предикату findall, який приймає ціль як один із своїх аргументів, збираючи всі розв.язки цієї цілі в один список.

Предикат findall має три аргументи:

- перший аргумент VarName визначає який з аргументів в зазначеному предикаті потрібно відібрати в список.

- другий, Мypredicate, визначає предикат, з якого будуть збиратись значення.

- третій аргумент, ListParam, з змінною, якою позначається список значень, зібраних через перебір з поверненнями.

Зазначимо, що попередньо повинна бути визначена область, до якої повинні належати значення ListParam.

Наступна програма використовує findall, для визначення середнього віку групи людей.

domains

name, address = string

age = integer

list = age*

predicates

person(name, address, age)

sumlist(list, age, integer)

goal

findall(Age, person(_, _, Age), L),

sumlist(L, Sum, N),

Ave = Sum/N,

write(|Average =|, Ave), nl.

clauses

sumlist([], 0, 0).

sumlist([H|T], Sum, N) :- sumlist(T, S1, N1), Sum=H+S1,

N=1+N1.

person(|Sherlock Holmes|, |22B Baker Street|, 42).

person(|Pete Spiers|, |Apt. 22, 21st Street|, 36).

person(|Mary Darrow|, |Suite 2, Omega Home|, 51).

В цій програмі фраза findall створює список L, який містить числа, які відповідають віку всіх людей з предикату person. Якщо ви бажаєте зробити список людей вік яких 42 роки, тоді ви можете написати ціль

findall(Who, person(Who,_,42), List).

10.6 Складні списки.

Часто вам потрібно зберігати комбінацію різних типів елементів всередині одного списку, як наприклад

[2,3,5,12,[Food,|goo|], |new|]

Складними будемо називати списки, які містять більше одного типу аргументів. Нам будуть потрібні спеціальні декларації, щоб обробити списки багатотипних елементів, тому що Пролог вимагає, щоб всі елементи списку належали одній області. Для опису різнотипних списків потрібно використовувати функтори.

Наступний приклад декларації області для списку, який може мати символ, ціле число, літеру, стрічку або ж список із них. Декларація списку повинна мати функтор, а потім повинна бути декларована рекурсивно. Так, попередній список повинен бути описаний:

domains

llist = l(list). s(symbol). i(integer). c(char). t(string)

list = llist *

І буде задаватись в Пролозі:

[i(2),і(3),i(5),i(12), [c(food),s(«goo»)],s(«new»)]

Наступний приклад з предикатом append показує як використовувати такий опис:

domains

llist = l(list). s(symbol). i(integer). c(char). t(string)

list = llist *

predicates

append(list, list, list)

goal

makewindow(1,7,7, |answer|,15,0,8,80),

/*Note how you can use the same code but need functors *

* append([likes,[bill,mary]],[bill,sue],Ans) */

append([s(likes),l([s(bill),s(mary)])],[s(bill),s(sue)],Ans), */

write(|First List:|, Ans), nl,nl,

/*The trick is to write the list first, than add the functors *

* append([apple,[[[47], .1.]], [[[|This is a string|,b,7, *

* .w.]],bee], [.c.], Ans2) */

*

append([l[s(|This|), s(|is|,s(|a|),s(|list|))]), s(bee)],

[c(.c.)], Ans2),

write(|Second List :|, Ans2), nl.

clauses

/* Concatenate two lists */

append([], L, L).

append([X|L1], L2, [X|L3]) :- append(L1, L2, L3).

Соседние файлы в предмете [НЕСОРТИРОВАННОЕ]