Общее правило пролога для поиска кузенов и т. д.

Вопрос в том, чтобы написать общее правило, чтобы найти любой уровень относительности!

cousin(N,Child1,Child2).

Так что это правда, если Child1 и Child2 Nth двоюродные братья. Таким образом, двоюродный брат 1(ребенок 1, ребенок 2) = двоюродный брат (1 ребенок, 1 ребенок 2) и двоюродный брат 2(ребенок 1, ребенок 2) = двоюродный брат (2 ребенка 1, ребенок 2) и т. Д. Для двоюродных братьев третьего, четвертого и даже более высокого уровня.

Что у меня так далеко:

/* first person is parent of second person */
parent(a, b).
parent(b, f).
parent(a, d).
parent(f, g).
parent(a, k).
parent(f, h).
parent(k, l).
parent(f, i).
parent(k, m).
parent(l, t).
parent(b, e).

sibling(X,Y) :- parent(Z,X), parent(Z,Y), not(X=Y).

grandparent(X, Z) :-
    parent(X, Y),
    parent(Y, Z).

greatgrandparent(X, Z) :-
    parent(X, Y),
    parent(Y, P),
    parent(P, Z).

cousin1(Child1,Child2) :-
    parent(Y1,Child1),
    parent(Y2,Child2),
    sibling(Y1,Y2).

cousin2(Child1,Child2) :-
    greatgrandparent(Z, Child1),
    greatgrandparent(Z, Child2),
    \+sibling(Child1, Child2),
    \+cousin1(Child1, Child2),
    Child1 \= Child2.

Это возвращает false независимо от введенных значений, так что я понятия не имею, что я делаю, ПОЖАЛУЙСТА, помогите!

cousin(N,Child1,Child2) :-
    nth0(N, parent(Y1,Child1),Y1),
    nth0(N, parent(Y2,Child2),Y2),
    cousin1(Y1,Y2).

1 ответ

Решение

Я старался:

    % first person is parent of second person
    parent(a, b).
    parent(b, f).
    parent(a, d).
    parent(f, g).
    parent(a, k).
    parent(f, h).
    parent(k, l).
    parent(f, i).
    parent(k, m).
    parent(l, t).
    parent(b, e).

    sibling(Sib1,Sib2) :- parent(SomeParent,Sib1), 
             parent(SomeParent,Sib2), 
             \+ Sib1 = Sib2.

    % first person is ancestor of second person
    ancestor(Older,Younger,L) :-
             parent(Older,Younger),
             L is 1.
    ancestor(Older,Younger,Level) :- 
             parent(Older,Child),
             ancestor(Child,Younger,L),
             Level is L + 1. 


    %nth_cousin(Level,Cous1,Cous2) :- 
    %             ancestor(Sib1,Cous1,Level),
    %             ancestor(Sib2,Cous2,Level), 
    %             sibling(Sib1,Sib2).  

    nth_cousin(Level,Cous1,Cous2) :- 
             setof((Cous1,Cous2), Sib1^Sib2^(ancestor(Sib1,Cous1,Level), 
                                             ancestor(Sib2,Cous2,Level), 
                                             sibling(Sib1,Sib2)
                                            ), 
                   Cousins),
             member((Cous1,Cous2), Cousins),
             \+ (Cous2@<Cous1, member((Cous2,Cous1), Cousins)).  

Ex.

    1 ?- nth_cousin(1,Cous1,Cous2).
    Cous1 = e,
    Cous2 = l ;
    Cous1 = e,
    Cous2 = m ;
    Cous1 = f,
    Cous2 = l ;
    Cous1 = f,
    Cous2 = m ;
    false.

    2 ?- nth_cousin(2,Cous1,Cous2).
    Cous1 = g,
    Cous2 = t ;
    Cous1 = h,
    Cous2 = t ;
    Cous1 = i,
    Cous2 = t ;
    false.
Другие вопросы по тегам