Prolog

advertisement
Prolog Programming
Lecture Module 13
Objective
●
●
●
●
●
●
●
●
●
●
●
What is Prolog?
Prolog program
Syntax of Prolog
Prolog Control Strategy
Execution of Prolog Query
Programming Techniques
Termination
Rule Order
Goal Order
Rule Redundancy
Iterative Programming
What is Prolog?
● Prolog is acronym of PROgramming in LOGic.
● Prolog program is sequence of rules and facts.
● Prolog Rule for grandfather is defined as:
gfather(X, Y) :- father(X, Z), parent(Z, Y).
father(james, robert).
Glimpse of Prolog Program
/* Rules: “X is grand father of Y if X is father of Z who is parent of Y */
gfather(X, Y) :father(X, Z), parent(Z, Y).
parent(X, Y) :father(X, Y).
parent(X, Y) :mother(X, Y).
/*Facts “James is father of robert” */
father(james, robert).
father(mike, william).
father(william, james).
father(robert, hency).
father(robert, cris).
/* Goals ‘ Is abraham a grand father of mike?’*/
?- gfather(abraham, mike).
?- gfather(X, Y).
?- father(X, Y), father(Y, mike).
(1)
(2)
(3)
(4)
(5)
(6)
(7)
(8)
Back
General Syntax of Prolog
● The rules and facts in Prolog are terminated by full stop (.).
● Goal can be included in the program or given at the prompt.
● Goal can be single or conjunction of sub-goal(s) terminated by full
●
●
●
●
stop.
Constants are numerals and symbols such as, 4, mary, etc.
String is a sequence of characters and is enclosed within single
quotes e.g., 'This is a string'.
Predicate names must start with alphabet and are formed by
using lower case letters, numerals and underscore ( _ ).
Variable names are formed similar to predicate names but it must
start with upper case letter. Normally, we use X, Y, Z, ... for
variables.
Prolog Control Strategy
● Prolog contains three basic control strategies.
− Forward movement
− Matching (Unification)
− Backward movement (Backtracking)
● Prolog uses depth first search strategy.
Forward Movement
● Choose a rule by
− searching sequentially in the program from top to
bottom whose head matches with the goal with
possible unifier.
− Remember the position of the matched rule.
− Join the rule body in front of the sequence of sub
goals to be solved.
● Repeat until either goal is satisfied or is
failed.
Forward Movement – Cont…
● Consider the following query consisting of n sub-goals.
?-
G1, ... , Gi-1, Gi, ... , Gn
● Starts executing or satisfying G1
− If G1 is satisfied using some rule, put a place marker so
that next rule for G1 will be searched after the marker.
− Continue in forward direction to satisfy G2 and if it is
satisfied then continue with common bindings of the
variables in sub-goals so far satisfied.
− Such a movement is called forward
Unification
● Unification is a process of matching or finding the
most general unifier.
− Constant matches with the same constant.
− A variable can match with any constant or any another
variable.
● Examples
− love(X, sita) unifies with love(ram, sita) with mgu as
{X / ram}.
− love(X, Y) unify with love(ram, sita) with mgu as
{X / ram, Y / sita} respectively.
Backtracking
● Backtracking refers to backtrack in search
process for a solution.
● In Prolog, backtracking takes place in two
situations.
− First when a sub goal fails and
− Other when the last sub goal succeeds, it
backtracks to find alternative solutions.
Types of backtracking
● Prolog has two kinds of backtracking.
− Shallow and
− Deep backtracking
● Both backtrackings take place in conjunction with
each other.
● Shallow backtracking occurs
− when alternative definition of the goal is tried.
− Suppose G1,.., Gi-1 have been satisfied with
some common bindings of the variables in subgoals.
− While satisfying Gi., if it fails for some rule then try
another rule of Gi
− This is called shallow backtracking to rules of the
same sub-goal
Backtracking Cont…
● Deep backtracking occurs
− When sub-goal Gi fails for all rules of Gi then backtrack
to Gi-1.
− This backtracking is called deep backtracking.
− Remove the bindings created earlier to satisfy Gi-1.
− Try to satisfy Gi-1 again by using an alternative rule, if it
exists, from last place marker of Gi-1. and continue the
process.
− If sub goal Gi-1 is satisfied then move forward to Gi,
else backtrack to Gi-2.
− The process of backtracking continues till we reach
G1.
− If G1 fails for all possible definitions, then entire query
fails.
Backtracking Cont…
● If entire conjunction of sub-goals succeed, then the
solution is displayed.
● For alternative solution, the sub goal Gn is tried to be
satisfied with alternative rule, if it exist, after last
place marker (Shallow backtracking) of Gn .
● Otherwise it backtracks to previous sub goal Gn-1
(Deep backtracking).
● Process is repeated till all possible solutions are
generated.
Prolog Query
● Two types of query
− Ground query
 No variable in the argument
 Example:
