gpt4 book ai didi

prolog - 简单的 prolog 程序过早返回 false

转载 作者:行者123 更新时间:2023-12-02 04:43:32 25 4
gpt4 key购买 nike

我已经有一段时间没有使用 Prolog 编程了。今天,我试着做了一个简单的程序。它列出了属于同一家庭的一些事实。如果两个人属于一个家庭,则不能互赠礼物。我想要获得允许某人赠送礼物的所有人(或至少一个人)。

family(john, jack).
family(matt, ann).
family(ann, jack).
family(jordan, michael).
family(michael, liz).

sameFamily(X, Y) :-
family(X, Y).
sameFamily(X, X) :-
false.
sameFamilySym(X, Y) :-
sameFamily(X, Y).
sameFamilySym(X, Y) :-
sameFamily(Y, X).
sameFamilyTrans(X, Z) :-
sameFamilySym(X, Y),
sameFamilySym(Y, Z).

gift(X, Y) :-
not(sameFamilyTrans(X, Y)).

有些查询如果 sameFamilyTrans/2 返回 false 而实际上它们应该返回 true

sameFamilyTrans/2 显然是错误的。我想我需要保留一份中间传递性列表。像这样:

sameFamilyTrans(X, Z, [Y|Ys]) :-
sameFamilySym(X, Y, []),
sameFamilyTrans(Y, Z, Ys).

但是我不知道怎么调用它。

附言我正在使用 SWI-Prolog,如果这有什么不同的话。

最佳答案

是的,您走在了正确的轨道上。诀窍是用一个空的累加器调用传递闭包,并在每一步中检查是否找到一个循环(即,我们之前是否见过这个家庭成员。正如“false”所指出的,这些人需要不过,在进入 not 之前已经实例化了。

总而言之,这是可行的:

family(john, jack).
family(matt, ann).
family(ann, jack).
family(jordan, michael).
family(michael, liz).

sameFamily(X, Y) :-
family(X, Y).
sameFamilySym(X, Y) :-
sameFamily(X, Y).
sameFamilySym(X, Y) :-
sameFamily(Y, X).

sameFamilyTrans(X, Y, Acc) :-
sameFamilySym(X, Y),
not(member(Y,Acc)).

sameFamilyTrans(X, Z, Acc) :-
sameFamilySym(X, Y),
not(member(Y,Acc)),
sameFamilyTrans(Y, Z, [X|Acc]).

person(X) :- family(X, _).
person(X) :- family(_, X).

gift(X, Y) :-
person(X),
person(Y),
X \= Y,
not(sameFamilyTrans(X, Y, [])).

背景知识:传递闭包实际上不是一阶可定义的(参见 https://en.wikipedia.org/wiki/Transitive_closure#In_logic_and_computational_complexity )。所以可以预料这会有点棘手。

关于prolog - 简单的 prolog 程序过早返回 false,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20317450/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com