-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathunify.pl
139 lines (96 loc) · 6.32 KB
/
unify.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
:- op(20,xfy,?=).
% Prédicats d'affichage fournis
% set_echo: ce prédicat active l'affichage par le prédicat echo
set_echo :- assert(echo_on).
% clr_echo: ce prédicat inhibe l'affichage par le prédicat echo
clr_echo :- retractall(echo_on).
% echo(T): si le flag echo_on est positionné, echo(T) affiche le terme T
% sinon, echo(T) réussit simplement en ne faisant rien.
echo(T) :- echo_on, !, write(T).
echo(_).
/* occur_check(V,T) permet de vérifier si la variable V est contenue dans le terme T */
occur_check(V,T) :- V==T -> !; compound(T), functor(T,N,A), ocheck(V,T,A),!.
ocheck(V,T,A) :- A==1 -> arg(1,T,X), occur_check(V,X); arg(A,T,X), occur_check(V,X); A2 is (A-1), ocheck(V,T,A2),!.
% Rappel : E = Equation. R = Règle de transformation. P = Système d'équations. Q = Système d'équation
% Unifie(P,S) où S = choix_premier
/* regle(E,R) : Détermine la règle de transformation R qui peut s'appliquer à l'équation E. */
regle((X ?= T), clash) :- compound(X), compound(T), functor(X,_,N1), functor(T,_,N2), N1 \== N2,!;
compound(X), compound(T), functor(X,A1,_), functor(T,A2,_), A1 \== A2,!.
regle((_ ?= T),rename) :- var(T),!.
regle((_ ?= T),simplify) :- atomic(T),!.
regle((X ?= T), check) :- X \== T, occur_check(X,T),!.
regle((X ?= T), expand) :- var(X), compound(T), not(occur_check(X,T)),!.
regle((X ?= T), decompose) :- compound(X), compound(T), functor(X,A1,N1), functor(T,A2,N2),(A1==A2),(N1==N2),!.
regle((T ?= _), orient) :- nonvar(T),!.
/* reduit(R,E,P,Q) : Transforme le système d'équations P en le système d'équations Q par application de la règle de transformation R à l'équation E.*/
reduit(rename,(X ?= T), P;S, A;[X = T|B]) :- echo(system : [X ?= T|P]), nl, echo(rename : X ?= T), nl, substitution(X,T,P,A), substitution(X,T,S,B).
reduit(simplify,(X ?= T), P;S, A;[X = T|B]) :- echo(system : [X ?= T|P]), nl, echo(simplify : X ?= T), nl, substitution(X,T,P,A), substitution(X,T,S,B).
reduit(check,_,_,_) :- echo(system : [X ?= T|P]), nl, echo(check : X ?= T), nl, fail.
reduit(expand,(X ?= T), P;S, A;[X = T|B]) :- echo(system : [X ?= T|P]), nl, echo(expand : X ?= T), nl,
substitution(X,T,P,A), substitution(X,T,S,B).
reduit(orient,(X ?= T), P;S, [T ?= X|P];S) :- echo(system : [X ?= T|P]), nl, echo(orient : X ?= T), nl.
reduit(decompose,(X ?= T), P;S, Q;S) :- echo(system : [X ?= T|P]), nl, echo(decompose : X ?= T), nl,
decomposer(X,T,L), concat(L,P,Q).
reduit(clash,_,_,_) :- echo(system : [X ?= T|P]), nl, echo(clash : X ?= T), nl, fail.
/* Transorfme une liste d'équations (3ème argument) en remplaçant X par T, pour retourner le résultat dans une liste modifiée (4ème argument) avec les valeurs remplacées */
substitution(_,_,[],[]).
substitution(X,T,[A ?= B|P],[A2 ?= B2|P2]) :- substitution_terme(X,T,A,A2), substitution_terme(X,T,B,B2), substitution(X,T,P,P2).
substitution(X,T, [ A = B |P], [A2 = B2 | P2]) :- substitution_terme(X,T,A,A2), substitution_terme(X,T,B,B2), substitution(X,T,P,P2).
/* Remplace le terme A par T si A est égal à X. Si ce n'est pas le cas, et que A n'est pas composé, rien n'est remplacé. Si A n'est pas égal à X mais que A est tout de même composé, on remplace X par T.*/
substitution_terme(X,T,A,T) :- A==X.
substitution_terme(X,_,A,A) :- not(compound(A)), A\==X.
substitution_terme(X,T,A,B) :- A\==X, compound(A), substitutionSousTerme(X,T,A,B).
/* Crée le terme B à partir de A où l'on remplace les X par T. */
substitutionSousTerme(X,T,A,B) :- functor(A,N,_), creerListe(X,T,A,0,L), B =..[N|L].
/* On remplace directement X par T. */
creerListe(_,_,F,N,[]) :- functor(F,_,A), (N>=A).
/* Lorsque E n'est pas composé. */
creerListe(X,T,F,N,[E|L]) :- N2 is N+1, arg(N2,F,E), (E\==X), not(compound(E)), creerListe(X,T,F,N2,L).
/* E est composé, on rappelle substitutionSousTerme sur E pour y substituer les X aux T */
creerListe(X,T,F,N,[G|L]) :- N2 is N+1, arg(N2,F,E), (E\==X), compound(E), substitutionSousTerme(X,T,E,G), creerListe(X,T,F,N2,L).
/* On avance dans la liste. */
creerListe(X,T,F,N,[T|L]) :- N2 is N+1, arg(N2,F,E), (E==X), creerListe(X,T,F,N2,L).
/* A partir d'une équation de la forme f(X1,...,Xn) ?= f(Y1,...,Yn), on décompose élément par élément l'équation pour renvoyer une liste d'égalités de la forme [X1 ?= Y1, ... , Xn ?= Yn] */
decomposer(T,X,L) :- decomposition(T,X,0,L).
decomposition(T,_,N,[]) :- functor(T,_,N).
decomposition(T,X,N,[A ?= B|L]) :- N2 is N+1, arg(N2,T,A), arg(N2,X,B), decomposition(T,X,N2,L).
/* Concaténation de deux listes */
concat([],X,X).
concat([X|P],Y,[X|Q]) :- concat(P,Y,Q).
/* Force l'identification => c'est ce qui permet à la fin d'afficher les valeurs des variables et non le nom que leur a donné le système */
ident([]).
ident([ X = X | T]) :- ident(T).
/* Poids de chaque règle selon l'énoncé*/
poidRegle(clash,5).
poidRegle(check,5).
poidRegle(rename,4).
poidRegle(simplify,4).
poidRegle(orient,3).
poidRegle(decompose,2).
poidRegle(expand,1).
/* A partir d'une liste d'équations, on renvoie l'équation dotée de la règle de plus haut poids
et la liste des équations restantes (non triées) ainsi que la règle associée */
choix_pondere([H|T],Q,F,R) :- regle(H,RE), poidRegle(RE,W), selectEqu1(T,Q,F,H,W,R,RE).
/*liste param :
1 : liste d'équations de départ
2 : liste d'équations finales
3 : équation résultat
4 : var de travail pour l'équation
5 : var de travail pour le poids
6 : résultat règle
7 : var de travail regle*/
selectEqu1([],[],E,E,_,R,R) :- !.
/* si on a déjà trouvé une règle de poids maximal, on arrête le parcours de la liste. */
selectEqu1(Q,Q,E,E,W,R,R) :- W >= 5, !.
selectEqu1([H|T],[H|Q],F,E,M,S,R) :- regle(H,RE), poidRegle(RE,W), (W =< M), selectEqu1(T,Q,F,E,M,S,R).
selectEqu1([H|T],[E|Q],F,E,M,S,_) :- regle(H,RE), poidRegle(RE,W), (W > M), selectEqu1(T,Q,F,H,W,S,RE).
/* Fonction choix_premier : renvoi simplement la règle qui s'applique à la première
équation du système */
choix_premier([E|P],P,E,R) :- regle(E,R).
unifie(P,S) :- set_echo, sousUnifie(P;[],S).
unifie(P) :- set_echo, sousUnifie(P;[]).
sousUnifie([];S) :- ident(S).
sousUnifie([];S,_) :- ident(S).
sousUnifie(P;S) :- choix_premier(P,Z,E,R), reduit(R,E,Z;S,Q), sousUnifie(Q,choix_premier).
sousUnifie(P;S,choix_premier) :- choix_premier(P,Z,E,R), reduit(R,E,Z;S,Q), sousUnifie(Q,choix_premier).
sousUnifie(P;S,choix_pondere) :- choix_pondere(P,Z,E,R), reduit(R,E,Z;S,Q), sousUnifie(Q,choix_pondere).