? gfather(ram, shyam) – Is ram a grand father of shyam?
 Answer to ground queries are Yes or No
 Proof tree is generated while solving the query.
− Non ground query
 Atleast one argument is a variable
 Example:
? gfather(X, shyam) – Who is a grand father of shyam?
 Answer to such queries are the values bound to the variables
in the query.
 Search tree is generated using DFS to find all possible
solutions.
Execution of ground query
Ground query:
Prolog Program
?- gfather(james, hency).
Proof tree:
?- gfather(james, hency).
(1)
?- father(james, Z), parent(Z, hency).
(4)
{ Z = robert }
?- parent(robert, hency).
(2)
?- father(robert, hency).
(7)
succeeds
Answer: Yes
Execution of Non ground query
Non-Ground query:
?- gfather(william, X).
Search tree:
Prolog Program
?- gfather(william, X).
(1)
?- father(william, Z), parent(Z, X).
(6)
{Z = james}
?- parent(james, X).
(2)
(3)
?- father(james, X).
?- mother(james, X).
(4) {X = robert}
succeeds
fails
Answer: X = robert
Programming Techniques
● Recursion is one of the most important execution
control techniques in Prolog.
● Iteration is another important
technique.
● Prolog does not support iteration.
● It can be achieved by
programming
− making recursive call to be the last sub goal in the rule
− using the concept of accumulators which are special types
of arguments used for holding intermediate results.
● The issues of termination, rule order, goal order and
redundancy are also important.
Recursion in Prolog
● Consider a classical example of computing factorial
using recursion.
● Recurrence relation or mathematical definition for
computing factorial of positive integer is:
1,
if n = 0
FACTORIAL (n) =
n * FACTORIAL (n-1),
otherwise
● In Prolog, program for computing factorial of a
positive number is written using above definition.
● Factorial predicate contains two arguments; one as
input and another as output.
Recursion - Cont..
/*fact(N, R) – succeeds if R is unified with the factorial of
N */
factorial(0, 1).
(1)
factorial(N, R) :- N1 is N–1, factorial(N1, R1), R is N * R1. (2)
● Rule (1) states that factorial of 0 is 1 which is a
terminating clause.
● Rule (2) is a recursive rule which states that factorial of
N is calculated by multiplying N with factorial of (N-1).
Goal: ?- factorial(3, R).
Execution of factorial query
Search tree:
?- factorial(3, R).
(2)
{N = 3}
?- N1 is N-1, factorial(N1, R1), R is N * R1
(2)
{N1 = 2}
?-N2 is N1-1,factorial(N2,R2),R1 is N1*R2,R is N*R1.
(2)
{N2 = 1}
?-N3 is N2-1, factorial(N3, R3), R2 is N2 * R3….
(1)
{N3 = 0, R3 = 1}
?- R2 is 1 * R3 , R1 is N1 * R2, R is N * R1.
Succeeds {R2 = 1, R1 = 2, R = 6}
List Manipulation in Prolog
 List is a common and very useful recursive data
structure in non numeric programming.
 In Prolog, a list is represented by
−
−
−
elements separated by comma and enclosed in
square brackets.
Examples: A = [2, 4, 6, 8], B = [[a, b], [c], [p]].
Empty list with no elements is represented by [ ]
and is useful for terminating recursive programs for
list.
Membership function
● An element is a member of a list if it matches with the
head or if it is contained in the tail of the list.
− The fact that it matches with the head of a list is written as:
mem_list(X, [X | _ ]).
− The rule that if an element does not match with the head,
then it may be in the tail is written as: mem_list(X, [ _ | Y]) :mem_list(X, Y).
mem_list(X, [X | _ ]).
mem_list(X, [_ | Y]) :- mem_list(X, Y).
− Underscore is used for non care entry.
(1)
(2)
Ground Query
Goal: ?- mem_list(d, [a, b, c, d, e, f]).
Proof tree:
?- mem_list(d, [a, b, c, d, e, f]).
(2)
{X = d, Y = [b, c, d, e, f]}
?- mem_list(d, [b, c, d, e, f]).
(2)
{X = d, Y1 = [c, d, e, f]}
?- mem_list(d, [c, d, e, f]).
(2)
{X = d, Y2 = [d, e, f]}
?- mem_list(d, [d, e, f]).
(1)
succeeds
Ans: Yes
Non ground Query
Goal: ?- mem_list(X, [a, b, c]).
Search tree:
?- mem_list(X, [a, b, c]).
(1)
(2)
{X = a}
{Y = [b, c]}
succeeds
?- mem_list(X, [b, c]).
(1)
(2)
{X = b}
{Y = [c]}
succeeds
?- mem_list(X, [c]).
(1)
(2)
{X = c}
{Y = []}
succeeds
?- mem_list(X, []).
Answer:X = a; X = b; X = c
fails
Termination
 Depth first traversal of Prolog has a serious problem. If
