Prolog
Prolog
Prolog
Amzi! offers both free and commercial Prolog development tools. This tutorial text
refers to the Amzi!® Eclipse IDE. Amzi! runs under all forms of Windows, and many
forms of Unix including Linux and Solaris. You can download a copy from our web
site:
www.amzi.com
You can find other freely available Prolog tools for Windows and other plat-forms in
the Prolog FAQ and on various Prolog and AI repositories. Our web site contains up-
to-date pointers to these sites.
You will also find numerous articles on our web site about:
• Books,
• Prolog source code repositories,
• Papers and FAQs,
• Other Prolog, AI and expert systems sites, and
• Newsgroups.
Getting Started
In fact, the term program does not accurately describe a Prolog collection of
executable facts, rules and logical relationships, so you will often see term
logicbase used in this book as well.
While Prolog is a fascinating language from a purely theoretical viewpoint, this book
will stress Prolog as a practical tool for application development.
Amzi Prolog 2/78
Much of the book will be built around the writing of a short adventure game. The
adventure game is a good example since it contains mundane programming
constructs, symbolic reasoning, natural language, data, and logic.
Through exercises you will also build a simple expert system, an intelligent
genealogical logicbase, and a mundane customer order entry application.
You should create a source file for the game, and enter the examples from the book
as you go. You should also create source files for the other three programs covered
in the exercises. Sample source code for each of the programs is included in the
appendix.
The adventure game is called Nani Search. Your persona as the adventurer is that
of a three year old girl. The lost treasure with magical powers is your nani (security
blanket). The terrifying obstacle between you and success is a dark room. It is
getting late and you're tired, but you can't go to sleep without your nani. Your
mission is to find the nani.
You control the game by using simple English commands (at the angle bracket (>)
prompt) expressing the action you wish to take. You can go to other rooms, look at
your surroundings, look in things, take things, drop things, eat things, inventory
the things you have, and turn things on and off.
Figure 1.1 shows a run of a completed version of Nani Search. As you develop your
own version you can of course change the game to reflect your own ideas of
adventure.
The game will be implemented from the bottom up, because that fits better with
the order in which the topics will be introduced. Prolog is equally adept at
supporting top-down or inside-out program development.
The predicates can be added and tested separately in a Prolog program, which
makes it possible to incrementally develop the applications described in the book.
Each chapter will call for the addition of more and more predicates to the game.
Similarly, the exercises will ask you to add predicates to each of the other
applications.
We will start with the Nani Search logicbase and quickly move into the commands
that examine that logicbase. Then we will implement the commands that
manipulate the logicbase.
Along the way there will be diversions where the same commands are rewritten
using a different approach for comparison. Occasionally a topic will be covered that
is critical to Prolog but has little application in Nani Search.
Amzi Prolog 3/78
One of the final tasks will be putting together the top-level command processor. We
will finish with the natural language interface.
You can't go to the cellar because it's dark in the cellar, and you're afraid of the dark.
You can't reach the switch and there's nothing to stand on.
> kitchen
As with any language, the best way to learn Prolog is to use it. This book is
designed to be used with a Prolog listener, and will guide you through the building
of four applications.
• Adventure game
• Intelligent genealogical logicbase
• Expert system
• Customer order entry business application
The adventure game will be covered in detail in the main body of the text, and the
others you will build yourself based on the exercises at the end of each chapter.
There will be two types of example code throughout the book. One is code, meant
to be entered in a source file, and the other is interactions with the listener. The
listener interactions are distinguished by the presence of the question mark and
dash (?-) listener prompt.
Here is a two-line program, meant to help you learn the mechanics of the editor
and your listener.
mortal(X) :- person(X).
person(socrates).
In the Amzi! Eclipse IDE, first create a project for your source files. Select File |
New | Project on the main menu, then click on 'Prolog' and 'Project', and enter the
name of your project, 'adventure'. Next, create a new source file. Select File | New
| File, and enter the name of your file, 'mortal.pro'. Enter the pro-gram in the edit
window, paying careful attention to upper and lowercase letters and punctuation.
Then select File | Save from the menu.
Next, start the Prolog listener by selecting Run | Run As | Interpreted Project.
Loading the source code in the Listener is called consulting. You should see a
message indicating that your source file, 'mortal.pro', was consulted. This message
is followed by the typical listener prompt.
?-
?- consult(mortal).
yes
See the documentation and/or online help for details on the Amzi! listener and
Eclipse IDE.
In all the listener examples in this book, you enter the text after the prompt (?),
the rest is provided by Prolog. When working with Prolog, it is important to
remember to include the final period and to press the 'return' key. If you forget the
period (and you probably will), you can enter it on the next line with a 'return.'
Once you've loaded the program, try the following Prolog queries.
?- mortal(socrates).
Amzi Prolog 5/78
yes
?- mortal(X).
X = socrates.
Now let's change the program. First type 'quit.' to end the listener. Go back to the
edit window and add the line
person(plato).
Select Run | Run As | Interpreted Project to start the listener again with your
updated source file. And test it.
?- mortal(plato).
yes
?- write('Hello World').
Hello World
yes
Logic Programming
Let's look at the simple example in more detail. In classical logic we might say "All
people are mortal," or, rephrased for Prolog, "For all X, X is mortal if X is a person."
mortal(X) :- person(X).
person(socrates).
From these two logical assertions, Prolog can now prove whether or not Socrates is
mortal.
?- mortal(socrates).
yes
?- mortal(X).
X = socrates
The following example illustrates a Prolog program that prints a report of all the
known mortals. It is a mixture of pure logic from before, extra-logical I/O, and
forced control of the Prolog execution behavior. The example is illustrative only,
and the concepts involved will be explained in later chapters.
First add some more philosophers to the 'mortal' source in order to make the report
more interesting. Place them after 'person(plato).'
person(zeno).
person(aristotle).
Next add the report-writing code, again being careful with punctuation and upper-
and lowercase. Note that the format of this program is the same as that used for
the logical assertions.
mortal_report:-
write('Known mortals are:'),nl,
mortal(X),
write(X),nl,
fail.
Figure 1.2 contains the full program, with some optional comments, indicated by
the percent sign (%) at the beginning of a line. Load the program in the listener
and try it. Note that the syntax of calling the report code is the same as the syntax
used for posing the purely logical queries.
?- mortal_report.
Known mortals are:
socrates
plato
aristotle
no
mortal(X) :- person(X).
person(socrates).
person(plato).
person(aristotle).
mortal_report:-
write('Known mortals are:'),nl,
mortal(X),
write(X),nl,
fail.
Amzi Prolog 7/78
Figure 1.2. Sample program
You should now be able to create and edit source files for Prolog, and be able to
load and use them from a Prolog listener.
You have had your first glimpse of Prolog and should understand that it is
fundamentally different from most languages, but can be used to accomplish the
same goals and more.
Jargon
With any field of knowledge, the critical concepts of the field are embedded in the
definitions of its technical terms. Prolog is no exception. When you understand
terms such as predicate, clause, backtracking, and unification you will have a
good grasp of Prolog. This section defines the terms used to describe Prolog
programs, such as predicate and clause. Execution-related terms, such as
backtracking and unification will be introduced as needed throughout the rest of the
text.
Prolog jargon is a mixture of programming terms, database terms, and logic terms.
You have probably heard most of the terms before, but in Prolog they don't
necessarily mean what you think they mean.
In Prolog the normally clear distinction between data and procedure becomes
blurred. This is evident in the vocabulary of Prolog. Almost every concept in Prolog
can be referred to by synonymous terms. One of the terms has a procedural flavor,
and the other a data flavor.
We can illustrate this at the highest level. A Prolog program is a Prolog logicbase.
As we introduce the vocabulary of Prolog, synonyms (from Prolog or other
computer science areas) for a term will follow in parentheses. For example, at the
highest level we have a Prolog program (logicbase).
In our sample program we saw three examples of predicates. They are: person/1,
mortal_report/0, and mortal/1. Each of these three predicates has a distinctly
different flavor.
person/1
looks like multiple data records with one data field for each.
mortal_report/0
looks like a procedure with no arguments.
mortal/1
a logical assertion or rule that is somewhere in between data and procedure.
A clause can be either a fact or a rule. The three clauses of the person/1 predicate
are all facts. The single clauses of mortal_report/0 and mortal/1 are both rules.
Amzi Prolog 8/78
Facts
This chapter describes the basic Prolog facts. They are the simplest form of Prolog
predicates, and are similar to records in a relational database. As we will see in the
next chapter they can be queried like database records.
where
pred
The name of the predicate
arg1, ...
The arguments
N
The arity
.
The syntactic end of all Prolog clauses
pred.
The arguments can be any legal Prolog term. The basic Prolog terms are
integer
A positive or negative number whose absolute value is less than some
implementation-specific power of 2
atom
A text constant beginning with a lowercase letter
variable
Begins with an uppercase letter or underscore (_)
structure
Complex terms, which will be covered in chapter 9
Various Prolog implementations enhance this basic list with other data types, such
as floating point numbers, or strings.
Integers are made from digits. Other numerical types are allowed in some Prolog
implementations.
Atoms are usually made from letters and digits with the first character being a
lowercase letter, such as
hello
Amzi Prolog 9/78
twoWordsTogether
x14
For readability, the underscore (_), but not the hyphen (-), can be used as a
separator in longer names. So the following are legal.
a_long_atom_name
z_23
no-embedded-hyphens
123nodigitsatbeginning
_nounderscorefirst
Nocapsfirst
Use single quotes to make any character combination a legal atom as follows.
'this-hyphen-is-ok'
'UpperCase'
'embedded blanks'
Do not use double quotes ("") to build atoms. This is a special syntax that causes
the character string to be treated as a list of ASCII character codes.
-->
++
Variables are similar to atoms, but are distinguished by beginning with either an
uppercase letter or the underscore (_).
X
Input_List
_4th_argument
Z56
Using these building blocks, we can start to code facts. The predicate name follows
the rules for atoms. The arguments can be any Prolog terms.
Facts are often used to store the data a program is using. For example, a business
application might have customer/3.
The single quotes are needed around the names because they begin with uppercase
letters and because they have embedded blanks.
Another example is a windowing system that uses facts to store data about the
various windows. In this example the arguments give the window name and
coordinates of the upper left and lower right corners.
disease(plague, infectious).
A Prolog listener provides the means for dynamically recording facts and rules in
the logicbase, as well as the means to query (call) them. The logicbase is updated
by 'consult'ing or 'reconsult'ing program source. Predicates can also be typed
directly into the listener, but they are not saved between sessions.
Nani Search
We will now begin to develop Nani Search by defining the basic facts that are
meaningful for the game. These include
Open a new source file and save it as 'myadven.pro', or whatever name you feel is
appropriate. You will make your changes to the program in that source file. (A
completed version of nanisrch.pro is in the Prolog samples directory,
samples/prolog/misc_one_file.)
First we define the rooms with the predicate room/1, which has five clauses, all of
which are facts. They are based on the game map in figure 2.1.
room(kitchen).
room(office).
room(hall).
room('dining room').
room(cellar).
location(desk, office).
location(apple, kitchen).
location(flashlight, desk).
location('washing machine', cellar).
location(nani, 'washing machine').
location(broccoli, kitchen).
location(crackers, kitchen).
Amzi Prolog 11/78
location(computer, office).
The symbols we have chosen, such as kitchen and desk have meaning to us, but
none to Prolog. The relationship between the arguments should also accurately
reflect our meaning.
For example, the meaning we attach to location/2 is "The first argument is located
in the second argument." Fortunately Prolog considers location(sink, kitchen) and
location(kitchen, sink) to be different. Therefore, as long as we are consistent in
our use of arguments, we can accurately represent our meaning and avoid the
potentially ambiguous interpretation of the kitchen being in the sink.
We are not as lucky when we try to represent the connections between rooms. Let's
start, however, with door/2, which will contain facts such as
door(office, hall).
We would like this to mean "There is a connection from the office to the hall, or
from the hall to the office."
door(office, hall).
door(hall, office).
The strictness about order serves our purpose well for location, but it creates this
problem for connections between rooms. If the office is connected to the hall, then
we would like the reverse to be true as well.
For now, we will just add one-way doors to the program; we will address the
symmetry problem again in the next chapter and resolve it in chapter 5.
door(office, hall).
door(kitchen, office).
door(hall, 'dining room').
door(kitchen, cellar).
door('dining room', kitchen).
Here are some other facts about properties of things the game player might try to
eat.
edible(apple).
edible(crackers).
tastes_yucky(broccoli).
Finally we define the initial status of the flashlight, and the player's location at the
beginning of the game.
turned_off(flashlight).
here(kitchen).
We have now seen how to use basic facts to represent data in a Prolog program.
Amzi Prolog 12/78
Exercises
During the course of completing the exercises you will develop three Prolog
applications in addition to Nani Search. The exercises from each chapter will build
on the work of previous chapters. Suggested solutions to the exercises are
contained in the Prolog source files listed in the appendix, and are also included in
samples/prolog/misc_one_file. The files are
gene
A genealogical intelligent logicbase
custord
A customer order entry application
birds
An expert system that identifies birds
Not all applications will be covered in each chapter. For example, the expert system
requires an understanding of rules and will not be started until the end of chapter
5.
Genealogical Logicbase
1- First create a source file for the genealogical logicbase application. Start by
adding a few members of your family tree. It is important to be accurate, since we
will be exploring family relationships. Your own knowledge of who your relatives are
will verify the correctness of your Prolog programs.
Start by recording the gender of the individuals. Use two separate predicates,
male/1 and female/1. For example
male(dennis).
male(michael).
female(diana).
male('Ghenghis Khan').
parent(dennis, michael).
parent(dennis, diana).
3- Create a source file for the customer order entry program. We will begin it with
three record types (predicates). The first is customer/3 where the three arguments
are
arg1
Customer name
arg2
Amzi Prolog 13/78
City
arg3
Credit rating (aaa, bbb, etc)
4- Next add clauses that define the items that are for sale. It should also have
three arguments
arg1
Item identification number
arg2
Item name
arg3
The reorder point for inventory (when at or below this level, reorder)
5- Next add an inventory record for each item. It has two arguments.
arg1
Item identification number (same as in the item record)
arg2
Amount in stock
Simple Queries
Now that we have some facts in our Prolog program, we can consult the program in
the listener and query, or call, the facts. This chapter, and the next, will assume the
Prolog program contains only facts. Queries against programs with rules will be
covered in a later chapter.
Prolog queries work by pattern matching. The query pattern is called a goal. If
there is a fact that matches the goal, then the query succeeds and the listener
responds with 'yes.' If there is no matching fact, then the query fails and the
listener responds with 'no.'
Prolog's pattern matching is called unification. In the case where the logicbase
contains only facts, unification succeeds if the following three conditions hold.
• The predicate named in the goal and logicbase are the same.
• Both predicates have the same arity.
• All of the arguments are the same.
Before proceeding, review figure 3.1, which has a listing of the program so far.
The first query we will look at asks if the office is a room in the game. To pose this,
we would enter that goal followed by a period at the listener prompt.
?- room(office).
yes
Prolog will respond with a 'yes' if a match was found. If we wanted to know if the
attic was a room, we would enter that goal.
?- room(attic).
no
Amzi Prolog 14/78
room(kitchen).
room(office).
room(hall).
room('dining room').
room(cellar).
door(office, hall).
door(kitchen, office).
door(hall, 'dining room').
door(kitchen, cellar).
door('dining room', kitchen).
location(desk, office).
location(apple, kitchen).
location(flashlight, desk).
location('washing machine', cellar).
location(nani, 'washing machine').
location(broccoli, kitchen).
location(crackers, kitchen).
location(computer, office).
edible(apple).
edible(crackers).
tastes_yucky(broccoli).
here(kitchen).
Prolog will respond with a 'no' if no match was found. Likewise, we can ask about
the locations of things.
?- location(apple, kitchen).
yes
?- location(kitchen, apple).
no
Prolog responds to our location query patterns in a manner that makes sense to us.
That is, the kitchen is not located in the apple.
However, here is the problem with the one-way doors, which we still haven't fixed.
It is mentioned again to stress the importance of the order of the arguments.
?- door(office, hall).
yes
?- door(hall, office).
no
Goals can be generalized by the use of Prolog variables. They do not behave like
the variables in other languages, and are better called logical variables (although
Prolog does not precisely correspond to logic). The logical variables replace one or
more of the arguments in the goal.
Amzi Prolog 15/78
Logical variables add a new dimension to unification. As before, the predicate
names and arity must be the same for unification to succeed. However, when the
corresponding arguments are compared, a variable will successfully match any
term.
After successful unification, the logical variable takes on the value of the term it
was matched with. This is called binding the variable. When a goal with a variable
successfully unifies with a fact in the logicbase, Prolog returns the value of the
newly bound variable.
Since there may be more than one value a variable can be bound to and still satisfy
the goal, Prolog provides the means for you to ask for alternate values. After an
answer you can enter a semicolon (;). It causes Prolog to look for alternative
bindings for the variables. Entering anything else at the prompt ends the query.
For example, we can use a logical variable to find all of the rooms.
?- room(X).
X = kitchen ;
X = office ;
X = hall ;
X = 'dining room' ;
X = cellar ;
no
Here's how to find all the things in the kitchen. (Remember, variables begin with
uppercase letters.)
?- location(Thing, kitchen).
Thing = apple ;
Thing = broccoli ;
Thing = crackers ;
no
?- location(Thing, Place).
Thing = desk
Place = office ;
Thing = apple
Place = kitchen ;
Thing = flashlight
Place = desk ;
...
no
?- chapter(2,Title).
?- window(main,Row1,Col1,Row2,Col2).
When Prolog tries to satisfy a goal about a predicate, such as location/2, it searches
through the clauses defining location/2. When it finds a match for its variables, it
marks the particular clause that was used to satisfy the goal. Then, if the user asks
for more answers, it resumes its search of the clauses at that place marker.
Referring to the list of clauses in figure 3.1, let's look closer at this process with the
query location(X, kitchen). First, unification is attempted between the query pattern
and the first clause of location/2.
Pattern Clause #1
location(X, kitchen) location(desk, office)
This unification fails. The predicate names are the same, the number of arguments
is the same, but the second argument in the pattern, kitchen, is different from the
second argument in the clause, office.
Next, unification is attempted between the pattern and the second clause of
location/2.
Pattern Clause #2
location(X, kitchen) location(apple, kitchen)
This unification succeeds. The predicate names, arity (number of arguments), and
second arguments are the same. The first arguments can be made the same if the
variable X in the pattern takes the value 'apple.'
Now that unification succeeds, the Prolog listener reports its success, and the
binding of the variable X.
?- location(X, kitchen).
X = apple
If the user presses a key other than the semicolon (;) at this point, the listener
responds with 'yes' indicating the query ended successfully.
If the user presses the semicolon (;) key, the listener looks for other solutions. First
it unbinds the variable X. Next it resumes the search using the clause following the
one that had just satisfied the query. This is called backtracking. In the example
that would be the third clause.
Pattern Clause #3
location(X, kitchen) location(flashlight, desk)
This fails, and the search continues. Eventually the sixth clause succeeds.
Pattern Clause #6
location(X, kitchen) location(broccoli, kitchen)
Amzi Prolog 17/78
As a result, the variable X is now rebound to broccoli, and the listener responds
X = broccoli ;
Again, entering a semicolon (;) causes X to be unbound and the search to continue
with the seventh clause, which also succeeds.
X = crackers ;
As before, entering anything except a semicolon (;) causes the listener to respond
'yes,' indicating success. A semicolon (;) causes the unbinding of X and the search
to continue. But now, there are no more clauses that successfully unify with the
pattern, so the listener responds with 'no' indicating the final attempt has failed.
no
The best way to understand Prolog execution is to trace its execution in the
debugger. But first it is necessary to have a deeper understanding of goals.
A Prolog goal has four ports representing the flow of control through the goal: call,
exit, redo, and fail. First the goal is called. If successful it is exited. If not it fails. If
the goal is retried, by entering a semicolon (;) the redo port is entered. Figure 3.2
shows the goal and its ports.
call
Begins searching for clauses that unify with the goal
exit
Indicates the goal is satisfied, sets a place marker at the clause and binds
the variables appropriately
redo
Retries the goal, unbinds the variables and resumes search at the place
marker
fail
Indicates no more clauses match the goal
Prolog debuggers use these ports to describe the state of a query. Figure 3.3 shows
a trace of the location(X, kitchen) query. Study it carefully because it is the key to
your understanding of Prolog. The number in parentheses indicates the current
clause.
?- location(X, kitchen).
CALL: - location(X, kitchen)
EXIT:(2) location(apple, kitchen)
X = apple ;
REDO: location(X, kitchen)
EXIT:(6) location(broccoli, kitchen)
X = broccoli ;
Amzi Prolog 18/78
Because the trace information presented in this book is designed to teach Prolog
rather than debug it, the format is a little different from that used in the actual
debugger. Run the Amzi! Source Code Debugger on these queries to see how they
work for real.
To start the Amzi! Debugger, highlight your project name or edit a source file in
your project, then select Run | Debug As | Interpreted Project from the main menu.
You will see a separate perspective with multiple views that contain trace
information. Enter the query 'location(X, kitchen)' in the Debug Listener view. You
will see the trace start in the debugger view.
Use the 'Step Over' button in the debugger to creep from port to port. When output
appears in the listener view, enter semicolons (;) to continue the search. See the
help files for more details on the debugger.
Unification between goals and facts is actually more general than has been
presented. Variables can also occur in the facts of the Prolog logicbase as well.
For example, the following fact could be added to the Prolog program. It might
mean everyone sleeps.
sleeps(X).
You can add it directly in the listener, to experiment with, like this.
?- assert(sleeps(X)).
yes
Queries against a logicbase with this fact give the following results.
?- sleeps(jane).
yes
?- sleeps(tom).
yes
Notice that the listener does not return the variable bindings of 'X=jane' and
'X=tom.' While they are surely bound that way, the listener only lists variables
mentioned in the query, not those used in the program.
?- sleeps(Z).
Z = H116
?- sleeps(X).
X = H247
Amzi Prolog 19/78
When two unbound variables match, they are both bound, but not to a value! They
are bound together, so that if either one takes a value, the other takes the same
value. This is usually implemented by binding both variables to a common internal
variable. In the first query above, both Z in the query and X in the fact are bound
to internal variable 'H116.' In this way Prolog remembers they have the same
value. If either one is bound to a value later on, both automatically bind to that
value. This feature of Prolog distinguishes it from other languages and, as we will
discover later, gives Prolog much of its power.
The two queries above are the same, even though one uses the same character X
that is used in the fact sleeps(X). The variable in the fact is considered different
from the one in the query.
Exercises
The exercise sections will often contain nonsense Prolog questions. These are
queries against a meaningless logicbase to strengthen your understanding of Prolog
without the benefit of meaningful semantics. You are to predict the answers to the
query and then try them in Prolog to see if you are correct. If you are not, trace the
queries to better understand them.
Nonsense Prolog
easy(1).
easy(2).
easy(3).
gizmo(a,1).
gizmo(b,3).
gizmo(a,2).
gizmo(d,5).
gizmo(c,3).
gizmo(a,3).
gizmo(c,4).
and predict the answers to the queries below, including all alternatives when the
semicolon (;) is entered after an answer.
?- easy(2).
?- easy(X).
?- gizmo(a,X).
?- gizmo(X,3).
?- gizmo(d,Y).
?- gizmo(X,X).
harder(a,1).
harder(c,X).
harder(b,4).
harder(d,2).
Adventure Game
3- Enter the listener and reproduce some of the example queries you have seen
against location/2. List or print location/2 for reference if you need it. Remember to
respond with a semicolon (;) for multiple answers. Trace the query.
Genealogical Logicbase
5- If parent/2 seems to be working, you can add additional family members to get
a larger logicbase. Remember to include the corresponding male/1 or female/1
predicate for each individual added.
Compound Queries
Simple goals can be combined to form compound queries. For example, we might
want to know if there is anything good to eat in the kitchen. In Prolog we might ask
Whereas a simple query had a single goal, the compound query has a conjunction
of goals. The comma separating the goals is read as "and."
Logically (declaratively) the example means "Is there an X such that X is located in
the kitchen and X is edible?" If the same variable name appears more than once in
a query, it must have the same value in all places it appears. The query in the
above example will only succeed if there is a single value of X that can satisfy both
goals.
However, the variable name has no significance to any other query, or clause in the
logicbase. If X appears in other queries or clauses, that query or clause gets its own
copy of the variable. We say the scope of a logical variable is a query.
Amzi Prolog 21/78
Trying the sample query we get
The 'broccoli' does not show up as an answer because we did not include it in the
edible/1 predicate.
Each goal can be entered from either the left or the right, and can be left from
either the left or the right. These are the ports of the goal as seen in the last
chapter.
A compound query begins by calling the first goal on the left. If it succeeds, the
next goal is called with the variable bindings as set from the previous goal. If the
query finishes via the exit port of the rightmost goal, it succeeds, and the listener
prints the values in the variable table.
If the user types semicolon (;) after an answer, the query is re-entered at the redo
port of the rightmost goal. Only the variable bindings that were set in that goal are
undone.
If the query finishes via the fail port of the leftmost goal, the query fails. Figure 4.1
shows a compound query with the listener interaction on the ending ports.
Figure 4.2 contains the annotated trace of the sample query. Make sure you
understand it before proceeding.
The trace has a new feature, which is a number in the first column that indicates the
goal being worked on.
First the goal location(X, kitchen) is called, and the trace indicates that pattern
matches the second clause of location.
Amzi Prolog 22/78
Next, the second goal edible(X) is called. However, X is now bound to apple, so it is
called as edible(apple).
2 CALL edible(apple)
It succeeds on the first clause of edible/1, thus exiting the query successfully.
Entering semicolon (;) causes the listener to backtrack into the rightmost goal of the
query.
2 REDO edible(apple)
2 FAIL edible(apple)
Leaving the fail port of the second goal causes the listener to enter the redo port of
the first goal. In so doing, the variable binding that was established by that goal is
undone, leaving X unbound.
The second goal is called again with the new variable binding. This is a fresh call, just
as the first one was, and causes the search for a match to begin at the first clause
2 CALL edible(broccoli)
2 FAIL edible(broccoli)
The first goal is then re-entered at the redo port, undoing the variable binding.
2 CALL edible(crackers)
Amzi Prolog 23/78
Typing semicolon (;) initiates backtracking again, which fails through both goals and
leads to the ultimate failure of the query.
2 REDO edible(crackers)
2 FAIL edible(crackers)
1 REDO location(X, kitchen)
1 FAIL location(X, kitchen)
no
In this example we had a single variable, which was bound (given a value) by the
first goal and tested in the second goal. We will now look at a more general
example with two variables. It is attempting to ask for all the things located in
rooms adjacent to the kitchen.
In logical terms, the query says "Find a T and R such that there is a door from the
kitchen to R and T is located in R." In procedural terms it says "First find an R with
a door from the kitchen to R. Use that value of R to look for a T located in R."
R = office
T = computer ;
R = cellar
T = 'washing machine' ;
no
In this query, the backtracking is more complex. Figure 4.3 shows its trace.
Notice that the variable R is bound by the first goal and T is bound by the second.
Likewise, the two variables are unbound by entering the redo port of the goal that
bound them. After R is first bound to office, that binding sticks during backtracking
through the second goal. Only when the listener backtracks into the first goal does
R get unbound.
1 CALL door(kitchen, R)
1 EXIT (2) door(kitchen, office)
2 CALL location(T, office)
2 EXIT (1) location(desk, office)
R = office
T = desk ;
2 REDO location(T, office)
2 EXIT (8) location(computer, office)
R = office
T = computer ;
2 REDO location(T, office)
2 FAIL location(T, office)
Amzi Prolog 24/78
1 REDO door(kitchen, R)
1 EXIT (4) door(kitchen, cellar)
2 CALL location(T, cellar)
2 EXIT (4) location('washing machine', cellar)
R = cellar
T = 'washing machine' ;
2 REDO location(T, cellar)
2 FAIL location(T, cellar)
1 REDO door(kitchen, R)
1 FAIL door(kitchen, R)
no
Built-in Predicates
Up to this point we have been satisfied with the format Prolog uses to give us
answers. We will now see how to generate output that is customized to our needs.
The example will be a query that lists all of the items in the kitchen. This will
require performing I/O and forcing the listener to automatically backtrack to find all
solutions.
Built-in predicates are usually written in the language used to implement the
listener. They can perform functions that have nothing to do with logical theorem
proving, such as writing to the console. For this reason they are sometimes called
extra-logical predicates.
However, since they appear as Prolog goals they must be able to respond to either
a call from the left or a redo from the right. Its response in the redo case is referred
to as its behavior on backtracking.
We will introduce specific built-in predicates as we need them. Here are the I/O
predicates that will let us control the output of our query.
write/1
This predicate always succeeds when called, and has the side effect of
writing its argument to the console.It always fails on backtracking.
Backtracking does not undo the side effect.
nl/0
Succeeds, and starts a new line. Like write, it always succeeds when called,
and fails on backtracking.
tab/1
It expects the argument to be an integer and tabs that number of spaces. It
succeeds when called and fails on backtracking.
Figure 4.4 is a stylized picture of a goal showing its internal control structure. We
will compare this with the internal flow of control of various built-in predicates.
Amzi Prolog 25/78
In figure 4.4, the upper left diamond represents the decision point after a call.
Starting with the first clause of a predicate, unification is attempted between the
query pattern and each clause, until either unification succeeds or there are no
more clauses to try. If unification succeeded, branch to exit, marking the clause
that successfully unified, if it failed, branch to fail.
The lower right diamond represents the decision point after a redo. Starting with
the most recent clause found in the predicate, unification is again attempted
between the query pattern and remaining clauses. If it succeeds, branch to exit, if
not, branch to fail.
The I/O built-in predicates differ from normal goals in that they never change the
direction of the flow of control. If they get control from the left, they pass control to
the right. If they get control from the right, they pass control to the left as shown in
figure 4.5.
The output I/O predicates do not affect the variable table; however, they may
output values from it. They simply leave their mark at the console each time control
passes through them from left to right.
There are built-in predicates that do affect backtracking, and we have need of one
of them for the first example. It is fail/0, and, as its name implies, it always fails.
If fail/0 gets control from the left, it immediately passes control back to the redo
port of the goal on the left. It will never get control from the right, since it never
allows control to pass to its right. Figure 4.6 shows its internal control structure.
Previously we relied on the listener to display variable bindings for us, and used the
semicolon (;) response to generate all of the possible solutions. We can now use
the I/O built-in predicates to display the variable bindings, and the fail/0 predicate
to force backtracking so all solutions are displayed.
The final 'no' means the query failed, as it was destined to, due to the fail/0.
3 EXIT nl
4 CALL fail
4 FAIL fail
3 REDO nl
3 FAIL nl
2 REDO write(apple)
2 FAIL write(apple)
1 REDO location(X, kitchen)
1 EXIT (6) location(broccoli, kitchen)
2 CALL write(broccoli)
broccoli
2 EXIT write(broccoli)
3 CALL nl
3 EXIT nl
4 CALL fail
4 FAIL fail
3 REDO nl
3 FAIL nl
2 REDO write(broccoli)
2 FAIL write(broccoli)
1 REDO location(X, kitchen)
1 EXIT (7) location(crackers, kitchen)
2 CALL write(crackers)
crackers
2 EXIT write(crackers)
3 CALL nl
Amzi Prolog 27/78
3 EXIT nl
4 CALL fail
4 FAIL fail
3 REDO nl
3 FAIL nl
2 REDO write(crackers)
2 FAIL write(crackers)
1 REDO location(X, kitchen)
1 FAIL location(X, kitchen)
no
Exercises
Nonsense Prolog
easy(1).
easy(2).
easy(3).
gizmo(a,1).
gizmo(b,3).
gizmo(a,2).
gizmo(d,5).
gizmo(c,3).
gizmo(a,3).
gizmo(c,4).
harder(a,1).
harder(c,X).
harder(b,4).
harder(d,2).
Predict the results of the following queries. Then try them and trace them to see if
you were correct.
?- gizmo(a,X),easy(X).
?- gizmo(c,X),easy(X).
?- gizmo(d,Z),easy(Z).
?- easy(Y),gizmo(X,Y).
?- easy(X),harder(Y,X).
?- harder(Y,X),easy(X).
Adventure Game
Genealogical Logicbase
Write similar queries for fathers, sons, and daughters. Trace these queries to
understand their behavior (or misbehavior if they are not working right for you).
5- Experiment with the ordering of the goals. In particular, contrast the queries.
Do they both give the same answer? Trace both queries and see which takes more
steps.
6- The same predicate can be used multiple times in the same query. For example,
we can find grandparents
8- Write a query against the item and inventory records that returns the inventory
level for an item when you only know the item name.
Rules
head :- body.
where
head
a predicate definition (just like a fact)
:-
the neck symbol, sometimes read as "if"
body
one or more goals (a query)
For example, the compound query that finds out where the good things to eat are
can be stored as a rule with the predicate name where_food/2.
Amzi Prolog 29/78
where_food(X,Y) :-
location(X,Y),
edible(X).
We can now use the new rule directly in a query to find things to eat in a room. As
before, the semicolon (;) after an answer is used to find all the answers.
?- where_food(X, kitchen).
X = apple ;
X = crackers ;
no
?- where_food(apple, kitchen).
yes
?- where_food(Thing, Room).
Thing = apple
Room = kitchen ;
Thing = crackers
Room = kitchen ;
no
Just as we had multiple facts defining a predicate, we can have multiple rules for a
predicate. For example, we might want to have the broccoli included in
where_food/2. (Prolog doesn't have an opinion on whether or not broccoli is
legitimate food. It just matches patterns.) To do this we add another where_food/2
clause for things that 'taste_yucky.'
where_food(X,Y) :-
location(X,Y),
edible(X).
where_food(X,Y) :-
location(X,Y),
tastes_yucky(X).
Now the broccoli shows up when we use the semicolon (;) to ask for everything.
?- where_food(X, kitchen).
X = apple ;
X = crackers ;
X = broccoli ;
no
Until this point, when we have seen Prolog try to satisfy goals by searching the
clauses of a predicate, all of the clauses have been facts.
Amzi Prolog 30/78
How Rules Work
With rules, Prolog unifies the goal pattern with the head of the clause. If unification
succeeds, then Prolog initiates a new query using the goals in the body of the
clause.
Rules, in effect, give us multiple levels of queries. The first level is composed of the
original goals. The next level is a new query composed of goals found in the body of
a clause from the first level.
Each level can create even deeper levels. Theoretically, this could continue forever.
In practice it can continue until the listener runs out of space.
Figure 5.1 shows the control flow after the head of a rule has been matched. Notice
how backtracking from the third goal of the first level now goes into the second
level.
In this example, the middle goal on the first level succeeds or fails if its body
succeeds or fails. When entered from the right (redo) the goal reenters its body
query from the right (redo). When the query fails, the next clause of the first-level
goal is tried, and if the next clause is also a rule, the process is repeated with the
second clause's body.
means the exit occurred at the second level, first goal using clause number seven.
The query is
?- where_food(X, kitchen).
The pattern matches the head of the first clause, and while it is not at a port, the
trace could inform us of the clause it is working on.
Amzi Prolog 31/78
The body of the first clause is then set up as a query, and the trace continues.
From this point the trace proceeds exactly as it did for the compound query in the
previous chapter.
Since the body has succeeded, the goal from the previous (first) level succeeds.
Backtracking goes from the first-level goal, into the second level, proceeding as
before.
Now any attempt to backtrack into the query will result in no more answers, and the
query will fail.
This causes the listener to look for other clauses whose heads match the query
pattern. In our example, the second clause of where_food/2 also matches the query
pattern.
Again, although traces usually don't tell us so, it is building a query from the body of
the second clause.
Now the second query proceeds as normal, finding the broccoli, which tastes_yucky.
Amzi Prolog 32/78
To better understand the relationship, we will slowly step through the process of
transferring control. Subscripts identify the variable levels.
?- where_food(X1, kitchen)
where_food(X2, Y2)
Remember the 'sleeps' example in chapter 3 where a query with a variable was
unified with a fact with a variable? Both variables were set to be equal to each
other. This is exactly what happens here. This might be implemented by setting
both variables to a common internal variable. If either one takes on a new value,
both take on a new value.
So, after unification between the goal and the head, the variable bindings are
X1 = _01
X2 = _01
Y2 = kitchen
Amzi Prolog 33/78
The second-level query is built from the body of the clause, using these bindings.
When internal variable _01 takes on a value, such as 'apple,' both X's then take on
the same value. This is fundamentally different from the assignment statements
that set variable values in most computer languages.
Using Rules
Using rules, we can solve the problem of the one-way doors. We can define a new
two-way predicate with two clauses, called connect/2.
connect(X,Y) :- door(X,Y).
connect(X,Y) :- door(Y,X).
?- connect(kitchen, office).
yes
?- connect(office, kitchen).
yes
We can list all the connections (which is twice the number of doors) with a general
query.
?- connect(X,Y).
X = office
Y = hall ;
X = kitchen
Y = office ;
...
X = hall
Y = office ;
X = office
Y = kitchen ;
...
With our current understanding of rules and built-in predicates we can now add
more rules to Nani Search. We will start with look/0, which will tell the game player
where he or she is, what things are in the room, and which rooms are adjacent.
To begin with, we will write list_things/1, which lists the things in a room. It uses
the technique developed at the end of chapter 4 to loop through all the pertinent
facts.
list_things(Place) :-
location(X, Place),
tab(2),
write(X),
nl,
fail.
Amzi Prolog 34/78
We use it like this.
?- list_things(kitchen).
apple
broccoli
crackers
no
There is one small problem with list_things/1. It gives us the list, but it always fails.
This is all right if we call it by itself, but we won't be able to use it in conjunction
with other rules that follow it (to the right as illustrated in our diagrams). We can
fix this problem by adding a second list_things/1 clause which always succeeds.
list_things(Place) :-
location(X, Place),
tab(2),
write(X),
nl,
fail.
list_things(AnyPlace).
Now when the first clause fails (because there are no more location/2s to try) the
second list_things/1 clause will be tried. Since its argument is a variable it will
successfully match with anything, causing list_things/1 to always succeed and leave
through the 'exit' port.
As with the second clause of list_things/1, it is often the case that we do not care
what the value of a variable is, it is simply a place marker. For these situations
there is a special variable called the anonymous variable, represented as an
underscore (_). For example
list_things(_).
Next we will write list_connections/1, which lists connecting rooms. Since rules can
refer to other rules, as well as to facts, we can write list_connections/1 just like
list_things/1 by using the connection/2 rule.
list_connections(Place) :-
connect(Place, X),
tab(2),
write(X),
nl,
fail.
list_connections(_).
Trying it gives us
?- list_connections(hall).
dining room
office
yes
Now we are ready to write look/0. The single fact here(kitchen) tells us where we
are in the game. (In chapter 7 we will see how to move about the game by
dynamically changing here/1.) We can use it with the two list predicates to write
the full look/0.
Amzi Prolog 35/78
look :-
here(Place),
write('You are in the '), write(Place), nl,
write('You can see:'), nl,
list_things(Place),
write('You can go to:'), nl,
list_connections(Place).
?- look.
You are in the kitchen
You can see:
apple
broccoli
crackers
You can go to:
office
cellar
dining room
yes
With this level of understanding, we can make a lot of progress on the exercise
applications. Take some time to work with the programs to consolidate your
understanding before moving on to the following chapters.
Amzi Prolog 36/78
Exercises
Nonsense Prolog
a(a1,1).
a(A,2).
a(a3,N).
b(1,b1).
b(2,B).
b(N,b3).
Predict the answers to the following queries, then check them with Prolog, tracing.
?- a(X,2).
?- b(X,kalamazoo).
?- c(X,b3).
?- c(X,Y).
?- d(X,Y).
Adventure Game
2- Experiment with the various rules that were developed during this chapter,
tracing them all.
3- Write look_in/1 for Nani Search. It should list the things located in its argument.
For example, look_in(desk) should list the contents of the desk.
Genealogical Logicbase
4- Build rules for the various family relationships that were developed as queries in
the last chapter. For example
mother(M,C):-
parent(M,C),
female(M).
5- Build a rule for siblings. You will probably find your rule lists an individual as
his/her own sibling. Use trace to figure out why.
6- We can fix the problem of individuals being their own siblings by using the built-
in predicate that succeeds if two values are unequal, and fails if they are the same.
The predicate is \=(X,Y). Jumping ahead a bit (to operator definitions in chapter
12), we can also write it in the form X \= Y.
7- Use the sibling predicate to define additional rules for brothers, sisters, uncles,
aunts, and cousins.
Amzi Prolog 37/78
8- If we want to represent marriages in the family logicbase, we run into the two-
way door problem we encountered in Nani Search. Unlike parent/2, which has two
arguments with distinct meanings, married/2 can have the arguments reversed
without changing the meaning.
Using the Nani Search door/2 predicate as an example, add some basic family data
with a spouse/2 predicate. Then write the predicate married/2 using connect/2 as a
model.
9- Use the new married predicate to add rules for uncles and aunts that get uncles
and aunts by marriage as well as by blood. You should have two rules for each of
these relationships, one for the blood case and one for the marriage case. Use trace
to follow their behavior.
11- Write a predicate for grandparent/2. Use it to find both a grandparent and a
grandchild.
grandparent(someone, X).
grandparent(X, someone).
Trace its behavior for both uses. Depending on how you wrote it, one use will
require many more steps than the other. Write two predicates, one called
grandparent/2 and one called grandchild/2. Order the goals in each so that they are
efficient for their intended uses.
12- Write a rule item_quantity/2 that is used to find the inventory level of a named
item. This shields the user of this predicate from having to deal with the item
numbers.
13- Write a rule that produces an inventory report using the item_quantity/2
predicate. It should display the name of the item and the quantity on hand. It
should also always succeed. It will be similar to list_things/2.
14- Write a rule which defines a good customer. You might want to identify
different cases of a good customer.
Expert Systems
Expert systems are often called rule-based systems. The rules are "rules of thumb"
used by experts to solve certain problems. The expert system includes an
inference engine, which knows how to use the rules.
The code listing for 'birds' in the appendix contains a sample system that can be
used to identify birds. You will be asked to build a similar system in the exercises.
It can identify anything, from animals to cars to diseases.
Amzi Prolog 38/78
15- Decide what kind of expert system you would like to build, and add a few initial
identification rules. For example, a system to identify house pets might have these
rules.
16- For now, we can use these rules by putting the known facts in the logicbase.
For example, if we add size(medium) and noise(meow) and then pose the query
pet(X) we will find X=cat.
Many Prologs allow clauses to be entered directly at the listener prompt, which
makes using this expert system a little easier. The presence of the neck symbol (:-)
signals to the listener that the input is a clause to be added. So to add facts directly
to the listener workspace, they must be made into rules, as follows.
?- size(medium) :- true.
recorded
?- noise(meow) :- true.
recorded
?- assert(size(medium)).
yes
?- assert(noise(meow)).
yes
These examples use the predicates in the general form attribute(value). In this
simple example, the pet attribute is deduced. The size and noise attributes must be
given.
17- Improve the expert system by having it ask for the attribute/values it can't
deduce. We do this by first adding the rules
For now, ask/2 will simply check with the user to see if an attribute/value pair is
true or false. It will use the built-in predicate read/1 which reads a Prolog term
(ending in a period of course).
ask(Attr, Val):-
write(Attr),tab(1),write(Val),
tab(1),write('(yes/no)'),write(?),
read(X),
X = yes.
The last goal, X = yes, attempts to unify X and yes. If yes was read, then it
succeeds, otherwise, it fails.
Arithmetic
Amzi Prolog 39/78
Prolog must be able to handle arithmetic in order to be a useful general purpose
programming language. However, arithmetic does not fit nicely into the logical
scheme of things.
X is <arithmetic expression>
?- X is 2 + 2.
X=4
?- X is 3 * 4 + 2.
X = 14
?- X is 3 * (4 + 2).
X = 18
?- X is (8 / 4) / 2.
X=1
X >Y
X <Y
X >= Y
X =< Y
?- 4 > 3.
yes
?- 4 < 3.
no
?- X is 2 + 2, X > 3.
X=4
Amzi Prolog 40/78
?- X is 2 + 2, 3 >= X.
no
They can be used in rules as well. Here are two example predicates. One converts
centigrade temperatures to Fahrenheit, the other checks if a temperature is below
freezing.
c_to_f(C,F) :-
F is C * 9 / 5 + 32.
freezing(F) :-
F =< 32.
?- c_to_f(100,X).
X = 212
yes
?- freezing(15).
yes
?- freezing(45).
no
Exercises
2- Write a reorder/1 predicate which checks inventory levels in the inventory record
against the reorder quantity in the item record. It should write a message indicating
whether or not it's time to reorder.
Managing Data
asserta(X)
Adds the clause X as the first clause for its predicate. Like the other I/O
predicates, it always fails on backtracking and does not undo its work.
assertz(X)
Same as asserta/1, only it adds the clause X as the last clause for its
predicate.
retract(X)
Amzi Prolog 41/78
Removes the clause X from the logicbase, again with a permanent effect
that is not undone on backtracking.
The ability to manipulate the logicbase is obviously an important feature for Nani
Search. With it we can dynamically change the location of the player, as well as the
stuff that has been picked up and moved.
We will first develop goto/1, which moves the player from one room to another. It
will be developed from the top down, in contrast to look/0 which was developed
from the bottom up.
When the player enters the command goto, we first check if they can go to the
place and if so move them so they can look around the new place. Starting from
this description of goto/1, we can write the main predicate.
goto(Place):-
can_go(Place),
move(Place),
look.
can_go(Place):-
here(X),
connect(X, Place).
?- can_go(office).
yes
?- can_go(hall).
no
Now, can_go/1 succeeds and fails as we want it to, but it would be nice if it gave us
a message when it failed. By adding a second clause, which is tried if the first one
fails, we can cause can_go/1 to write an error message. Since we want can_go/1 to
fail in this situation we also need to add a fail to the second clause.
can_go(Place):-
here(X),
connect(X, Place).
can_go(Place):-
write('You can''t get there from here.'), nl,
fail.
?- can_go(hall).
You can't get there from here.
no
Next we develop move/1, which does the work of dynamically updating the
logicbase to reflect the new location of the player. It retracts the old clause for
here/1 and replaces it with a new one. This way there will always be only one
here/1 clause representing the current place. Because goto/1 calls can_go/1 before
move/1, the new here/1 will always be a legal place in the game.
Amzi Prolog 42/78
move(Place):-
retract(here(X)),
asserta(here(Place)).
We can now use goto/1 to explore the game environment. The output it generates
is from look/0, which we developed in chapter 5.
?- goto(office).
You are in the office
You can see:
desk
computer
You can go to:
hall
kitchen
yes
?- goto(hall).
You are in the hall
You can see:
You can go to:
dining room
office
yes
?- goto(kitchen).
You can't get there from here.
no
We will also need 'asserta' and 'retract' to implement 'take' and 'put' commands in
the game.
Here is take/1. For it we will define a new predicate, have/1, which has one clause
for each thing the game player has. Initially, have/1 is not defined because the
player is not carrying anything.
take(X):-
can_take(X),
take_object(X).
can_take(Thing) :-
here(Place),
location(Thing, Place).
can_take(Thing) :-
write('There is no '), write(Thing),
write(' here.'),
nl, fail.
take_object(X):-
retract(location(X,_)),
asserta(have(X)),
write('taken'), nl.
Amzi Prolog 43/78
As we have seen, the variables in a clause are local to that clause. There are no
global variables in Prolog, as there are in many other languages. The Prolog
logicbase serves that purpose. It allows all clauses to share information on a wider
basis, replacing the need for global variables. 'asserts' and 'retracts' are the tools
used to manipulate this global data.
As with any programming language, global data can be a powerful concept, easily
overused. They should be used with care, since they hide the communication of
information between clauses. The same code will behave differently if the global
data is changed. This can lead to hard-to-find bugs.
Eliminating global data and the 'assert' and 'retract' capabilities of Prolog is a goal
of many logic programmers. It is possible to write Prolog programs without
dynamically modifying the logicbase, thus eliminating the problem of global
variables. This is done by carrying the information as arguments to the predicates.
In the case of an adventure game, the complete state of the game could be
represented as predicate arguments, with each command called with the current
state and returning a new modified state. This approach will be discussed in more
detail in chapter 14.
Although the database-like approach presented here may not be the purest method
from a logical standpoint, it does allow for a very natural representation of this
game application.
Sometimes it is desirable to have a predicate retract its assertions when the redo
port is entered. It is easy to write versions of 'assert' and 'retract' that undo their
work on backtracking.
backtracking_assert(X):-
asserta(X).
backtracking_assert(X):-
retract(X),fail.
The first time through, the first clause is executed. If a later goal fails, backtracking
will cause the second clause to be tried. It will undo the work of the first and fail,
thus giving the desired effect.
Exercises
Adventure Game
1- Write put/1 which retracts a have/1 clause and asserts a location/2 clause in the
current room.
3- Use goto/1, take/1, put/1, look/0, and inventory/0 to move about and examine
the game environment so far.
4- Write the predicates turn_on/1 and turn_off/1 for Nani Search. They will be used
to turn the flashlight on or off.
Amzi Prolog 44/78
5- Add an open/closed status for each of the doors. Write open and closepredicates
that do the obvious. Fix can_go/1 to check whether a door is open and write the
appropriate error message if its not.
7- We can now use the various predicates developed for the customer order entry
system to write a predicate that prompts the user for order information and
generates the order. The predicate can be simply order/0.
order/0 should first prompt the user for the customer name, the item name and the
quantity. For example
It should then use the rules for good_customer and valid_order to verify that this is
a valid order. If so, it should assert a new type of record, order/3, which records
the order information. It can then update_inventory and check whether its time to
reorder.
The customer order entry application has been designed from the bottom up, since
that is the way the material has been presented for learning. The order predicate
should suggest that Prolog is an excellent tool for top-down development as well.
One could start with the concept that processing an order means reading the date,
checking the order, updating inventory, and reordering if necessary. The necessary
details of implementing these predicates could be left for later.
Expert System
8- The expert system currently asks for the same information over and over again.
We can use the logicbase to remember the answers to questions so that ask/2
doesn't re-ask something.
Add a first clause to ask/2 that checks whether the answer is already known and, if
so, succeeds. Add a second clause that checks if the answer is known to be false
and, if so, fails.
The third clause makes sure the answer is not already known, and then asks the
user as before. To do this, the built-in predicate not/1 is used. It fails if its
argument succeeds.
8
Amzi Prolog 45/78
Recursion
Recursion in any language is the ability for a unit of code to call itself, repeatedly, if
necessary. Recursion is often a very powerful and convenient way of representing
certain programming constructs.
In Prolog, recursion occurs when a predicate contains a goal that refers to itself.
As we have seen in earlier chapters, every time a rule is called, Prolog uses the
body of the rule to create a new query with new variables. Since the query is a new
copy each time, it makes no difference whether a rule calls another rule or itself.
A recursive definition (in any language, not just Prolog) always has at least two
parts, a boundary condition and a recursive case.
The boundary condition defines a simple case that we know to be true. The
recursive case simplifies the problem by first removing a layer of complexity, and
then calling itself. At each level, the boundary condition is checked. If it is reached
the recursion ends. If not, the recursion continues.
We will illustrate recursion by writing a predicate that can detect things which are
nested within other things.
Currently our location/2 predicate tells us the flashlight is in the desk and the desk
is in the office, but it does not indicate that the flashlight is in the office.
?- location(flashlight, office).
no
Using recursion, we will write a new predicate, is_contained_in/2, which will dig
through layers of nested things, so that it will answer 'yes' if asked if the flashlight
is in the office.
To make the problem more interesting, we will first add some more nested items to
the game. We will continue to use the location predicate to put things in the desk,
which in turn can have other things inside them.
location(envelope, desk).
location(stamp, envelope).
location(key, envelope).
To list all of things in the office, we would first have to list those things that are
directly in the office, like the desk. We would then list the things in the desk, and
the things inside the things in the desk.
If we generalize a room into being just another thing, we can state a two-part rule
which can be used to deduce whether something is contained in (nested in)
something else.
We will now express this in Prolog. The first rule translates into Prolog in a
straightforward manner.
Amzi Prolog 46/78
is_contained_in(T1,T2) :-
location(T1,T2).
is_contained_in(T1,T2) :-
location(X,T2),
is_contained_in(T1,X).
?- is_contained_in(X, office).
X = desk ;
X = computer ;
X = flashlight ;
X = envelope ;
X = stamp ;
X = key ;
no
?- is_contained_in(envelope, office).
yes
?- is_contained_in(apple, office).
no
As in all calls to rules, the variables in a rule are unique, or scoped, to the rule. In
the recursive case, this means each call to the rule, at each level, has its own
unique set of variables. So the values of X, T1, and T2 at the first level of recursion
are different from those at the second level.
However, unification between a goal and the head of a clause forces a relationship
between the variables of different levels. Using subscripts to distinguish the
variables, and internal Prolog variables, we can trace the relationships for a couple
of levels of recursion.
?- is_contained_in(XQ, office).
is_contained_in(T11, T21) :-
location(X1, T21),
is_contained_in(T11, X1).
When the query is unified with the head of the clause, the variables become bound.
The bindings are
XQ = _01
T11 = _01
T21 = office
X1 = _02
Amzi Prolog 47/78
Note particularly that XQ in the query becomes bound to T11 in the clause, so when
a value of _01 is found, both variables are found.
is_contained_in(_01, office) :-
location(_02, office),
is_contained_in(_01, _02).
When the location/2 goal is satisfied, with _02 = desk, the recursive call becomes
is_contained_in(_01, desk)
That goal unifies with the head of a new copy of the clause, at the next level of the
recursion. After that unification the variables are
When the recursion finds a solution, such as 'envelope,' all of the T1s and X0
immediately take on that value. Figure 8.1 contains a full annotated trace of the
query.
The query is
?- is_contained_in(X, office).
Each level of the recursion will have its own unique variables, but as in all calls to
rules, the variables at a called level will be bound in some relationship to the
variables at the calling level. In the following trace, we will use Prolog internal
variables, so we can see which variables are bound together and which are not. The
items directly in the office are found easily, as the variable _0 is bound to X in the
query and T1 in the rule.
When there are no more location(X, office) clauses, the first clause of
is_contained_in/2 fails, and the second clause is tried. Notice that the call to location
does not have its first argument bound to the same variable. It was X in the rule,
and it gets a new internal value, _4. T1 stays bound to _0.
Having exhausted the things located in the desk, it next begins to look for things
within things located in the desk.
Next, it tries to find things in the envelope and comes up with the stamp.
The simplest way to do this is by always defining the boundary condition first,
ensuring that it is always tried first and that the recursive case is only tried if the
boundary condition fails.
Amzi Prolog 50/78
Pragmatics
We now come to some of the pragmatics of Prolog programming. First consider that
the goal location(X,Y) will be satisfied by every clause of location/2. On the other
hand, the goals location(X, office) or location(envelope, X) will be satisfied by fewer
clauses.
Let's look again at the second rule for is_contained_in/2, and an equally valid
alternate coding.
is_contained_in(T1,T2):-
location(X,T2),
is_contained_in(T1,X).
is_contained_in(T1,T2):-
location(T1,X),
is_contained_in(X,T2).
Both will give correct answers, but the performance of each will depend on the
query. The query is_contained_in(X, office) will execute faster with the first version.
That is because T2 is bound, making the search for location(X, T2) easier than if
both variables were unbound. Similarly, the second version is faster for queries
such as is_contained_in(key, X).
Exercises
Adventure Game
1- Trace the two versions of is_contained_in/2 presented at the end of the chapter
to understand the performance differences between them.
2- Currently, the can_take/1 predicate only allows the player to take things which
are directly located in a room. Modify it so it uses the recursive is_contained_in/2
so that a player can take anything in a room.
Genealogical Logicbase
4- Use ancestor/2 for finding all of a person's ancestors and all of a person's
descendants. Based on your experience with grandparent/2 and grandchild/2, write
a descendant/2 predicate optimized for descendants, as opposed to ancestor/2,
which is optimized for ancestors.
Data Structures
So far we have worked with facts, queries, and rules that use simple data
structures. The arguments to our predicates have all been atoms or integers, the
basic building blocks of Prolog. Examples of atoms we've used are
functor(arg1,arg2,...)
Each of the structure's arguments can be either a primitive data type or another
structure. For example, the things in the game are currently represented using
atoms, such as 'desk' or 'apple,' but we can use structures to create a richer
representation of these things. The following structures describe the object and its
color, size, and weight.
These structures could be used directly in the second argument of location/2, but
for experimentation we will instead create a new predicate, location_s/2. Note that
even though the structures describing the objects in the game are complex, they
still take up only one argument in location_s/2.
Prolog variables are typeless, and can be bound as easily to structures as to atoms.
In fact, an atom is just a simple structure with a functor and no arguments. So we
can ask
?- location_s(X, kitchen).
X = object(candle, red, small, 1) ;
X = object(apple, red, small, 1) ;
X = object(apple, green, small, 1) ;
X = object(table, blue, big, 50) ;
no
We can also pick apart the structure with variables. We can now find all the red
things in the kitchen.
X = apple
S = small
W=1;
no
If we didn't care about the size and weight we could replace the size, S, and
weight, W, variables with the anonymous variable (_).
We can use these structures to add more realism to the game. For example, we can
modify our can_take/1 predicate, developed in chapter 7, so that we can only take
small objects.
can_take_s(Thing) :-
here(Room),
location_s(object(Thing, _, small,_), Room).
We can also change the error messages to reflect the two reasons why a thing
cannot be taken. To ensure that backtracking does not cause both errors to be
displayed, we will construct each clause so its message is displayed only when its
unique conditions are met. To do this, the built-in predicate not/1 is used. Its
argument is a goal, and it succeeds if its argument fails, and fails if its argument
succeeds. For example
?- not( room(office) ).
no
Note that semantically, not in Prolog means the goal cannot be successfully solved
with current logicbase of facts and rules. Here is how we use not/1 in our new
version, can_take_s/1.
can_take_s(Thing) :-<
here(Room),
location_s(object(Thing, _, small, _), Room).
can_take_s(Thing) :-
here(Room),
location_s(object(Thing, _, big, _), Room),
write('The '), write(Thing),
write(' is too big to carry.'), nl,
fail.
can_take_s(Thing) :-
here(Room),
not (location_s(object(Thing, _, _, _), Room)),
write('There is no '), write(Thing), write(' here.'), nl,
fail.
?- can_take_s(candle).
yes
?- can_take_s(table).
The table is too big to carry.
no
?- can_take_s(desk).
There is no desk here.
no
?- list_things_s(kitchen).
A small red candle, weighing 1 pounds
A small red apple, weighing 1 pounds
A small green apple, weighing 1 pounds
A big blue table, weighing 50 pounds
yes
If you are bothered by the grammatically incorrect '1 pounds', you can fix it by
adding another rule to write the weight, which would replace the direct 'writes' now
used.
write_weight(1) :-
write('1 pound').
write_weight(W) :-
W > 1,
write(W), write(' pounds').
?- write_weight(4).
4 pounds
yes
?- write_weight(1).
1 pound
yes\
Notice that we did not need to put a test, such as 'W = 1,' in the first clause. By
putting the 1 directly in the argument at the head of the clause we ensure that that
clause will only be fired when the query goal is write_weight(1). All other queries
will go to the second clause because the goal pattern will fail to unify with the head
of the first clause.
It is important, however, to put the test 'W > 1' in the second rule. Otherwise both
rules would work for a weight of 1. The first time the predicate was called would not
be a problem, but on backtracking we would get two answers if we had not included
the test.
Exercises
Adventure Game
1- Incorporate the new location into the game. Note that due to data and procedure
abstraction, we need only change the low level predicates that deal directly with
location. The higher level predicates, such as look/0 and take/1 are unaffected by
the change.
2- Use structures to enhance the customer order entry application. For example,
include a structure for each customers address.
10
Unification
The full definition of unification is similar to the one given in chapter 3, with the
addition of a recursive definition to handle data structures. This following table
summarizes the unification process.
In order to experiment with unification we will introduce the built-in predicate =/2,
which succeeds if its two arguments unify and fails if they do not. It can be written
in operator syntax as follows.
arg1 = arg2
which is equivalent to
=(arg1, arg2)
Amzi Prolog 55/78
WARNING: The equal sign (=) does not cause assignment as in most programming
languages, nor does it cause arithmetic evaluation. It causes Prolog unification.
(Despite this warning, if you are like most mortal programmers, you will be tripped
up by this difference more than once.)
Unification between two sides of an equal sign (=) is exactly the same as the
unification that occurs when Prolog tries to match goals with the heads of clauses.
On backtracking, the variable bindings are undone, just as they are when Prolog
backtracks through clauses.
The simplest form of unification occurs between two structures with no variables. In
this case, either they are identical and unification succeeds, or they are not, and
unification fails.
?- a = a.
yes
?- a = b.
no
?- location(apple, kitchen) =
location(apple, kitchen).
yes
?- location(apple, kitchen) =
location(pear, kitchen).
no
?- a(b,c(d,e(f,g))) = a(b,c(d,e(f,g))).
yes
?- a(b,c(d,e(f,g))) = a(b,c(d,e(g,f))).
no
Another simple form of unification occurs between a variable and a primitive. The
variable takes on a value that causes unification to succeed.
?- X = a.
X=a
?- 4 = Y.
Y=4
Variables can also unify with each other. Each instance of a variable has a unique
internal Prolog value. When two variables are unified to each other, Prolog notes
Amzi Prolog 56/78
that they must have the same value. In the following example, it is assumed Prolog
uses '_nn,' where 'n' is a digit, to represent unbound variables.
?- X = Y.
X = _01
Y = _01
Prolog remembers the fact that the variables are bound together and will reflect
this if either is later bound.
?- X = Y, Y = hello.
X = hello
Y = hello
?- X = Y, Y = 3, write(X).
3
X=3
Y=3
?- X = Y, tastes_yucky(X), write(Y).
broccoli
X = broccoli
Y = broccoli
When two structures with variables are unified with each other, the variables take
on values that make the two structures identical. Note that a structure bound to a
variable can itself contain variables.
?- X = a(b,c).
X = a(b,c)
?- a(b,X) = a(b,c(d,e)).
X = c(d,e)
?- a(b,X) = a(b,c(Y,e)).
X = c(_01,e)
Y = _01
Even in these more complex examples, the relationships between variables are
remembered and updated as new variable bindings occur.
food(_01,_02)
food(broccoli, apple)
X = broccoli
Y = apple
Z = food(broccoli, apple)
If a new value assigned to a variable in later goals conflicts with the pattern set
earlier, the goal fails.
The second goal failed since there is no value of Y that will allow hello to unify with
c(Y,e). The following will succeed.
If there is no possible value the variable can take on, then unification fails.
?- a(X) = a(b,c).
no
?- a(b,c,d) = a(X,X,d).
no
The last example failed because the pattern asks that the first two arguments be
the same, and they aren't.
?- a(c,X,X) = a(Y,Y,b).
no
Did you understand why this example fails? Matching the first argument binds Y to
c. The second argument causes X and Y to have the same value, in this case c. The
third argument asks that X bind to b, but it is already bound to c. No value of X and
Y will allow these two structures to unify.
The anonymous variable (_) is a wild variable, and does not bind to values. Multiple
occurrences of it do not imply equal values.
?- a(c,X,X) = a(_,_,b).
X=b
Unification occurs explicitly when the equal (=) built-in predicate is used, and
implicitly when Prolog searches for the head of a clause that matches a goal
pattern.
Exercises
Nonsense Prolog
?- a(X,c(d,X)) = a(2,c(d,Y)).
?- a(X,Y) = a(b(c,Y),Z).
11
Lists
Lists are powerful data structures for holding and manipulating groups of things.
In Prolog, a list is simply a collection of terms. The terms can be any Prolog data
types, including structures and other lists. Syntactically, a list is denoted by square
brackets with the terms separated by commas. For example, a list of things in the
kitchen is represented as
This gives us an alternative way of representing the locations of things. Rather than
having separate location predicates for each thing, we can have one location
predicate per container, with a list of things in the container.
There is a special list, called the empty list, which is represented by a set of empty
brackets ([]). It is also referred to as nil. It can describe the lack of contents of a
place or thing.
loc_list([], hall)
Unification works on lists just as it works on other data structures. With what we
now know about lists we can ask
?- loc_list(X, kitchen).
X = [apple, broccoli, crackers]
This last example is an impractical method of getting at list elements, since the
patterns won't unify unless both lists have the same number of elements.
For lists to be useful, there must be easy ways to access, add, and delete list
elements. Moreover, we should not have to concern ourselves about the number of
list items, or their order.
Two Prolog features enable us to accomplish this easy access. One is a special
notation that allows reference to the first element of a list and the list of remaining
elements, and the other is recursion.
Amzi Prolog 59/78
These two features allow us to write list utility predicates, such as member/2, which
finds members of a list, and append/3, which joins two lists together. List
predicates all follow a similar strategy--try something with the first element of a
list, then recursively repeat the process on the rest of the list.
[X | Y]
When this structure is unified with a list, X is bound to the first element of the list,
called the head. Y is bound to the list of remaining elements, called the tail.
We will now look at some examples of unification using lists. The following example
successfully unifies because the two structures are syntactically equivalent. Note
that the tail is a list.
?- [a|[b,c,d]] = [a,b,c,d].
yes
This next example fails because of misuse of the bar (|) symbol. What follows the
bar must be a single term, which for all practical purposes must be a list. The
example incorrectly has three terms after the bar.
?- [a|b,c,d] = [a,b,c,d].
no
In the previous and following examples, the tail is a list with one element.
?- [H|T] = [apples].
H = apples
T = []
The empty list does not unify with the standard list syntax because it has no head.
?- [H|T] = [].
no
Amzi Prolog 60/78
NOTE: This last failure is important, because it is often used to test for the
boundary condition in a recursive routine. That is, as long as there are elements in
the list, a unification with the [X|Y] pattern will succeed. When there are no
elements in the list, that unification fails, indicating that the boundary condition
applies.
We can specify more than just the first element before the bar (|). In fact, the only
rule is that what follows it should be a list.
Notice in the next examples how each of the variables is bound to a structure that
shows the relationships between the variables. The internal variable numbers
indicate how the variables are related. In the first example Z, the tail of the right-
hand list, is unified with [Y|T]. In the second example T, the tail of the left-hand list
is unified with [Z]. In both cases, Prolog looks for the most general way to relate or
bind the variables.
?- [X,Y|T] = [a|Z].
X=a
Y = _01
T = _03
Z = [_01 | _03]
Study these last two examples carefully, because list unification is critical in
building list utility predicates.
A list can be thought of as a head and a tail list, whose head is the second element
and whose tail is a list whose head is the third element, and so on.
?- [a|[b|[c|[d|[]]]]] = [a,b,c,d].
yes
We have said a list is a special kind of structure. In a sense it is, but in another
sense it is just like any other Prolog term. The last example gives us some insight
into the true nature of the list. It is really an ordinary two-argument predicate. The
first argument is the head and the second is the tail. If we called it dot/2, then the
list [a,b,c,d] would be
dot(a,dot(b,dot(c,dot(d,[]))))
In fact, the predicate does exist, at least conceptually, and it is called dot, but it is
represented by a period (.) instead of dot.
To see the dot notation, we use the built-in predicate display/1, which is similar to
write/1, except it always uses the dot syntax for lists when it writes to the console.
From these examples it should be clear why there is a different syntax for lists. The
easier syntax makes for easier reading, but sometimes obscures the behavior of the
predicate. It helps to keep this "real" structure of lists in mind when working with
predicates that manipulate lists.
This structure of lists is well-suited for the writing of recursive routines. The first
one we will look at is member/2, which determines whether or not a term is a
member of a list.
As with most recursive predicates, we will start with the boundary condition, or the
simple case. An element is a member of a list if it is the head of the list.
member(H,[H|T]).
This clause also illustrates how a fact with variable arguments acts as a rule.
member(X,[H|T]) :- member(X,T).
member(H,[H|T]).
member(X,[H|T]) :- member(X,T).
Note that both clauses of member/2 expect a list as the second argument. Since T
in [H|T] in the second clause is itself a list, the recursive call to member/2 works.
The query is
?- member(b, [a,b,c]).
The goal pattern fails to unify with the head of the first clause of member/2, because
Amzi Prolog 62/78
the pattern in the head of the first clause calls for the head of the list and first
argument to be identical. The goal pattern can unify with the head of the second
clause.
It succeeds because the call pattern unifies with the head of the first clause.
As with many Prolog predicates, member/2 can be used in multiple ways. If the
first argument is a variable, member/2 will, on backtracking, generate all of the
terms in a given list.
We will now trace this use of member/2 using the internal variables. Remember
that each level has its own unique variables, but that they are tied together based
on the unification patterns between the goal at one level and the head of the clause
on the next level.
In this case the pattern is simple in the recursive clause of member. The head of
the clause unifies X with the first argument of the original goal, represented by _0
in the following trace. The body has a call to member/2 in which the first argument
is also X, therefore causing the next level to unify with the same _0.
The query is
?- member(X,[a,b,c]).
The goal succeeds by unification with the head of the first clause, if X = a.
Further backtracking causes an attempt to find a member of the empty list. The
empty list does not unify with either of the list patterns in the member/2 clauses, so
the query fails back to the beginning.
Another very useful list predicate builds lists from other lists or alternatively splits
lists into separate pieces. This predicate is usually called append/3. In this
predicate the second argument is appended to the first argument to yield the third
argument. For example
?- append([a,b,c],[d,e,f],X).
X = [a,b,c,d,e,f]
It is a little more difficult to follow, since the basic strategy of working from the
head of the list does not fit nicely with the problem of adding something to the end
of a list. append/3 solves this problem by reducing the first list recursively.
The boundary condition states that if a list X is appended to the empty list, the
resulting list is also X.
append([],X,X).
The recursive condition states that if list X is appended to list [H|T1], then the head
of the new list is also H, and the tail of the new list is the result of appending X to
the tail of the first list.
Amzi Prolog 64/78
append([H|T1],X,[H|T2]) :-
append(T1,X,T2).
append([],X,X).
append([H|T1],X,[H|T2]) :-
append(T1,X,T2).
Real Prolog magic is at work here, which the trace alone does not reveal. At each
level, new variable bindings are built, that are unified with the variables of the
previous level. Specifically, the third argument in the recursive call to append/3 is
the tail of the third argument in the head of the clause. These variable relationships
are included at each step in the annotated trace shown in Figure 11.3.
The query is
?- append([a,b,c],[d,e,f],X).
1-1 CALL append([a,b,c],[d,e,f],_0)
X = _0
2-1 CALL append([b,c],[d,e,f],_5)
_0 = [a|_5]
3-1 CALL append([c],[d,e,f],_9)
_5 = [b|_9]
4-1 CALL append([],[d,e,f],_14)
_9 = [c|_14]
By making all the substitutions of the variable relationships, we can see that at this
point X is bound as follows (thinking in terms of the dot notation for lists might make
append/3 easier to understand).
X = [a|[b|[c|_14]]]
We are about to hit the boundary condition, as the first argument has been reduced
to the empty list. Unifying with the first clause of append/3 will bind _14 to a value,
namely [d,e,f], thus giving us the desired result for X, as well as all the other
intermediate variables. Notice the bound third arguments at each level, and compare
them to the variables in the call ports above.
Like member/2, append/3 can also be used in other ways, for example, to break
lists apart as follows.
?- append(X,Y,[a,b,c]).
X = []
Y = [a,b,c] ;
X = [a]
Y = [b,c] ;
X = [a,b]
Amzi Prolog 65/78
Y = [c] ;
X = [a,b,c]
Y = [] ;
no
Now that we have tools for manipulating lists, we can use them. For example, if we
choose to use loc_list/2 instead of location/2 for storing things, we can write a new
location/2 that behaves exactly like the old one, except that it computes the answer
rather than looking it up. This illustrates the sometimes fuzzy line between data
and procedure. The rest of the program cannot tell how location/2 gets its results,
whether as data or by computation. In either case it behaves the same, even on
backtracking.
location(X,Y):-
loc_list(List, Y),
member(X, List).
In the game, it will be necessary to add things to the loc_lists whenever something
is put down in a room. We can write add_thing/3 which uses append/3. If we call it
with NewThing and Container, it will provide us with the NewList.
Testing it gives
However, this is a case where the same effect can be achieved through unification
and the [Head|Tail] list notation.
We can simplify it one step further by removing the explicit unification, and using
the implicit unification that occurs at the head of a clause, which is the preferred
form for this type of predicate.
add_thing3(NewTh, Container,[NewTh|OldList]) :-
loc_list(OldList, Container).
put_thing(Thing,Place) :-
retract(loc_list(List, Place)),
asserta(loc_list([Thing|List],Place)).
Whether you use multiple logicbase entries or lists for situations, such as we have
with locations of things, is largely a matter of style. Your experience will lead you to
one or the other in different situations. Sometimes backtracking over multiple
predicates is a more natural solution to a problem and sometimes recursively
dealing with a list is more natural.
You might find that some parts of a particular application fit better with multiple
facts in the logicbase and other parts fit better with lists. In these cases it is useful
to know how to go from one format to the other.
Going from a list to multiple facts is simple. You write a recursive routine that
continually asserts the head of the list. In this example we create individual facts in
the predicate stuff/1.
break_out([]).
break_out([Head | Tail]):-
assertz(stuff(Head)),
break_out(Tail).
?- stuff(X).
X = pencil ;
X = cookie ;
X = snow ;
no
Transforming multiple facts into a list is more difficult. For this reason most Prologs
provide built-in predicates that do the job. The most common one is findall/3. The
arguments are
arg1
A pattern for the terms in the resulting list
arg2
A goal pattern
arg3
The resulting list
findall/3 automatically does a full backtracking search of the goal pattern and
stores each result in the list. It can recover our stuff/1 back into a list.
Fancier patterns are available. This is how to get a list of all the rooms connecting
to the kitchen.
Amzi Prolog 67/78
?- findall(X, connect(kitchen, X), L).
L = [office, cellar, 'dining room']
The pattern in the first argument can be even fancier and the second argument can
be a conjunction of goals. Parentheses are used to group the conjunction of goals in
the second argument, thus avoiding the potential ambiguity. Here findall/3 builds a
list of structures that locates the edible things.
Exercises
List Utilities
2- Because write/1 only takes a single argument, multiple 'writes' are necessary for
writing a mixed string of text and variables. Write a list utility respond/1 which
takes as its single argument a list of terms to be written. This can be used in the
game to communicate with the player. For example
3- Lists with a variable tail are called open lists. They have some interesting
properties. For example, member/2 can be used to add items to an open list.
Experiment with and trace the following queries.
?- member(a,X).
?- member(b, [a,b,c|X]).
?- member(d, [a,b,c|X]).
?- OpenL = [a,b,c|X], member(d, OpenL), write(OpenL).
Nonsense Prolog
?- [a,b,c,d] = [H|T].
?- [a,[b,c,d]] = [H|T].
?- [] = [H|T].
?- [a] = [H|T].
?- [apple,3,X,'What?'] = [A,B|Z].
?- [[a,b,c],[d,e,f],[g,h,i]] = [H|T].
?- [a(X,c(d,Y)), b(2,3), c(d,Y)] = [H|T].
Genealogical Logicbase
parent(p1,p2).
Amzi Prolog 68/78
parent(p2,p3).
parent(p3,p4).
parent(p4,p5).
ancestor(A,D,[A]) :- parent(A,D).
ancestor(A,D,[X|Z]) :-
parent(X,D),
ancestor(A,X,Z).
?- ancestor(a2,a3,X).
?- ancestor(a1,a5,X).
?- ancestor(a5,a1,X).
?- ancestor(X,a5,Z).
Expert System
8- Lists provide a convenient way to provide a simple menu capability to our expert
system. We can replace the 'ask' predicate with menuask/3 where appropriate.
menuask/3 will ask the player to select an item from a menu. The format is
For example
12
Operators
functor(arg1,arg2,...,argN).
This is the ONLY data structure in Prolog. However, Prolog allows for other ways to
syntactically represent the same data structure. These other representations are
sometimes called syntactic sugaring. The equivalence between list syntax and the
dot (.) functor is one example. Operator syntax is another.
Each arithmetic operator is an ordinary Prolog functor, such as -/2, +/2, and -/1.
The display/1 predicate can be used to see the standard syntax.
?- display(2 + 2).
+(2,2)
Amzi Prolog 69/78
?- display(3 * 4 + 6).
+(*(3,4),6)
?- display(3 * (4 + 6)).
*(3,+(4,6))
You can define any functor to be an operator, in which case the Prolog listener will
be able to read the structure in a different format. For example, if location/2 was an
operator we could write
instead of
location(apple, kitchen).
infix
Example: 3 + 4
prefix
Example: -7
postfix
Example: 8 factorial
They have a number representing precedence which runs from 1 to 1200. When a
term with multiple operators is converted to pure syntax, the operators with higher
precedences are converted first. A high precedence is indicated by a low number.
Operators are defined with the built-in predicate op/3, whose three arguments are
precedence, associativity, and the operator name.
For our current purposes, we will again rework the location/2 predicate and rename
it is_in/2 to go with its new look, and we will represent rooms in the structure
room/1.
is_in(apple, room(kitchen)).
?- op(35,xfx,is_in).
?- apple is_in X.
X = room(kitchen)
or
Amzi Prolog 70/78
?- X is_in room(kitchen).
X = apple
To verify that Prolog treats both syntaxes the same we can attempt to unify them.
Let's now make room/1 a prefix operator. Note that in this case the associativity
pattern fx is used to indicate the functor comes before the argument. Also we chose
a precedence (33) higher (higher precedence has lower number) than that used for
is_in (35) in order to nest the room structure inside the is_in structure.
?- op(33,fx,room).
?- apple is_in X.
X = room kitchen\
CAUTION: If you mix up the precedence (easy to do) you will get strange bugs. If
room/1 had a lower precedence (higher number) than is_in/2, then the structure
would be
room(is_in(apple, kitchen))
Not only doesn't this capture the information as intended, it also will not unify the
way we want.
?- op(33,xf,turned_on).
Amzi Prolog 71/78
We can now say
flashlight turned_on.
and
Operators are useful for making more readable data structures in a program and
for making quick and easy user interfaces.
In our command-driven Nani Search, we use a simple natural language front end,
which will be described in the last chapter. We could have alternatively made the
commands operators so that
goto(kitchen)
becomes goto kitchen.
turn_on(flashlight)
becomes turn_on flashlight.
take(apple)
becomes take apple.
It's not natural language, but it's a lot better than parentheses and commas.
We have seen how the precedence of operators affects their translation into
structures. When operators are of equal precedence, the Prolog reader must decide
whether to work from left to right, or right to left. This is the difference between
right and left associativity.
The same pattern used for precedence is used for associativity with the additional
character y. The options are
Infix:
xfx non-associative
xfy right to left
yfx left to right
Prefix
fx non-associative
fy left to right
Postfix:
xf non-associative
yf right to left
?- op(35,xfy,is_in).
yes
?- op(35,yfx,is_in).
yes
Many built-in predicates are actually defined as infix operators. That means that
rather than following the standard predicate(arg1,arg2) format, the predicate can
appear between the arguments as
The arithmetic operators we have seen already illustrate this. For example +, -, *,
and / are used as you would expect. However, it is important to understand that
these arithmetic structures are just structures like any others, and do not imply
arithmetic evaluation. 3 + 4 is not the same as 7 any more than plus(3,4) is or
likes(3,4). It is just +(3,4).
Only special built-in predicates, like is/2, actually perform an arithmetic evaluation
of an arithmetic expression. As we have seen, is/2 causes the right side to be
evaluated and the left side is unified with the evaluated result.
This is in contrast to the unification (=) predicate, which just unifies terms without
evaluating them.
?- X is 3 + 4.
X=7
?- X = 3 + 4.
X=3+4
?- 10 is 5 * 2.
yes
?- 10 = 5 * 2.
no
?- X is 3 * 4 + (6 / 2).
X = 15
?- X = 3 * 4 + (6 / 2).
X = 3 * 4 + (6 / 2)
Amzi Prolog 73/78
The operator predicates can also be written in standard notation.
?- X is +(*(3,4) , /(6,2)).
X = 15
?- 3 * 4 + (6 / 2) = +(*(3,4),/(6,2)).
yes
To underscore that these arithmetic operators are really ordinary predicates with no
special meaning unless being evaluated by is/2, consider
?- X = 3 * 4 + likes(john, 6/2).
X = 3 * 4 + likes(john, 6/2).
?- X is 3 * 4 + likes(john, 6/2).
error
We have seen that Prolog programs are composed of clauses. These clauses are
simply Prolog data structures written with operator syntax. The functor is the neck
(:-) which is defined as an infix operator. There are two arguments.
:-(Head, Body).
The body is a data structure with the functor 'and' represented by a comma (,). The
body looks like
,(goal1, ,(goal2,,goal3))
Note the ambiguous use of the comma (,) as a conjunctive operator and as a
separator of arguments in a Prolog structure. This can cause confusion in Prolog
programs that manipulate Prolog clauses. It might have been clearer if an
ampersand (&) was used instead of a comma for separating goals. Then the above
pattern would be
But that is not how it was done, so the two forms are
The arithmetic operators are often used by Prolog programmers to syntactically join
related terms. For example, the write/1 predicate takes only one argument, but
operators give an easy way around this restriction.
The slash (/) can be used the same way. In addition, some Prologs define the colon
(:) as an operator just for this purpose. It can improve readability by removing
Amzi Prolog 74/78
some parentheses. For example, the complex structures for defining things in the
game can be syntactically represented with the colon as well.
The pattern matching is the same as always, but instead of size(small) we use the
pattern size:small, which is really :(size,small).
Exercises
Adventure Game
1- Define all of the Nani Search commands as operators so the current version of
the game can be played without parentheses or commas.
Genealogical Logicbase
13
Cut
The cut effectively tells Prolog to freeze all the decisions made so far in this
predicate. That is, if required to backtrack, it will automatically fail without trying
other alternatives.
We will first examine the effects of the cut and then look at some practical reasons
to use it.
We will write some simple predicates that illustrate the behavior of the cut, first
adding some data to backtrack over.
data(one).
data(two).
data(three).
Here is the first test case. It has no cut and will be used for comparison purposes.
cut_test_a(X) :-
data(X).
cut_test_a('last clause').
cut_test_b(X) :-
data(X),
!.
cut_test_b('last clause').
Note that it stops backtracking through both the data/1 subgoal (left), and the
cut_test_b parent (above).
cut_test_c(X,Y) :-
data(X),
!,
data(Y).
cut_test_c('last clause').
Note that the cut inhibits backtracking in the parent cut_test_c and in the goals to
the left of (before) the cut (first data/1). The second data/1 to the right of (after)
the cut is still free to backtrack.
Performance is the main reason to use the cut. This separates the logical purists
from the pragmatists. Various arguments can also be made as to its effect on code
readability and maintainability. It is often called the 'goto' of logic programming.
You will most often use the cut when you know that at a certain point in a given
predicate, Prolog has either found the only answer, or if it hasn't, there is no
answer. In this case you insert a cut in the predicate at that point.
Similarly, you will use it when you want to force a predicate to fail in a certain
situation, and you don't want it to look any further.
We will now introduce to the game the little puzzles that make adventure games
fun to play. We will put them in a predicate called puzzle/1. The argument to
puzzle/1 will be one of the game commands, and puzzle/1 will determine whether
or not there are special constraints on that command, reacting accordingly.
We will see examples of both uses of the cut in the puzzle/1 predicate. The
behavior we want is
The puzzle in Nani Search is that in order to get to the cellar, the game player
needs to both have the flashlight and turn it on. If these criteria are met we know
there is no need to ever backtrack through puzzle/1 looking for other clauses to try.
For this reason we include the cut.
puzzle(goto(cellar)):-
have(flashlight),
turned_on(flashlight),
!.
If the puzzle constraints are not met, then let the player know there is a special
problem. In this case we also want to force the calling predicate to fail, and we
don't want it to succeed by moving to other clauses of puzzle/1. Therefore we use
the cut to stop backtracking, and we follow it with fail.
puzzle(goto(cellar)):-
write('It''s dark and you are afraid of the dark.'),
!, fail.
The final clause is a catchall for those commands that have no special puzzles
associated with them. They will always succeed in a call to puzzle/1.
puzzle(_).
For logical purity, it is always possible to rewrite the predicates without the cut.
This is done with the built-in predicate not/1. Some claim this provides for clearer
code, but often the explicit and liberal use of 'not' clutters up the code, rather than
clarifying it.
Amzi Prolog 77/78
When using the cut, the order of the rules becomes important. Our second clause
for puzzle/1 safely prints an error message, because we know the only way to get
there is by the first clause failing before it reached the cut.
The third clause is completely general, because we know the earlier clauses have
caught the special cases.
If the cuts were removed from the clauses, the second two clauses would have to
be rewritten.
puzzle(goto(cellar)):-
not(have(flashlight)),
not(turned_on(flashlight)),
write('Scared of dark message'),
fail.
puzzle(X):-
not(X = goto(cellar)).
It is interesting to note that not/1 is defined using the cut. It also uses call/1,
another built-in predicate that calls a predicate.
In the next chapter we will see how to add a command loop to the game. Until then
we can test the puzzle predicate by including a call to it in each individual
command. For example
goto(Place) :-
puzzle(goto(Place)),
can_go(Place),
move(Place),
look.
Assuming the player is in the kitchen, an attempt to go to the cellar will fail.
?- goto(cellar).
It's dark and you are afraid of the dark.
no
?- goto(office).
You are in the office...
Then if the player takes the flashlight, turns it on, and return to the kitchen, all
goes well.
?- goto(cellar).
You are in the cellar...
Exercises
Adventure Game
1- Test the puzzle/1 predicate by setting up various game situations and seeing
how it responds. When testing predicates with cuts you should always use the
Amzi Prolog 78/78
semicolon (;) after each answer to make sure it behaves correctly on backtracking.
In our case puzzle/1 should always give one response and fail on backtracking.
Expert System
3- Modify the ask and menuask predicates to use cut to replace the use of not.
4- Modify the good_customer rules to use cut to prevent the search of other cases
once we know one has been found.