Da die Aussagenlogik entscheidbar ist, ist es möglich einen Theorem Prover zu bauen, der für jede Formel entscheidet, ob sie gültig ist.
Ein solcher Beweiser ist jedoch nie effizient, da das Problem NP-Vollständig ist.
Beispiel: Schubfachprinzip
/*------------------------------------------------------------------------
Pigeonehole principle with 3 pigeons and 2 pigeonholes.
------------------------------------------------------------------------*/
formula(imp(and(and(or(p11,p12),or(p21,p22)),or(p31,p32)),
or(or(or(or(or(and(p11,p21),and(p11,p31)),and(p21,p31)),
and(p12,p22)),and(p12,p32)),and(p22,p32))), 'Theorem').
pij
Taube i in Loch j Wenn σ={x/a,y/g(b)} was ist dann
σx ist die folgende Substitution: xσx=x und vσx=vσ für alle v≠x.
Wenn σ={x/a,y/g(b)} was ist dann
disagreement pair: w and k(u,v)
Ein “disagreement pair” (d1,d2) heißt “simple”, genau dann wenn der Ausdruck d1 oder d2 (oder beide) eine Variable ist, die nicht in dem anderen Ausdruck vorkommt.
Term Unification:
input terms t1 and t2
let s:={}
while t1s \= t2s
choose a disagreement pair (d1,d2) for t1s, t2s
if (d1,d2) is not simple then
write t1 and t2 are not unifiable and HALT
else
let s := s U r, where r is the relevant repair
endif
endwhile
The algorithm always terminates and if it outputs a substitution, it outputs an idempotent substitution (σσ=σ) that leads to the mgu of t1 and t2.
Vorsicht, gewöhnliche Prolog-Unifikation verzichtet auf den occurs-check. Es gibt aber das eingebaute unify_with_occurs_check/2
-Prädikat, das eine Unifikation mit occurs-check durchführt.
Frage: Warum müssen die Substitutionen idempotent sein?
Beispiel für die Wichtigkeit der Skolem-Terme als Funktionsterme über den freien Variablen: ∃y(¬R(x,y)∧R(x,x)) Die Formel enthält die freie Variable x, daher muss die existenzgebundene Variable y durch den Skolemterm s(y) substituiert werden: ¬R(x,s(x))∧R(x,x) Nur so wird sichergestellt, dass x und y nicht unifiziert werden können.
Hätte man x durch eine einfache Skolemkonstante s substituiert, dann könnte man später y ebenfalls durch s substituieren und käme dann zur nicht erfüllbaren Formel ¬R(s,s)∧R(s,s).
Der folgende Code ist in freeVarTabl.pl
. Lediglich die Reihenfolge der Prädikate wurde verändert:
/*========================================================================
Try to create a tableau expansion for f(X) that is closed allowing
Qdepth applications of the universal rule.
========================================================================*/
tprove(X,Qdepth):-
notatedFormula(NotatedFormula,[],f(X)),
closedTableau([[NotatedFormula]],Qdepth).
/*========================================================================
Notate the free variables of a formula
========================================================================*/
notatedFormula(n(Free,Formula),Free,Formula).
Der Zähler Qdepth
kontrolliert, wie oft eine universal rule angewandt werden darf. Ein solcher Zähler ist notwendig, um den kontrollierten Abbruch zu erzwingen, da die Prädikatenlogik nicht entscheidbar ist. Bei einem Abbruch wissen wir aber nicht, ob kein Beweis für die Gültigkeit einer Formel existiert, oder ob wir nur noch keinen gefunden haben.
Für jede Formel wird explizit die Liste der freien Variablen angegeben (notatedFormula
). Statt mit einer Formel wie love(X,Y)
arbeiten wir mit der notated Formel n([X,Y],love(X,Y))
.
Der folgende Code unterscheidet sich kaum von der Tableau-Implementierung für die Aussagenlogik:
/*========================================================================
Expand Tableau until it is closed, allowing Qdepth
applications of the universal rule.
========================================================================*/
closedTableau([],_Q):- !.
closedTableau(OldTableau,Qdepth):-
expand(OldTableau,Qdepth,TempTableau,NewQdepth), !,
removeClosedBranches(TempTableau,NewTableau),
closedTableau(NewTableau,NewQdepth).
/*========================================================================
Remove all closed branches
========================================================================*/
removeClosedBranches([],[]).
removeClosedBranches([Branch|Rest],Tableau):-
closedBranch(Branch), !,
removeClosedBranches(Rest,Tableau).
removeClosedBranches([Branch|Rest],[Branch|Tableau]):-
removeClosedBranches(Rest,Tableau).
/*========================================================================
Check whether a branch is closed
========================================================================*/
closedBranch(Branch):-
memberList(n(_,t(X)),Branch),
memberList(n(_,f(Y)),Branch),
basicFormula(X),
basicFormula(Y),
unify_with_occurs_check(X,Y).
basicFormula/1
ist in comsemPredicates.pl
definiert und überprüft, ob eine Formel ein Basisausdruck der Form p(t1,...,tn)
ist, wobei p keine logische Konstante ist und alle ti simple Terme sind. Teste: ?- basicFormula(imp(a,b))
, ?- basicFormula(love(a,X))
, ?- basicFormula(X)
, ?- basicFormula(love(a,father(X)))
.
Die folgenden Zeilen sind wieder völlig analog zu der Implementierung für die Aussagenlogik:
/*========================================================================
Newtableau with Q-depth NewQdepth is the result of applying
a tableau expansion rule to Oldtableau with a Q-depth of OldQdepth.
========================================================================*/
expand([Branch|Tableau],QD,[NewBranch|Tableau],QD):-
unaryExpansion(Branch,NewBranch).
expand([Branch|Tableau],QD,[NewBranch|Tableau],QD):-
conjunctiveExpansion(Branch,NewBranch).
expand([Branch|Tableau],QD,[NewBranch|Tableau],QD):-
existentialExpansion(Branch,NewBranch).
expand([Branch|Tableau],QD,[NewBranch1,NewBranch2|Tableau],QD):-
disjunctiveExpansion(Branch,NewBranch1,NewBranch2).
% da universalExpansions mehrfach auf dieselbe Formel angewandt werden kann,
% schieben wir die Formeln immer wieder nach hinten im Tableau
% (sonst droht Endlosschleife):
expand([Branch|Tableau],OldQD,NewTableau,NewQD):-
universalExpansion(Branch,OldQD,NewBranch,NewQD),
appendLists(Tableau,[NewBranch],NewTableau).
expand([Branch|Rest],OldQD,[Branch|Newrest],NewQD):-
expand(Rest,OldQD,Newrest,NewQD).
/*========================================================================
Take Branch as input, and return NewBranches if a tableau rule
allows unary expansion.
========================================================================*/
unaryExpansion(Branch,[NotatedComponent|Temp]) :-
unary(SignedFormula,Component),
notatedFormula(NotatedFormula,Free,SignedFormula),
removeFirst(NotatedFormula,Branch,Temp),
notatedFormula(NotatedComponent,Free,Component).
/*========================================================================
Take Branch as input, and return the NewBranch if a tableau rule
allows conjunctive expansion.
========================================================================*/
conjunctiveExpansion(Branch,[NotatedComp1,NotatedComp2|Temp]):-
conjunctive(SignedFormula,Comp1,Comp2),
notatedFormula(NotatedFormula,Free,SignedFormula),
removeFirst(NotatedFormula,Branch,Temp),
notatedFormula(NotatedComp1,Free,Comp1),
notatedFormula(NotatedComp2,Free,Comp2).
/*========================================================================
Take Branch as input, and return the NewBranch1 and NewBranch2
if a tableau rule allows disjunctive expansion.
========================================================================*/
disjunctiveExpansion(Branch,[NotComp1|Temp],[NotComp2|Temp]):-
disjunctive(SignedFormula,Comp1,Comp2),
notatedFormula(NotatedFormula,Free,SignedFormula),
removeFirst(NotatedFormula,Branch,Temp),
notatedFormula(NotComp1,Free,Comp1),
notatedFormula(NotComp2,Free,Comp2).
/*========================================================================
Take Branch as input, and return the NewBranch if a tableau rule
allows existential expansion.
========================================================================*/
existentialExpansion(Branch,[NotatedInstance|Temp]):-
notatedFormula(NotatedFormula,Free,SignedFormula),
existential(SignedFormula),
removeFirst(NotatedFormula,Branch,Temp),
skolemFunction(Free,Term),
instance(SignedFormula,Term,Instance),
notatedFormula(NotatedInstance,Free,Instance).
Das Prädikat existentialExpansion/2
greift auf das Prädikat skolemFunction/2
zu, das einen neuen Skolemterm zu einer liste von freien Variablen generiert:
/*========================================================================
VarList is a list of free variables, and SkolemTerm is a previously
unused Skolem function symbol fun(N) applied to those free variables.
========================================================================*/
% Teste mit '?- freeVarTabl:skolemFunction([X,Y,Z],S)'
skolemFunction(VarList,SkolemTerm):-
newFunctionCounter(N),
compose(SkolemTerm,fun,[N|VarList]).
Weiter geht es mit der universalExpansion
. Hier wird bei jedem Aufruf der Zähler QDepth um eins runtergesetzt.
/*========================================================================
Take Branch and OldQD as input, and return the NewBranch and
NewQDepthif a tableau rule allow universal expansion.
========================================================================*/
universalExpansion(Branch,OldQD,NewBranch,NewQD):-
OldQD > 0,
NewQD is OldQD - 1,
memberList(NotatedFormula,Branch),
notatedFormula(NotatedFormula,Free,SignedFormula),
universal(SignedFormula),
removeFirst(NotatedFormula,Branch,Temp),
instance(SignedFormula,V,Instance),
notatedFormula(NotatedInstance,[V|Free],Instance),
appendLists([NotatedInstance|Temp],[NotatedFormula],NewBranch).
Das Prädikat universalExpansion/2
ruft das Prädikat instance/3
auf (substitute/4
ist in comsemPred.pl` definiert):
/*========================================================================
Remove quantifier from signed quantified formula, and replacing all
free occurrences of the quantified variable by occurrences of Term.
========================================================================*/
% ?- freeVarTabl:instance(t(all(X,love(X,vincent))),Y,F).
instance(t(all(X,F)),Term,t(NewF)):- substitute(Term,X,F,NewF).
instance(f(some(X,F)),Term,f(NewF)):- substitute(Term,X,F,NewF).
instance(t(some(X,F)),Term,t(NewF)):- substitute(Term,X,F,NewF).
instance(f(all(X,F)),Term,f(NewF)):- substitute(Term,X,F,NewF).
Der Rest ist analog zur Implementierung der Aussagenlogik:
/*========================================================================
Decompose conjunctive signed formula
========================================================================*/
conjunctive(t(and(X,Y)),t(X),t(Y)).
conjunctive(f(or(X,Y)),f(X),f(Y)).
conjunctive(f(imp(X,Y)),t(X),f(Y)).
/*========================================================================
Decompose disjunctive signed formula
========================================================================*/
disjunctive(f(and(X,Y)),f(X),f(Y)).
disjunctive(t(or(X,Y)),t(X),t(Y)).
disjunctive(t(imp(X,Y)),f(X),t(Y)).
/*========================================================================
Decompose unary signed formula
========================================================================*/
unary(t(not(X)),f(X)).
unary(f(not(X)),t(X)).
/*========================================================================
Universal Signed Formulas
========================================================================*/
universal(t(all(_,_))).
universal(f(some(_,_))).
/*========================================================================
Existential Signed Formulas
========================================================================*/
existential(t(some(_,_))).
existential(f(all(_,_))).
FOL Resolution (Grundidee)
Umwandlung der Formel in CNF (Clause Normal Form)
Hinweise
Ist ein Existenzquantor im Skopus eines Allquantors, setzen wir statt Skolemkonstanten komplexe Skolemterme (sprich Skolemfunktionen) ein. Bsp.: ∀x∃yR(x,y) wird skolemisiert zu ∀xR(x,s(x)). Der komplexe Skolemterm drückt die Abhängigkeit der existenzgebundenen Variablen von der allgebundenen aus. s(x) kann also als Funktion betrachtet werden, die jedem Element des Models ein Element zuordnet, zu dem es in der Relation R steht.
Beispiel: Bringe (∀xB(x)∧∃yA(y,x))→∃z∀x∃yC(x,z)∨B(z,y) in setCNF
Wir sorgen also per α-Konversion zunächst dafür, dass die Variablenmengen in den beiden Klauseln disjunkt sind.
Non-redundant Factors:
Beispiel: Seien w,x,y,z Variablen und a,b,c Konstanten. Die Klausel [P(a),P(x),R(y,b),R(z,x),¬Q(w),¬Q(f(z))] kann reduziert werden zu
Wir müssen jeweils alle nichtredundanten Faktoren einer Klausel behalten, da wir sonst möglicherweise keinen Beweis finden, obwohl es einen gäbe.
FOL in CNF umwandeln:
cnf(Formula,SetCNF):-
nnf(Formula,NNF),
skolemise(NNF,Skolemised,[]),
cnf([[Skolemised]],[],CNF),
setCnf(CNF,SetCNF).
Abweichungen zur Aussagenlogik (nnf)
nnf(not(all(X,F)),some(X,N)):-
nnf(not(F),N).
nnf(all(X,F),all(X,N)):-
nnf(F,N).
nnf(not(some(X,F)),all(X,N)):-
nnf(not(F),N).
nnf(some(X,F),some(X,N)):-
nnf(F,N).
Resolution
rprove(Formula):-
cnf(not(Formula),CNF),
nonRedundantFactors(CNF,NRF),
refute(NRF).
Abweichungen und Erweiterungen
/*========================================================================
Resolve two clauses
========================================================================*/
resolve(Clause1,Clause2,NewClause):-
selectFromList(Lit1,Clause1,Temp1),
selectFromList(not(Lit2),Clause2,Temp2),
unify_with_occurs_check(Lit1,Lit2),
unionSets(Temp1,Temp2,NewClause).
resolve(Clause1,Clause2,NewClause):-
selectFromList(not(Lit1),Clause1,Temp1),
selectFromList(Lit2,Clause2,Temp2),
unify_with_occurs_check(Lit1,Lit2),
unionSets(Temp1,Temp2,NewClause).
/*========================================================================
Compute Non-Redundant Factors for a list of clauses
========================================================================*/
nonRedundantFactors([],[]).
nonRedundantFactors([C1|L1],L4):-
findall(C2,nonRedFact(C1,C2),L3),
nonRedundantFactors(L1,L2),
appendLists(L3,L2,L4).
/*========================================================================
Compute Non-Redundant Factors for a Clause
========================================================================*/
nonRedFact([],[]).
nonRedFact([X|C1],C2):-
memberList(Y,C1),
unify_with_occurs_check(X,Y),
nonRedFact(C1,C2).
nonRedFact([X|C1],[X|C2]):-
nonRedFact(C1,C2).
Bearbeiten Sie die Aufgaben 5.6.1 bis 5.6.3.
Zusammenfassung
Auch haben wir die Behandlung von Gleichheit bisher nicht implementiert. Man könnte sie theoretisch hinzufügen, indem man jedem Aufruf die Axiome des Gleichheitsprädikats hinzufügt:
Viele moderne Theorembeweiser beruhen auf der Resolutionsmethode. Sie verwenden aber wesentlich effizientere Methoden zur Verarbeitung von Gleichheit und implementieren diverse Strategien zur Wahl möglichst effektiver Resolutionsschritte.
Teste
?- EQ = and(and(all(X,gleich(X,X)),all(X,forall(Y,imp(gleich(X,Y),gleich(Y,X))))),all(X,all(Y,all(Z,imp(and(gleich(X,Y),gleich(Y,Z)),gleich(X,Z)))))),
Kontext = and(all(X,imp(woman(X),snort(X))),some(X,and(woman(X),gleich(mia,X)))),
New = not(snort(mia)),
rprove(imp(and(EQ,Kontext),not(New))).
Im folgenden konzentrieren wir uns auf den Theorembeweiser Otter:
Beispiel: ∀xdance(x)→¬∃x¬dance(x)
?- fol2otter(imp(all(X,dance(X)),not(some(Y,not(dance(Y))))),user).
set(auto).
assign(max_seconds,30).
clear(print_proofs).
set(prolog_style_variables).
formula_list(usable).
((all A dance(A)) -> -((exists B -(dance(B))))).
end_of_list.
?- callTP(imp(all(X,dance(X)),not(some(Y,not(dance(Y))))),P,E).
Im folgenden fokussieren wir uns auf den Model Builder MACE.
Beispiel: ∀xboxer(x)→slow(x)∧boxer(butch)∧likes(mia,butch)
?- fol2mace(and(all(X,imp(boxer(X),slow(X))),and(boxer(butch),likes(mia,butch))),user).
set(auto).
clear(print_proofs).
set(prolog_style_variables).
formula_list(usable).
((all A (boxer(A) -> slow(A))) &
(boxer(butch) &
likes(mia,butch))).
end_of_list.
callMB/4
callMB(Problem,DomainSize, Model, ModelBuilder).
?- callMB(and(all(X,imp(boxer(X),slow(X))),and(boxer(butch),likes(mia,butch))),5,Model,MB).
Model = D=[d1]
f(1,boxer,[d1])
f(1,slow,[d1])
f(0,butch,d1)
f(0,mia,d1)
f(2,likes,[(d1,d1)])
,
MB = mace.