the search tree of a goal with respect to a program
contains an infinite branch, the computation will not
terminate.
 Prolog may fail to find a solution of a goal, even if the
goal has a finite computation.
 Non termination
arises because of the following
reasons:
− Left recursive rules: The rule having a recursive goal
as the first sub goal in the body.
Example
● Consider acyclic directed graph
p
q
r
s
t
● Graph G is represented by set of connected edges.
G = { edge(p, q), edge(q, r), edge(q, s), edge(s, t)}
● Write Prolog Program to check whether there is a
route from one node to another node.
Program
● Route from node A to B is defined as follows:
− route from A to B if there is a root from A some node C and an
edge from C to B.
− route from A to B if there is a direct edge from A to B.
● Prolog Program
route(A, B) :- route(A, C), edge(C, B).
route(A, B) :- edge(A, B).
edge(p, q).
edge(q, r).
edge(q, s).
edge(s, t).
(1)
(2)
back
Query: Check whether there exist a route between nodes p and t .
Proof tree
Goal: ?- route(p, t).
Proof tree:
?- route(p, t).
(1)
?- route(p, X), edge(X, t)
(1)
?- route(p, Y), edge(Y, X), edge(X, t).
Non terminating
● Non terminating program because of left recursion.
● Better way of writing a rule is to write recursive call as the last
call.
route(A, B) :- edge(A, C), route(C, B).
Circular Definition
● Non termination occurs because of Circular definition
of the rules.
● For example, ground and nonground queries will not
terminate with respect to the following program.
teacher(X, Y) :- student(Y, X).
student(X, Y) :- teacher(Y, X).
student(robert, mike).
student(john, robert).
Example:
?- teacher(mike, robert)
(1)
?- student(robert, mike)
(2)
?- teacher(mike, robert)
Non termination
(1)
(2)
Rule Order
● A rule order is very important issue in Prolog.
● Change of the order of rules in a program, the
branches in any search tree for a goal to be solved
with that program get permuted.
● It may affect the efficiency of executing goal.
● Since the search tree is traversed using depth first
approach, the order of solutions will be different.
− Check the order of solutions obtained if rules are
interchanged in a Prolog program (go).
Goal Order
● Goal order in Prolog is more significant than rule order.
● It may affect the termination because
− subsequent sub goals have different bindings and hence
different search trees.
● Consider the following recursive rules.
ancester(X, Y) :- parent(X, Z), ancester(Z, Y).
route(X, Y)
:- edge(X, Z), route(Z, Y).
● If the sub goals in the body of above rules are
swapped, then the rules becomes left recursive and all
the computations will become non terminating.
● It also affects the amount of search the program does
in solving a query .
Redundancy
● Redundancy of the solution to the queries is another
●
●
●
●
important issue in Prolog programs.
In Prolog, each possible deduction means an extra
branch in the search tree.
The bigger the search tree, higher the computation
time.
It is desirable in general to keep the size of the search
tree as small as possible.
Redundant program at times may cause exponential
increase in runtime in the event of backtracking.
Redundancy – Cont..
● There
are various reasons
redundancy gets introduced.
by
which
− The same case is covered by several rules
− Redundancy in computation appears in the
program by not handling special
separately.
− See examples given in the book.
cases
Iteration in Prolog
 Prolog does not have storage variables that can
− hold intermediate results of the computation
− be modified as the computation progress.
 To implement iterative algorithm one requires the
storage of intermediate results, Prolog predicates are
augmented
with
additional
arguments
called
accumulators.
 These are similar to local variables in procedural
languages.
 The programs written using accumulators are iterative
in nature.
 In iterative form of a program, make the recursive call
as the last call if possible.
Iteration – cont…
● Prolog program for factorial using accumulators.
● Accumulators I and T are used for storing the counter
value and intermediate computations
fact (N, R)
:fact1(N, R, I, T) :-
fact1(N, R, 0, 1).
I < N, J is I + 1, T1 is J * T,
fact1(N, R, J, T1).
fact1(N, R, N, R).
(1)
(2)
(3)
● In prolog, predicates can be overloaded. So we can
also use ‘fact’ instead of fact1.
Search tree
Goal: ?- fact (3, R).
Search tree:
?- fact(3, R).
(1)
?- fact1(3, R, 0, 1).
(2)
?- 0 < 3, J is 1, T1 is 1 * 1, fact1(3, R, 1, 1).
(2)
?- 1 < 3, J is 2, T1 is 2 * 1, fact1(3, R, 2, 2).
(2)
?- 2 < 3, J is 3, T1 is 3 * 2, fact1(3, R, 3, 6).
(2)
fails
Answer: R = 6
(3)
{R = 6}
succeeds
Download