I'm a beginner in Prolog and I am dealing with a problem that might seem stupid to you, but I really can't understand what I'm doing wrong! Ok, I have this file fruits.pl and inside that I have something like this:
fruit(apple,small,sweet).
fruit(lemon,small,nosweet).
fruit(melon,big,sweet).
I have already (inside that file made a coexist(X,Y) atom that checks if two fruits can be put together in a plate. It works fine! But now I can't create a suggest(X) that takes as a parameter a fruit and returns a list of fruits that can be put together in the same plate. The thing is I was trying to make something like that
suggest(X) :- findall(Y,fruit(Y,_,_), List), coexist(X,Y).
What do you think? Every time I try to run this in swi prolog there is a warning 'singleton variable' and when I press
suggest(apple).
then it says false.. sorry for my english :/
Predicates in Prolog do not return anything. You have goals that are satisfied or not and you can interpret that as returning
true
orfalse
.Your predicate
suggest(X)
should contain another parameter that will be bound to the list of fruits that go together withX
. An option would be:suggest(X, List)
which describes the following relation:List
represents all the fruits that go together withX
. Then, you could ask:The goal
findall(Y, ... , ...)
uses theY
variable internally andY
is still unbound after the goal is satisfied. So, you should movecoexist(X,Y)
inside the second argument offindall/3
which is the goal that is satisfied in all possible ways. Th rule below works only ifX
is instantiated (suggest(+X, -List)
).You can read this as follows: "
List
represents all fruitsY
that coexist withX
".When you try to define a predicate in Prolog, first of all pretend that you have written that predicate already and start with imagining how you would use it. That is, what queries you would like to pose.
To me, it looks as if
coexist/2
already describes what you want. BTW,may_coexist/2
might be a more descriptive name. Why do you want this in a separate list? And why usingfruit/3
at all? But for the sake of the question let's assume that this makes sense. So essentially you would have now a relationfruit_compatible/2
:And now, let's assume you want this list too. So you would have a relation
fruit_suggestions/2
. How to use it?or ... should it be rather
L = [pear, cherry]
? Or both?So every time I want a suggestion I have to think of a fruit. Always thinking: what fruit should it be? Fortunately there is a less demanding way in Prolog: Simply use a variable instead of the fruit! Now we should get all suggestions at once!
So we need to implement it such that it will behave that way.
findall/3
alone does not solve this. And implementing it manually is far from trivial. But there issetof/3
which handles variables in exactly that manner. Many of the tiny nitty-gritty design decisions have already been made, like that the list will be sorted ascendingly.Edit: Due to the discussion below, here would be a solution that also permits empty lists. Note that this sounds trivial but it is not. To see this, consider the query:
What does it mean? What should
F
be? Also things that are no fruits at all? In that case we would have to produce solutions for everything. LikeF = badger ; F = 42 ; ...
. Most probably this does not make much sense. What might be intended is those fruits that are incompatible with everything. To this end, we need to add a new rule: