prolog - 如何在Prolog中返回推荐列表?

标签 prolog prolog-findall

对于我的作业,我应该列出 20 种潜在的宠物,然后定义有关每种宠物的事实。然后我需要问潜在的宠物主人五个问题,这将有助于决定哪些宠物是好的推荐。我试图根据用户输入返回宠物列表,但它每次都返回 true,并且实际上并没有列出推荐的宠物。不知道我哪里出错了。我只会在我的代码示例中包含一些宠物,因此它不会很长。

pet_advisor.pl:

pet(cat).
pet(chameleon).
pet(chicken).
pet(chinchilla).
pet(cow).

size(cat, small).
sleeps(cat, night).
stays(cat, indoor).
stays(cat, outdoor).
class(cat, mammal).
live(cat, 12)

size(chameleon, small).
sleeps(chameleon, night).
stays(chameleon, indoor).
class(chameleon, reptile).
live(chameleon,5).

size(chicken, small).
sleeps(chicken, night).
stays(chicken, outdoor).
class(chicken, bird).
live(chicken,10).

size(chinchilla, small).
sleeps(chinchilla, day).
stays(chinchilla, indoor).
class(chinchilla, mammal).
live(chinchilla,15).

size(cow, large).
sleeps(cow, night).
stays(cow, outdoor).
class(cow, mammal).
live(cow,22).

pet_size_ok(X) :- pet_size(X), size(Y, X).
sleep_type_ok(X) :- sleep_type(X), sleeps(Y, X).
pet_location_ok(X) :- pet_location(X), stays(Y, X).
kind_ok(X) :- kind(X), class(Y, X).
life_ok(X) :- life(X), live(Y, Z), Z =< X.

which_pet(X) :- pet_size_ok(X), sleep_type_ok(X), pet_location_ok(X), kind_ok(X), life_ok(X).

recommend :- write('Do you want a small, medium, or large sized pet? '), read(Size), nl, assert(pet_size(Size)),
             write('Do you want a pet that sleeps during the day or night? '), read(Sleep), nl, assert(sleep_type(Sleep)),
             write('Do you want an indoor or outdoor pet? '), read(Place), nl, assert(pet_location(Place)),
             write('Do you want a reptile, mammal, bird, or a fish? '), read(Type), nl, assert(kind(Type)),
             write('How long do you want your pet to live (years)? '), read(Age), nl, assert(life(Age)),
             findall(Pets, which_pet(Pets), Suggestions),
             write('I would recommend these pets for you: '), nl, writelist(Suggestions),
             retract(pet_size(Size)), retract(sleep_type(Sleep)), 
             retract(pet_location(Place)),
             retract(kind(Type)), retract(life(Age)).

writelist([]).
writelist([H|T]) :- writeonce(H,T), writelist(T).  
writeonce(H,T) :- member(H,T).
writeonce(H,T) :- not(member(H,T)), write(H), nl.

所以如果我要回答以下问题: 小的 夜晚 室内的 哺乳动物 15

它应该返回一个包含 [cat, chinchilla] 的列表,但它返回的都是 true。

最佳答案

您的代码存在几个问题。首先,对于大多数 Prolog 系统和 Prolog 标准,必须声明不连续谓词。在文件开头添加以下指令:

:- discontiguous([
    size/2, sleeps/2, stays/2, class/2, live/2
]).

接下来,在查询推荐时无需使用动态谓词以及断言和撤回事实:

which_pet(Size, Sleep, Place, Type, Age, Pet) :-
    size(Pet, Size),
    sleeps(Pet, Sleep),
    stays(Pet, Place),
    class(Pet, Type),
    live(Pet, Age0), Age0 =< Age.

recommend :-
    write('Do you want a small, medium, or large sized pet? '), read(Size), nl,
    write('Do you want a pet that sleeps during the day or night? '), read(Sleep),
    write('Do you want an indoor or outdoor pet? '), read(Place), nl,
    write('Do you want a reptile, mammal, bird, or a fish? '), read(Type), nl,
    write('How long do you want your pet to live (years)? '), read(Age), nl,
    findall(Pet, which_pet(Size,Sleep,Place,Type,Age,Pet), Suggestions),
    write('I would recommend these pets for you: '), nl, writelist(Suggestions).

这不是理想的重写,因为它的扩展性很差,但比使用动态谓词要好得多。

作为结束语,用于打印结果的代码会执行两项更好分开的任务:(1) 过滤重复项和 (2) 打印唯一结果。我建议您将这些任务分开。可以对结果进行过滤,例如使用 setof/2 而不是 findall/3 或在由 findall/3< 构造的列表上调用 sort/2/ 调用。我把重写留给你。还可以使用标准否定控制构造 \+/1,而不是旧版/已弃用的 not/1 谓词。

调用示例:

| ?- recommend.
Do you want a small, medium, or large sized pet? small.
Do you want a pet that sleeps during the day or night? night.
Do you want an indoor or outdoor pet? outdoor.
Do you want a reptile, mammal, bird, or a fish? bird.

How long do you want your pet to live (years)? 20.

I would recommend these pets for you: 
chicken

yes

关于prolog - 如何在Prolog中返回推荐列表?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55068525/

相关文章:

Prolog findall/3 : more than one bag

prolog - 找出谓词的所有解

random - Prolog中的随机项目

prolog - 在 Prolog 中插入剪切以使关系子句绑定(bind)但双向

list - 我如何跟踪 Prolog 中的值?

prolog - 处理未绑定(bind)的变量和列表

Prolog 在限定时间内找到所有解决方案

prolog - 如何在 findall/3 元谓词中使用 and

prolog - 有没有比制定新规则来更改 maplist 的变量顺序更好的方法?