- Kombinatorische Logik
-
Kombinatorische Logik (Abgekürzt CL für engl. Combinatory Logic) ist eine Notation, die von Moses Schönfinkel und Haskell Brooks Curry eingeführt wurde, um die Verwendung von Variablen in der Mathematischen Logik zu vermeiden. Sie wird besonders in der Informatik als theoretisches Modell für Berechnung, als auch als Grundlage zum Design funktionaler Programmiersprachen eingesetzt.
Kombinatorische Logik in der Mathematik
Die Kombinatorische Logik war als einfachere Logik gedacht, die die Bedeutung von Variablen in der Notation von Logik klären und sie tatsächlich unnötig machen sollte.
Siehe Curry, 1958–72.
Kombinatorische Logik in der Informatik
In der Informatik dient die kombinatorische Logik als einfaches Modell einer Berechnung. Dieses wird in der Berechenbarkeitstheorie und der Beweistheorie benötigt. In der Tat erfasst die kombinatorische Logik viele Aspekte natürlicher Berechnung.
Man kann die kombinatorische Logik als Variation des Lambda-Kalküls auffassen, wobei die Ausdrücke des Lambda-Kalküls durch einige wenige Kombinatoren ersetzt werden. Kombinatoren sind primitive Funktionen, die keine freien Variablen enthalten. Da es sehr einfach ist, Lambda-Ausdrücke in Terme der CL umzuwandeln, und sich die Reduktion von Kombinatoren wesentlich einfacher gestaltet als die Lambda-Reduktion, wird die CL gerne als Implementationsgrundlage für nicht-strikte Sprachen verwendet.
Man kann die CL auch auf viele andere Art und Weisen betrachten, so zeigen viele frühere Abhandlungen die Äquivalenz verschiedener Kombinatoren zu Axiomen der Logik. [Hindley and Meredith, 1990].
Eine kurze Zusammenfassung des Lambda-Kalküls
Eine ausführliche Beschreibung des Lambda-Kalküls findet sich unter dessen Artikel. Lambda-Terme bestehen aus
- Variablen v,
- Abstraktionen λv.T,
- Applikationen (T1 T2),
wobei v hier für einen beliebigen Variablennamen und T, T1 und T2 wiederum für Lambda-Terme stehen. Lambda-Terme werden durch Beta-Reduktion vereinfacht, wobei die Applikation (λv.E a) ersetzt wird durch die Ersetzung E[v := a].
Die Kombinatorische Logik ist ein Berechnungsmodell, das äquivalent zum Lambda-Kalkül ist, aber ohne die Abstraktion auskommt.
Der kombinatorische Kalkül
Da Abstraktion im Lambda-Kalkül verwendet wird, um Funktionen zu modellieren, muss es im kombinatorischen Kalkül etwas vergleichbares geben. Hier gibt es statt der Abstraktion einige wenige primitive Funktionen (Kombinatoren), aus denen nun weitere Funktionen zusammengesetzt werden können.
Kombinatorische Terme
Kombinatorische Terme bestehen aus
- Kombinatoren P
- Applikationen (T1 T2)
T1 und T2 sind wiederum kombinatorische Terme. Die Applikation ist, wie im Lambda-Kalkül, linksassoziativ, das heißt T1 T2 T3 T4 ist gleichbedeutend mit (((T1 T2) T3) T4).
Beispiele von Kombinatoren
x, y usw. bezeichnen im folgenden beliebige Terme.
Der einfachste Kombinator ist I, der Identitätskombinator. Für ihn gilt:
- (I x) = x
Ein weiterer, einfacher Kombinator ist K, der Konstantenkombinator: (K x) modelliert dann eine Funktion, die für ein weiteres Argument immer x zurückgibt, also:
- (K x y) = x
Ein dritter Kombinator ist S, er stellt eine generalisierte Version der Applikation dar:
- (S x y z) = (x z (y z))
S wendet x auf y an, setzt jedoch zuvor z in beide ein.
Eigentlich ist I unnötig, wenn man S und K hat, denn
- ((S K K) x)
- = (S K K x)
- = (K x (K x))
- = x
Anmerkung: Es gilt zwar ((S K K) x) = (I x) für alle x, jedoch ist (S K K) nicht reduzierbar zu I. Man nennt dies extensionale Gleichheit.
Fixpunktkombinator
Noch interessanter ist der Fixpunktkombinator Y, der verwendet wird, um Rekursion zu implementieren.
- (Y x) = x (Y x)
Auch Y ist unnötig. Es kann, falls keine Typisierung erforderlich ist, als
- Y = S I I (S(K(S I))(S I I))
dargestellt werden.
Vollständigkeit der Grundlage S-K
Erstaunlich ist, dass man aus S und K allein Kombinatoren zusammensetzen kann, die extensional gleich jedem beliebigen Lambda-Term sind, und daher, gemäß der These von Church, extensional gleich jeder beliebigen Funktion. Als Beweis dient eine Transformation T[ ], die Lambda-Terme in einen Äquivalenten CL-Term verwandelt. Einzige Voraussetzung ist, dass der zu transformierende Lambda-Term keine freien Variablen enthält.
T[ ] kann folgendermaßen definiert werden
- T[V] ⇒ V
- T[(E1 E2)] ⇒ (T[E1] T[E2])
- T[λx.E] ⇒ (K T[E]) (nur, wenn x nicht frei in E)
- T[λx.x] ⇒ I
- T[λx.λy.E] ⇒ T[λx.T[λy.E]] (falls x freie Variable von E)
- T[λx.(E1 E2)] ⇒ (S T[λx.E1] T[λx.E2])
Die Transformation eines Lambda-Terms in einen CL-Term
Wir versuchen zum Beispiel, den Term λx.λy.(y x) in einen CL-Term zu übersetzen:
- T[λx.λy.(y x)]
- = T[λx.T[λy.(y x)]] (Regel 5)
- = T[λx.(S T[λy.y] T[λy.x])] (Regel 6)
- = T[λx.(S I T[λy.x])] (Regel 4)
- = T[λx.(S I (K x))] (Regel 3)
- = (S T[λx.(S I)] T[λx.(K x)]) (Regel 6)
- = (S (K (S I)) T[λx.(K x)]) (Regel 3)
- = (S (K (S I)) (S T[λx.K] T[λx.x])) (Regel 6)
- = (S (K (S I)) (S (K K) T[λx.x])) (Regel 3)
- = (S (K (S I)) (S (K K) I)) (Regel 4)
Wenn wir nun diesen Kombinator auf zwei Terme x und y anwenden, sieht die Reduktion folgendermaßen aus:
- (S (K (S I)) (S (K K) I) x y)
- = (K (S I) x (S (K K) I x) y)
- = (S I (S (K K) I x) y)
- = (I y (S (K K) I x y))
- = (y (S (K K) I x y))
- = (y (K K x (I x) y))
- = (y (K (I x) y))
- = (y (I x))
- = (y x)
Die kombinatorische Repräsentation (S (K (S I)) (S (K K) I)) ist viel länger als der Lambda-Term λx.λy.(y x). Das ist typisch für die Transformation. Generell kann es passieren, dass eine T[ ]-Konstruktion einen Lambda-Term der Länge n in einen Kombinator der Länge Θ(3n) umwandelt.
Erläuterung der T[ ]-Transformation
Die Motivation der T[ ]-Transformation ist die Eliminierung von Abstraktionen. Drei der Regeln sind trivial: Regel 4 transformiert λx.x in seine eindeutige Äquivalenz I, Regel 3 transformiert λx.E zu (K T[E]), was allerdings nur funktioniert, wenn die gebundene Variable x in E nicht verwendet wird (i.e.: in E nicht frei ist). Regel 2 transformiert die Applikation zweier Terme, was auch in den Termen der CL erlaubt ist.
Regel 1 ist etwas schwierig, denn sie konvertiert Variablen: Variablen können nur durch Regel 4 aufgelöst werden, daher bleiben sie fürs erste erhalten. Da es keine freien Variablen im Gesamtterm gibt, und jede Transformation von Abstraktionen die gebundenen Variablen berücksichtigt (Regeln 3,4,5,6), werden irgendwann alle Variablen aufgelöst.
Die Regeln 5 und 6 verschieben die Abstraktionen ins innere des Funktionskörpers, bis sie von Regel 4 aufgelöst werden können. Regel 5 konvertiert dazu zuerst den Körper der äußeren Abstraktion, danach die Abstraktion selbst. Regel 6 verteilt die Abstraktion über einer Applikation auf deren Teilterme:
λx.(E1 E2) ist eine Funktion, die ein Argument nimmt und im Lambda-Term (E1 E2) für x einsetzt. Genau dies tut der Kombinator S, nur für Funktionen E1[x] bzw. E2[x]:
- (λx.(E1 E2) a) = ((λx.E1 a) (λx.E2 a))
- = (S λx.E1 λx.E2 a)
- = ((S λx.E1 λx.E2) a)
Daher gilt, gemäß extensionaler Gleichheit:
- λx.(E1 E2) = (S λx.E1 λx.E2)
Um nun einen Kombinator für λx.(E1 E2) zu finden, reicht es aus einen Kombinator für (S λx.E1 λx.E2) zu finden, also:
- (S T[λx.E1] T[λx.E2])
Vereinfachung der Transformation
η-Reduktion
Die Kombinatoren, die T[ ] liefert, werden kürzer wenn wir die η-Reduktion aus dem Lambda-Kalkül verwenden:
T[λx.(E x)] = T[E] (nur, wenn x nicht frei in E)
[[λx.(E x)]] ist die Funktion, die ein Argument x nimmt, und es in die Funktion E einsetzt; dies ist extensional gleich der Funktion E selbst. Es ist daher ausreichend, einfach E in seine Kombinatorische Form zu transformieren.
Mit dieser Vereinfachung wird das obige Beispiel zu:
- T[λx.λy.(y x)]
- = …
- = (S (K (S I)) T[λx.(K x)])
- = (S (K (S I)) K) (durch η-Reduktion)
Dieser Kombinator ist (extensional) gleich dem längeren aus dem vorangegangenen Beispiel:
- (S (K (S I)) K x y)
- = (K (S I) x (K x) y)
- = (S I (K x) y)
- = (I y (K x y))
- = (y (K x y))
- = (y x)
Ganz ähnlich würde die normal T[]-Transformation die Identitätsfunktion λf.λx.(f x) verwandeln in (S (S (K S) (S (K K) I)) (K I)). Mit der neuen η-Reduktionsregel wird aus λf.λx.(f x) einfach nur I.
Die Kombinatoren B, C von Curry
Die Kombinatoren S und K tauchen bereits in der Arbeit von Schönfinkel auf (allein statt K hieß dort C), Curry führte in seiner Dissertation Grundlagen der kombinatorischen Logik weiterhin B, C, W (und K) ein. B und C können viele Reduktionen vereinfachen, sie sehen folgendermaßen aus:
- (C a b c) = (a c b)
- (B a b c) = (a (b c))
Für diese beiden Kombinatoren werden die Regeln folgendermaßen erweitert:
- T[V] ⇒ V
- T[(E1 E2)] ⇒ (T[E1] T[E2])
- T[λx.E] ⇒ (K T[E]) (nur, wenn x nicht frei in E)
- T[λx.x] ⇒ I
- T[λx.λy.E] ⇒ T[λx.T[λy.E]] (falls x freie Variable von E)
- T[λx.(E1 E2)] ⇒ (S T[λx.E1] T[λx.E2]) (falls x freie Variable von E1 und E2)
- T[λx.(E1 E2)] ⇒ (C T[λx.E1] T[E2]) (falls x freie Variable von E1 aber nicht von E2)
- T[λx.(E1 E2)] ⇒ (B T[E1] T[λx.E2]) (falls x freie Variable von E2 aber nicht von E1)
Zum Beispiel sieht die Transformation von λx.λy.(y x) so aus:
- T[λx.λy.(y x)]
- = T[λx.T[λy.(y x)]]
- = T[λx.(C T[λy.y] x)] (Regel 7)
- = T[λx.(C I x)]
- = (C I) (η-Reduktion)
- = C*(Notation für: X* = X I)
- = I'(Notation für: X' = C X)
Tatsächlich reduziert (C I x y) zu (y x):
(C I x y) = (I y x) = (y x)
B und C stellen beschränkte Versionen von S dar. Während S einen Wert sowohl in den Applikanten als auch in das Argument einsetzt, setzt C diesen nur in den Applikanten und B nur in das Argument ein.
Umgekehrte Transformation
Die Transformation L[ ] von CL-Termen in Lambda-Terme ist denkbar einfach, da wir im Lambda-Kalkül alle Möglichkeiten haben, die auch in der CL zur Verfügung stehen:
- L[I] = λx.x
- L[K] = λx.λy.x
- L[C] = λx.λy.λz.(x z y)
- L[B] = λx.λy.λz.(x (y z))
- L[S] = λx.λy.λz.(x z (y z))
- L[(E1 E2)] = (L[E1] L[E2])
Wichtig ist jedoch zu wissen, dass diese Transformation nicht das Inverse der T[ ]-Transformation ist, da T[L[x]] zwar extensional gleich x ist, aber der Term dabei nicht zwingend erhalten bleibt.
Unentscheidbarkeit des kombinatorischen Kalküls
Es ist unentscheidbar, ob ein genereller Kombinator eine Normalform hat, ob zwei Kombinatoren gleich sind, usw. Dies ergibt sich schon aus der Ähnlichkeit zum Lambda-Kalkül, aber hier noch einmal ein direkter Beweis:
Der Term
- Ω = (S I I (S I I))
hat keine Normalform, da er mit drei Schritten wieder zu sich selbst reduziert:
- (S I I (S I I))
- = (I (S I I) (I (S I I)))
- = (S I I (I (S I I)))
- = (S I I (S I I))
und es keinen anderen Weg geben kann, der den Kombinator kürzer macht.
Sei nun N ein Kombinator, der auf Normalform testet, so dass
- (N x) ⇒ T, wenn x eine Normalform hat,
- F, andernfalls.
(T und F sind hier die korrespondierenden Kombinatoren zu true und false aus dem Lambda-Kalkül:
- true = λx.λy.x = K = T
- false = λx.λy.y = (K I) = F
)
Sei nun
Z = (C (C (B N (S I I)) Ω) I)
Untersuchen wir den Term (S I I Z). Hat (S I I Z) eine Normalform? Falls ja, müssen auch die folgenden Ableitungen eine Normalform haben:
- (S I I Z)
- = (I Z (I Z))
- = (Z (I Z))
- = (Z Z)
- = (C (C (B N (S I I)) Ω) I Z) (Definition von Z)
- = (C (B N (S I I)) Ω Z I)
- = (B N (S I I) Z Ω I)
- = (N (S I I Z) Ω I)
Nun wenden wir N auf (S I I Z) an. Entweder hat (S I I Z) eine Normalform oder nicht. Wenn ja, dann wäre:
(N (S I I Z) Ω I) = (K Ω I) (Definition von N) = Ω
aber Ω hat keine Normalform, also führt dies zum Widerspruch. Falls (S I I Z) keine Normalform hat, reduziert sich der Term folgendermaßen:
(N (S I I Z) Ω I) = (K I Ω I) (Definition von N) = (I I) I
was bedeutet, dass (S I I Z) einfach I ist, also wieder ein Widerspruch. Daher kann es keinen solchen Kombinator N geben.
Anwendungsgebiete
Compilierung funktionaler Programmiersprachen
Funktionale Programmiersprachen basieren häufig auf der einfachen, aber universellen Semantik des Lambda-Kalküls.
David Turner benutzte CL für die Sprache SASL.
Referenzen
- „Über die Bausteine der mathematischen Logik“, Moses Schönfinkel, 1924, übersetzt als „On the Building Blocks of Mathematical Logic“ in From Frege to Gödel: a source book in mathematical logic, 1879–1931, ed. Jean van Heijenoort, Harvard University Press, 1977. ISBN 0-674-32449-8 – Die Originalpublikation über kombinatorische Logik.
- Combinatory Logic. Curry, Haskell B. et al., North-Holland, 1972. ISBN 0-7204-2208-6 – Eine umfassende Übersicht der CL, inklusive der historischen Umrisse.
- Functional Programming. Field, Anthony J. and Peter G. Harrison. Addison-Wesley, 1998. ISBN 0-201-19249-7
- „Foundations of Functional Programming“. Lawrence C. Paulson. University of Cambridge, 1995.
- „Lectures on the Curry-Howard Isomorphism“. Sørensen, Morten Heine B. and Pawel Urzyczyn. University of Copenhagen and University of Warsaw, 1999. (PDF-Datei; 1,33 MB)
- 1920–1931 Curry’s block notes
- „Principal Type-Schemes and Condensed Detachment“, Hindley and Meredith, Journal of Symbolic Logic (JSL), Volume 55 (1990), Number 1, pages 90–105
Weblinks
- Katalin Bimbó: Combinatory Logic, in: Stanford Encyclopedia of Philosophy (englisch, inklusive Literaturangaben)
Wikimedia Foundation.