• Keine Ergebnisse gefunden

WS 09,10; SS 12–14, 16–19

N/A
N/A
Protected

Academic year: 2022

Aktie "WS 09,10; SS 12–14, 16–19"

Copied!
34
0
0

Wird geladen.... (Jetzt Volltext ansehen)

Volltext

(1)

Fortgeschrittene Programmierung Vorlesung

WS 09,10; SS 12–14, 16–19

Johannes Waldmann, HTWK Leipzig 11. Juli 2019

– Typeset by FoilTEX

Einleitung

Programmierung im Studium bisher

•1. Sem: Modellierung (formale Spezifikationen (von konkreten und abstrakten Datentypen))

•1./2. Sem Grundlagen der (AO) Programmierung imperatives Progr. (Programm ist Folge von

Anweisungen, bewirkt Zustands¨anderung)

strukturiertes P. (genau ein Eingang/Ausgang je Teilp.) objektorientiertes P. (Interface=abstrakter Datentyp,

Klasse=konkreter Datentyp)

•2. Sem: Algorithmen und Datenstrukturen

(Spezifikation, Implementierung, Korrektheit, Komplexit¨at)

•3. Sem: Softwaretechnik (industrielle Softwareproduktion)

– Typeset by FoilTEX 1

Worin besteht jetzt der Fortschritt?

•deklarativeProgrammierung

(Programmistausf¨uhrbare Spezifikation)

•insbesondere:funktionaleProgrammierung Def: Programm berechnetFunktion f:Eingabe7→Ausgabe,

(kein Zustand, keine Zustands¨anderungen)

Daten (erster Ordnung) sind B¨aume Programm ist Gleichungssystem

Programme sind auch Daten (h¨oherer Ordnung)

•ausdrucksstark, sicher, effizient, parallelisierbar

– Typeset by FoilTEX 2

Formen der deklarativen Programmierung

•funktionale Programmierung:foldr (+) 0 [1,2,3]

foldr f z l = case l of

[] -> z ; (x:xs) -> f x (foldr f z xs)

•logische Programmierung:append(A,B,[1,2,3]).

append([],YS,YS).

append([X|XS],YS,[X|ZS]):-append(XS,YS,ZS).

•Constraint-Programmierung

(set-logic QF_LIA) (set-option :produce-models true) (declare-fun a () Int) (declare-fun b () Int) (assert (and (>= a 5) (<= b 30) (= (+ a b) 20))) (check-sat) (get-value (a b))

– Typeset by FoilTEX 3

Definition: Funktionale Programmierung

•Rechnen=Auswerten von Ausdr¨ucken (Termb¨aumen)

•Dabei wird einWertbestimmt

und es gibt keine (versteckte)Wirkung.

(engl.: side effect, dt.: Nebenwirkung)

•Werte k¨onnen sein:

“klassische” Daten (Zahlen, Listen, B¨aume. . . ) True :: Bool,[3.5, 4.5] :: [Double]

Funktionen (Sinus, . . . )

[sin, cos] :: [Double -> Double]

Aktionen (Datei lesen, schreiben, . . . ) readFile "foo.text" :: IO String

– Typeset by FoilTEX 4

Softwaretechnische Vorteile

. . . der funktionalen Programmierung

•Beweisbarkeit: Rechnen mit Programmen wie in der Mathematik mit Termen

•Sicherheit: es gibt keine Nebenwirkungen und Wirkungen sieht man bereits am Typ

•Aussdrucksst¨arke, Wiederverwendbarkeit: durch Funktionen h¨oherer Ordnung (sog. Entwurfsmuster)

•Effizienz: durch Programmtransformationen im Compiler,

•Parallelit¨at: keine Nebenwirkungen⇒keinedata races, fktl. Programme sindautomatisch parallelisierbar

– Typeset by FoilTEX 5

Beispiel Spezifikation/Test

import Test.LeanCheck

append :: forall t . [t] -> [t] -> [t]

append [] y = y

append (h : t) y = h : (append t y) associative f =

\ x y z -> f x (f y z) == f (f x y) z commutative f = \ x y -> ...

test = check

(associative (append::[Bool]->[Bool]->[Bool]))

¨Ubung: Kommutativit¨at (formulieren und testen)

– Typeset by Foil 6

Beispiel Verifikation

app :: forall t . [t] -> [t] -> [t]

app [] y = y

app (h : t) y = h : (app t y)

Lemma: app x (app y z) .=. app (app x y) z Proof by induction on List x

Case []

To show: app [] (app y z) .=. app (app [] y) z Case h:t

To show: app (h:t) (app y z) .=. app (app (h:t) y) z IH: app t (app y z) .=. app (app t y) z

CYPhttps://github.com/noschinl/cyp, ist vereinfachte Version

von Isabellehttps://isabelle.in.tum.de/

– Typeset by Foil 7

(2)

Beispiel Parallelisierung (Haskell)

Klassische Implementierung von Mergesort sort :: Ord a => [a] -> [a]

sort [] = [] ; sort [x] = [x]

sort xs = let ( left,right ) = split xs sleft = sort left

sright = sort right in merge sleft sright wird parallelisiert durchAnnotationen:

sleft = sort left

‘using‘ rpar ‘dot‘ spineList sright = sort right ‘using‘ spineList vgl.http://thread.gmane.org/gmane.comp.lang.

haskell.parallel/181/focus=202

– Typeset by FoilTEX 8

Beispiel Parallelisierung (C#, PLINQ)

•Die Anzahl der 1-Bits einer nichtnegativen Zahl:

Func<int,int>f =

x=>{int s=0; while(x>0){s+=x%2;x/=2;}return s;}

2X26−1 x=0

f(x) Enumerable.Range(0,1<<26).Select(f).Sum()

•automatische parallele Auswertung, Laufzeitvergleich:

Time(()=>Enumerable.Range(0,1<<26).Select(f).Sum()) Time(()=>Enumerable.Range(0,1<<26).AsParallel()

.Select(f).Sum())

vgl.Introduction to PLINQhttps://msdn.microsoft.

com/en-us/library/dd997425(v=vs.110).aspx

– Typeset by FoilTEX 9

Softwaretechnische Vorteile

. . . der statischen Typisierung

The language in which you write profoundly affects the design of programs written in that language.

For example, in the OO world, many people use UML to sketch a design. In Haskell or ML, one writes type signatures instead. Much of the initial design phase of a functional program consists of writing type definitions.

Unlike UML, though, all this design is incorporated in the final product, and is machine-checked throughout.

Simon Peyton Jones, in: Masterminds of Programing, 2009;

http://shop.oreilly.com/product/9780596515171.do

– Typeset by FoilTEX 10

Deklarative Programmierung in der Lehre

•funktionale Programmierung: diese Vorlesung

•logische Programmierung: inAngew. K¨unstl. Intell.

•Constraint-Programmierung: als Master-Wahlfach Beziehungen zu weiteren LV: Voraussetzungen

•B¨aume, Terme (Alg.+DS, Grundlagen Theor. Inf.)

•Logik (Grundlagen TI, Softwaretechnik) Anwendungen:

•Softwarepraktikum

•weitere Sprachkonzepte inPrinzipien v.

Programmiersprachen

•Programmverifikation(vorw. f. imperative Programme)

– Typeset by FoilTEX 11

Konzepte und Sprachen

Funktionale Programmierung ist einKonzept.

Realisierungen:

•in prozeduralen Sprachen:

Unterprogramme als Argumente (in Pascal) Funktionszeiger (in C)

•in OO-Sprachen: Befehlsobjekte

•Multi-Paradigmen-Sprachen:

Lambda-Ausdr¨ucke in C#, Scala, Clojure

•funktionale Programmiersprachen (LISP, ML, Haskell) Die Erkenntnisse sind sprachunabh¨angig.

•A good programmer can write LISP in any language.

•Learn Haskell and become a better Java programmer.

– Typeset by FoilTEX 12

Gliederung der Vorlesung

•Terme, Termersetzungssysteme algebraische Datentypen, Pattern Matching, Persistenz

•Funktionen (polymorph, h¨oherer Ordnung), Lambda-Kalk¨ul, Rekursionsmuster

•Typklassen zur Steuerung der Polymorphie

•Bedarfsauswertung, unendl. Datenstrukturen (Iterator-Muster)

•funktional-reaktive Programmierung (deklarative interaktive Programme)

•weitere Entwurfsmuster

•Code-Qualit¨at, Code-Smells, Refactoring

– Typeset by FoilTEX 13

Softwaretechnische Aspekte

algebraische Datentypen, Pattern Matching, Termersetzungssysteme

Scale: case class, Java: Entwurfsmuster Kompositum, immutable objects, das Datenmodell von Git

Funktionen (h¨oherer Ordnung), Lambda-Kalk¨ul, Rekursionsmuster

Lambda-Ausdr¨ucke in C#, Entwurfsmuster Besucher Codequalit¨at, code smells, Refaktorisierung

Typklassen zur Steuerung der Polymorphie

Interfaces in Java/C# , automatische Testfallgenerierung

Bedarfsauswertung, unendl. Datenstrukturen

Iteratoren, Str¨ome, LINQ

– Typeset by Foil 14

Organisation der LV

•jede Woche eine Vorlesung, eine ¨Ubung

•Hausaufgaben

Vorrechnen von ¨U-Aufgaben (an Tafel o. Bildschirm), gruppenweise

https:

//autotool.imn.htwk-leipzig.de/new/, individuell

•Pr¨ufungszulassung: regelm¨aßiges (d.h. innerhalb der jeweiligen Deadline) und erfolgreiches (jeweils ingesamt

≥50%der Pflichtaufgaben) Bearbeiten (und Pr¨asentieren) der ¨Ubungsaufgaben.

•Pr¨ufung: Klausur (ohne Hilfsmittel)

– Typeset by Foil 15

(3)

Literatur

Skript aktuelles Semesterhttp:

//www.imn.htwk-leipzig.de/˜waldmann/lehre.html vorige Semesterhttp://www.imn.htwk-leipzig.de/

˜waldmann/lehre-alt.html

http://haskell.org/ (Sprache, Werkzeuge, Tutorials), http://book.realworldhaskell.org/

•Kriterium f¨ur Haskell-Tutorials und -Lehrb¨ucher:

wo werdendata(benutzerdefinerte algebraische Datentypen) undcase(pattern matching) erkl¨art?

Je sp¨ater, desto schlechter!

vgl. https:

//www.imn.htwk-leipzig.de/˜waldmann/talk/17/wflp/

– Typeset by FoilTEX 16

Alternative Quellen

Q: Aber in Wikipedia/Stackoverflow steht, daß . . . A: Na und.

•Es mag eine in Einzelf¨allen n¨utzliche ¨Ubung sein, sich mit dem Halbwissen von Nichtfachleuten

auseinanderzusetzen.

Beachte aberhttps://xkcd.com/386/

•In VL und ¨Ubung verwenden und diskutieren wir die durch Dozenten/Skript/Modulbeschreibung vorgegebenen Quellen (Lehrb¨ucher, referierte Original-Artikel, Standards zu Sprachen und Bibliotheken)

•. . . gilt entsprechend f¨ur Ihre Bachelor- und Master-Arbeit.

– Typeset by FoilTEX 17

¨Ubungen KW15

•Benutztung Rechnerpool, ghci aufrufen

http://www.imn.htwk-leipzig.de/˜waldmann/etc/pool/

(aus Hochschulnetz bzw. VPN)

•Auf wieviele Nullen endet die Fakult¨at von 100?

Benutzefoldrzum Berechnen der Fakult¨at.

Beachte polymorphe numerische Literale.

(Aufl¨osung der Polymorphie durch Typ-Annotation.) Warum ist 100 Fakult¨at alsIntgleich 0?

Welches ist der Typ der FunktiontakeWhile? Beispiel:

odd 3 ==> True ; odd 4 ==> False takeWhile odd [3,1,4,1,5,9] ==> [3,1]

ersetze in der L¨osungtakeWhiledurch andere

– Typeset by FoilTEX 18

Funktionen des gleichen Typs (suche diese mit Hoogle), erkl¨are Semantik

typische Eigenschaften dieses Beispiels (nachmachen!) statische Typisierung, Schachtelung von

Funktionsaufrufen, Funktion h¨oherer Ordnung, Benutzung von Funktionen aus Standardbibliothek (anstatt selbstgeschriebener).

schlechte Eigenschaften (vermeiden!) Benutzung von Zahlen und Listen (anstatt anwendungsspezifischer Datentypen) vgl.

http://www.imn.htwk-leipzig.de/˜waldmann/

etc/untutorial/list-or-not-list/

•Haskell-Entwicklungswerkzeuge ghci (Fehlermeldungen, Holes)

– Typeset by FoilTEX 19

API-Suchmaschinehttp://www.haskell.org/hoogle/

IDE? brauchen wir (in dieser VL) nicht.

Ansonsten emacs/intero,http://xkcd.org/378/

•Softwaretechnik im autotool: http://www.imn.

htwk-leipzig.de/˜waldmann/etc/untutorial/se/

•Commercial Uses of Functional Programming

http://www.syslog.cl.cam.ac.uk/2013/09/22/

liveblogging-cufp-2013/

– Typeset by FoilTEX 20

Aufgaben (Auswertung in KW 16)

1. zu: E. W. Dijkstra:Answers to Questions from Students of Software Engineering(Austin, 2000) (EWD 1035)

•”putting the cart before the horse“

¨ubersetzen Sie w¨ortlich ins Deutsche, geben Sie eine entsprechende idiomatische

Redewendung in Ihrer Muttersprache an, wof¨ur stehencartundhorsehier konkret?

2. sind die empfohlenen exakten Techniken der Programmierung f¨ur große Systeme anwendbar?

Erkl¨aren Sie ”lengths of ...grow not much more than linear with the lengths of . . .“.

– Typeset by FoilTEX 21

•Welche L¨angen werden hier verglichen?

Modellieren Sie das System als Graph, die Knoten sind die Komponenten, die Kanten sind deren Beziehungen (direkte Abh¨angigkeiten).

•Welches asymptotische Wachstum ist bei undisziplinierter Entwicklung des Systems zu bef¨urchten?

•Welche Graph-Eigenschaft impliziert den linearen Zusammenhang?

•Wie gestaltet man den System-Entwurf, so daß diese Eigenschaft tats¨achlich gilt? Welchen Nutzen hat das f¨ur Entwicklung und Wartung?

3. ¨Uber ein Monoid(M,◦,1)mit Elementena, b∈M(sowie

– Typeset by Foil 22

eventuell weiteren) ist bekannt:a2=b2= (ab)2= 1.

Dabei istabeine Abk¨urzung f¨ura◦bunda2f¨uraa, usw.

•Geben Sie ein Modell mit16=a6=b6= 1an.

• ¨Uberpr¨ufen Sieab=bain Ihrem Modell.

•Leiten Sieab=baaus den Monoid-Axiomen und gegebenen Gleichungen ab.

Das ist eine ¨Ubung zur Wiederholung der Konzepte abstrakter undkonkreter Datentyp sowieSpezifikation.

4. im Rechnerpool live vorf¨uhren:

•ein Terminal ¨offnen

•ghcistarten (in der aktuellen Version), Fakult¨at von 100 ausrechnen

– Typeset by Foil 23

(4)

•DateiF.hsmit Texteditor anlegen und ¨offnen, Quelltext f = ...(Ausdruck mit Wert100!) schreiben, diese Datei inghciladen,fauswerten

Dabei wg. Projektion an die Wand:

Schrift 1. groß genug und 2. schwarz auf weiß.

Vorher Bildschirm(hintergrund) aufr¨aumen, so daß bei Projektion keine personenbezogenen Daten sichtbar werden. Beispiel:export PS1="$ "¨andert den Shell-Prompt (versteckt den Benutzernamen).

Wer (unsinnigerweise) eigenen Rechner im Pool benutzt:

•Aufgaben wie obenund

•ssh-Login auf einen Rechner des Pools

(damit wird die AusredeGHC (usw.) geht auf meinem

– Typeset by FoilTEX 24

Rechner nichthinf¨allig)

•ssh-Login oder remote-Desktop-Zugriffvoneinem Rechner des Pools auf Ihren Rechner (damit das projiziert werden kann,ohneden Beamer umzust¨opseln)

(falls das alles zu umst¨andlich ist, dann eben doch einen Pool-Rechner benutzen)

5. welcher Typ ist zu erwarten f¨ur die Funktion,

•(wurde bereits in der ¨Ubung behandlelt) die das Spiegelbild einer Zeichenkette berechnet?

•die die Liste aller (durch Leerzeichen getrennten) W¨orter einer Zeichenkette berechnet?

f "foo bar" = [ "foo", "bar" ]

– Typeset by FoilTEX 25

Suchen Sie nach Funktionen dieses Typs mit

https://www.haskell.org/hoogle/, erkl¨aren Sie einige der Resultate, welches davon ist das passende, rufen Sie diese Funktion auf (in ghci).

– Typeset by FoilTEX 26

Daten

Wiederholung: Terme

•(Pr¨adikatenlogik)Signatur Σist Menge von Funktionssymbolen mit Stelligkeiten ein Termtin SignaturΣist

Funktionssymbolf∈Σder Stelligkeitk

mit Argumenten(t1, . . . , tk), die selbst Terme sind.

Term(Σ) =Menge der Terme ¨uber SignaturΣ

•(Graphentheorie) ein Term ist ein gerichteter, geordneter, markierter Baum

•(Datenstrukturen)

Funktionssymbol=Konstruktor, Term=Baum

– Typeset by FoilTEX 27

Beispiele: Signatur, Terme

•Signatur:Σ1={Z/0, S/1, f /2}

•Elemente vonTerm(Σ1):

Z(), S(S(Z())), f(S(S(Z())), Z())

•Signatur:Σ2={E/0, A/1, B/1}

•Elemente vonTerm(Σ2): . . .

– Typeset by FoilTEX 28

Algebraische Datentypen

data Foo = Foo { bar :: Int, baz :: String } deriving Show

Bezeichnungen (benannte Notation)

•data Fooist Typname

•Foo { .. }ist Konstruktor

•bar, bazsind Komponenten x :: Foo

x = Foo { bar = 3, baz = "hal" } Bezeichnungen (positionelle Notation) data Foo = Foo Int String y = Foo 3 "bar"

– Typeset by FoilTEX 29

Datentyp mit mehreren Konstruktoren

Beispiel (selbst definiert)

data T = A { foo :: Int }

| B { bar :: String, baz :: Bool } deriving Show

Bespiele (in Prelude vordefiniert) data Bool = False | True data Ordering = LT | EQ | GT

– Typeset by Foil 30

Mehrsortige Signaturen

•(bisher) einsortige Signatur

Abbildung von Funktionssymbol nach Stelligkeit

•(neu) mehrsortige Signatur

Menge von SortensymbolenS={S1, . . .} Abb. von F.-Symbol nach Typ

Typist Element ausS×S

Folge der Argument-Sorten, Resultat-Sorte Bsp.:S={Z, B},Σ ={07→([], Z), p7→([Z, Z], Z), e7→([Z, Z], B), a7→([B, B], B)}.

•Term(Σ): konkrete Beispiele, allgemeine Definition?

– Typeset by Foil 31

(5)

Rekursive Datentypen

data Tree = Leaf {}

| Branch { left :: Tree , right :: Tree }

¨Ubung: Objekte dieses Typs erzeugen

(benannte und positionelle Notation der Konstruktoren)

– Typeset by FoilTEX 32

Daten mit Baum-Struktur

•mathematisches Modell: Term ¨uber Signatur

•programmiersprachliche Bezeichnung:algebraischer Datentyp(die Konstruktoren bilden eine Algebra)

•praktische Anwendungen:

Formel-B¨aume (in Aussagen- und Pr¨adikatenlogik) Suchb¨aume (in VL Algorithmen und Datenstrukturen, in

java.util.TreeSet<E>) DOM (Document Object Model)

https://www.w3.org/DOM/DOMTR

JSON (Javascript Object Notation) z.B. f¨ur AJAX http://www.ecma-international.org/

publications/standards/Ecma-404.htm

– Typeset by FoilTEX 33

Bezeichnungen f ¨ur Teilterme

•Position: Folge von nat¨urlichen Zahlen

(bezeichnet einen Pfad von der Wurzel zu einem Knoten) Beispiel: f¨urt=S(f(S(S(Z())), Z()))

ist[0,1]eine Position int.

•Pos(t) =die Menge der Positionen eines Termst Definition: wennt=f(t0, . . . , tk−1), d.h.,kKinder dannPos(t) ={[]} ∪ {[i]++p|0≤i < k∧p∈Pos(ti)}. dabei bezeichnen:

•[]die leere Folge,

•[i]die Folge der L¨ange 1 mit Elementi,

•++den Verkettungsoperator f¨ur Folgen

– Typeset by FoilTEX 34

Operationen mit (Teil)Termen

•t[p] =der Teilterm vontan Positionp Beispiel:S(f(S(S(Z())), Z()))[0,1] =. . .

Definition (durch Induktion ¨uber die L¨ange vonp): . . .

•t[p:=s]: wiet, aber mit Termsan Positionp Beispiel:S(f(S(S(Z())), Z()))[[0,1] :=S(Z)] =. . . Definition (durch Induktion ¨uber die L¨ange vonp): . . .

– Typeset by FoilTEX 35

Operationen mit Variablen in Termen

•Term(Σ, V) =Menge der Terme ¨uber SignaturΣmit Variablen ausV

Beispiel:Σ ={Z/0, S/1, f /2}, V ={y}, f(Z(), y)∈Term(Σ, V).

•Substitutionσ: partielle AbbildungV →Term(Σ) Beispiel:σ1={(y, S(Z()))}

•eine Substitution auf einen Term anwenden:tσ:

Intuition: wiet, aber stattvimmerσ(v) Beispiel:f(Z(), y)σ1=f(Z(), S(Z())) Definition durch Induktion ¨ubert

– Typeset by FoilTEX 36

Termersetzungssysteme

•Daten=Terme (ohne Variablen)

•ProgrammR=Menge von Regeln

Bsp:R={(f(Z(), y), y), (f(S(x), y), S(f(x, y)))}

•Regel=Paar(l, r)von Termen mit Variablen

•Relation→Rist Menge aller Paare(t, t0)mit es existiert(l, r)∈R

es existiert Positionpint

es existiert Substitutionσ: (Var(l)∪Var(r))→Term(Σ) so daßt[p] =lσundt0=t[p:=rσ].

– Typeset by FoilTEX 37

Termersetzungssysteme als Programme

• →RbeschreibteinenSchritt der Rechnung vonR,

•transitive und reflexive H¨ulle→R

beschreibtFolgevon Schritten.

•Resultateiner Rechnung ist Term inR-Normalform (:=ohne→R-Nachfolger)

dieses Berechnungsmodell ist im allgemeinen

•nichtdeterministischR1={C(x, y)→x, C(x, y)→y} (ein Term kann mehrere→R-Nachfolger haben, ein Term kann mehrere Normalformen erreichen)

•nicht terminierend R2={p(x, y)→p(y, x)} (es gibt eine unendliche Folge von→R-Schritten, es kann Terme ohne Normalform geben)

– Typeset by Foil 38

Konstruktor-Systeme

F¨ur TRSR¨uber SignaturΣ: Symbols∈Σheißt

•definiert, wenn∃(l, r)∈R:l[] =s(. . .) (das Symbol in der Wurzel ists)

•sonstKonstruktor.

Das TRSRheißtKonstruktor-TRS, falls:

•definierte Symbole kommen linksnur in den Wurzeln vor

¨Ubung: diese Eigenschaft formal spezifizieren

Beispiele:R1={a(b(x))→b(a(x))} ¨uberΣ1={a/1, b/1}, R2={f(f(x, y), z)→f(x, f(y, z)) ¨uberΣ2={f /2}: definierte Symbole? Konstruktoren? Konstruktor-System?

Funktionale Programme sind ¨ahnlich zu Konstruktor-TRS.

– Typeset by Foil 39

(6)

¨Ubung Terme, TRS

•Geben Sie die Signatur des Terms√

a·a+b·ban.

•Geben Sie ein Elementt∈Term({f /1, g/3, c/0})an mit t[1] =c().

mitghci:

•data T = F T | G T T T | C deriving Show erzeugen Sie o.g. Termt(durch Konstruktoraufrufe)

•Geben SiePos(t)an

DieGr¨oßeeines Termstist definiert durch

|f(t0, . . . , tk−1)|= 1 +Pk−1 i=0 |ti|.

– Typeset by FoilTEX 40

•Bestimmen Sie|√

a·a+b·b|.

•Beweisen Sie∀Σ :∀t∈Term(Σ) :|t|=|Pos(t)|. F¨ur die SignaturΣ ={Z/0, S/1, f /2}:

•f¨ur welche Substitutionσgiltf(x, Z)σ=f(S(Z), Z)?

•f¨ur diesesσ: bestimmen Sief(x, S(x))σ.

Notation f¨ur Termersetzungsregeln: anstatt(l, r)schreibe l→r.

Abk¨urzung f¨ur Anwendung von 0-stelligen Symbolen:

anstattZ()schreibeZ. (Vorsicht: dann kann man Variablen nicht mehr von 0-stelligen Symbolen unterscheiden. Man muß dann immer die Signatur explizit angeben oder auf andere Weise vereinbaren, wie man Variablen erkennt, z.B.

– Typeset by FoilTEX 41

”Buchstaben am Ende das Alphabetes (. . . , x, y, . . .) sind Variablen“, das ist aber riskant)

•F¨urR={f(S(x), y)→f(x, S(y)), f(Z, y)→y} bestimme alleR-Normalformen vonf(S(Z), S(Z)).

•f¨urRd=R∪ {d(x)→f(x, x)}

bestimme alleRd-Normalformen vond(d(S(Z))).

•Bestimme die SignaturΣdvonRd.

Bestimme die Menge der Terme ausTerm(Σd), die Rd-Normalformen sind.

Abk¨urzung f¨ur mehrfache Anwendung eines einstelligen Symbols:A(A(A(A(x)))) =A4(x)

– Typeset by FoilTEX 42

•f¨ur{A(B(x))→B(A(x))}

¨uber Signatur{A/1, B/1, E/0}: bestimme Normalform vonAk(Bk(E)) f¨urk= 1,2,3,allgemein.

•f¨ur{A(B(x))→B(B(A(x)))}

¨uber Signatur{A/1, B/1, E/0}: bestimme Normalform vonAk(B(E)) f¨urk= 1,2,3,allgemein.

– Typeset by FoilTEX 43

Hausaufgaben (Diskussion in KW18)

1. autotool-Aufgabe Data-1

Allgemeine Hinweise zur Bearbeitung von Haskell-L¨uckentext-Aufgaben:

•Schreiben Sie den angezeigten Quelltext (vollst¨andig!

ohne zus¨atzliche Leerzeichen am Zeilenanfang!) in eine Datei mit Endung.hs, starten Sie ghci mit diesem Dateinamen als Argument

• ¨andern Sie den Quelltext: ersetzen Sieundefined durch einen geeigneten Ausdruck, hier z.B.

solution = S.fromList [ False, G ] im Editor speichern, in ghci neu laden (:r)

•reparieren Sie Typfehler, werten Sie geeignete Terme

– Typeset by FoilTEX 44

aus, hier z.B.S.size solution

•werten Sietestaus, wenntestden Wert True ergibt, dann tragen Sie die L¨osung in autotool ein.

2. Geben Sie einen Typ (einedata-Deklaration) mit genau 100 Elementen an. Sie k¨onnen andere

Data-Deklarationen benutzen (wie in autotool-Aufgabe).

Minimieren Sie die Gesamtzahl aller deklarierten Konstruktoren.

3. Vervollst¨andigen Sie die Definition derTiefevon Termen:

depth(f()) = 0

k >0 ⇒ depth(f(t0, . . . , tk−1)) =. . .

– Typeset by FoilTEX 45

•Bestimmen Siedepth(√

a·a+b·b)

•Beweisen Sie∀Σ :∀t∈Term(Σ) : depth(t)≤ |t| −1.

•F¨ur welche Terme gilt hier die Gleichheit?

4. autotool-Aufgabe TRS-1

5. (Zusatz-Aufgabe) f¨ur die Signatur{A/2, D/0}:

•definiere Termet0=D, ti+1=A(ti, D).

Zeichnet3. Bestimme|ti|,depth(ti).

•f¨urS={A(A(D, x), y)→A(x, A(x, y))}

bestimmeS-Normalform(en), soweit existieren, der Termet2, t3, t4.

Geben Sie f¨urt2die ersten Ersetzungs-Schritte explizit an.

•Normalform vontiallgemein.

– Typeset by Foil 46 – Typeset by Foil 47

(7)

Programme

Funktionale Programme (Bsp.)

Signatur:{(S,1),(Z,0),(f,2)}, Variablenmenge{x0, y} Ersetzungssystem{f(Z, y)→y, f(S(x0), y)→S(f(x0, y))}. Konstruktor-System mit definiertem Symbol{f},

Konstruktoren{S, Z},data N = Z | S N

Starttermf(S(S(Z)), S(Z)). funktionales Programm:

f :: N -> N -> N -- Typdeklaration -- Gleichungssystem:

f Z y = y ; f (S x’) y = S (f x’ y)

Aufruf:f (S (S Z)) (S Z)

alternativ: eine Gleichung, mit Pattern Match

f x y = case x of

{ Z -> y ; S x’ -> S (f x’ y) }

– Typeset by FoilTEX 48

Pattern Matching

data Tree = Leaf | Branch Tree Tree size :: Tree -> Int

size t = case t of { ... ; Branch l r -> ... }

•Syntax:case <Diskriminante> of { <Muster> -> <Ausdruck> ; ... }

•<Muster>enth¨alt Konstruktoren und Variablen, entspricht linker Seite einer Term-Ersetzungs-Regel,

<Ausdruck>entspricht rechter Seite

•statische Semantik:

jedes<Muster>hat gleichen Typ wie<Diskrim.>, alle<Ausdruck>haben ¨ubereinstimmenden Typ.

•dynamische Semantik:

Def.:tpaßt zum Musterl: es existiertσmitlσ=t f¨ur das erste passende Muster wirdrσausgewertet

– Typeset by FoilTEX 49

Eigenschaften von Case-Ausdr ¨ucken

eincase-Ausdruck heißt

•disjunkt, wenn die Muster nicht ¨uberlappen

(es gibt keinen Term, der zu mehr als 1 Muster paßt)

•vollst¨andig, wenn die Muster den gesamten Datentyp abdecken

(es gibt keinen Term, der zu keinem Muster paßt) Bespiele (f¨urdata N = F N N | S N | Z) -- nicht disjunkt:

case t of { F (S x) y -> .. ; F x (S y) -> .. } -- nicht vollst¨andig:

case t of { F x y -> .. ; Z -> .. }

– Typeset by FoilTEX 50

data

und

case

typisches Vorgehen beim Verarbeiten algebraischer Daten vom TypT:

•F¨ur jeden Konstruktor des Datentyps data T = C1 ...

| C2 ...

•schreibe einen Zweig in der Fallunterscheidung f x = case x of

C1 ... -> ...

C2 ... -> ...

•Argumente der Konstruktoren sind Variablen⇒ Case-Ausdruck ist disjunkt und vollst¨andig.

– Typeset by FoilTEX 51

Pattern Matching in versch. Sprachen

•Scala: case classeshttp://docs.scala-lang.org/

tutorials/tour/case-classes.html

•C#:

https://github.com/dotnet/csharplang/blob/

master/proposals/csharp-8.0/patterns.md

•Javascript?

Nicht verwechseln mitregular expression matchingzur String-Verarbeitung. Es geht um algebraische (d.h.

baum-artige) Daten!

– Typeset by FoilTEX 52

Peano-Zahlen

data N = Z | S N

plus :: N -> N -> N plus x y = case x of

Z -> y

S x’ -> S (plus x’ y) Aufgaben:

•implementiere Multiplikation, Potenz

•beweise die ¨ublichen Eigenschaften (Addition, Multiplikation sind assoziativ, kommutativ, besitzen neutrales Element)

– Typeset by FoilTEX 53

Spezifikation und Test

Bsp: Addition von Peano-Zahlen

•Spezifikation:

Typ:plus :: N -> N -> N

Axiome (Bsp):plusist assoziativ und kommutativ

•Test der Korrektheit durch Aufz¨ahlen einzelner Testf¨alle

plus(S (S Z))(S Z) == plus(S Z)(S (S Z)) Notieren von Eigenschaften (properties)

plus_comm :: N -> N -> Bool

plus_comm x y = plus x y == plus y x und automatische typgesteuerte Testdatenerzeugung Test.LeanCheck.checkFor 10000 plus_comm

– Typeset by Foil 54

Spezifikation und Verifikation

Beweis f¨ur: Addition von Peano-Zahlen ist assoziativ

•zu zeigen ist

plus a (plus b c) == plus (plus a b) c

•Beweismethode: Induktion (nachx)

und Umformen mit Gleichungen (¨aquiv. zu Implement.) plus Z y = y

plus (S x’) y = S (plus x’ y)

•Anfang:plus Z (plus b c) == ..

•Schritt:plus (S a’) (plus b c) ==

== S (plus a’ (plus b c)) == ..

– Typeset by Foil 55

(8)

¨Ubung Pattern Matching, Programme

•F¨ur die Deklarationen

-- data Bool = False | True (aus Prelude) data T = F T | G T T T | C

entscheide/bestimme f¨ur jeden der folgenden Ausdr¨ucke:

syntaktisch korrekt?

statisch korrekt?

Resultat (dynamische Semantik) disjunkt? vollst¨andig?

1. case False of { True -> C } 2. case False of { C -> True }

– Typeset by FoilTEX 56

3. case False of { False -> F F }

4. case G (F C) C (F C) of { G x y z -> F z } 5. case F C of { F (F x) -> False }

6. case F C of { F x -> False ; True -> False } 7. case True of { False -> C ; True -> F C } 8. case True of { False -> C ; False -> F C }

9. case C of { G x y z -> False; F x -> False; C -> True }

•Operationen auf Wahrheitswerten:

import qualified Prelude

data Bool = False | True deriving Prelude.Show not :: Bool -> Bool -- Negation

not x = case x of ... -> ...

– Typeset by FoilTEX 57

... -> ...

Syntax: wenn nachofkein{folgt:

implizite{ ; }durchAbseitsregel (layout rule).

•(&&) :: Bool -> Bool -> Bool x && y = case ... of ...

Syntax: Funktionsname

beginnt mit Buchstabe: steht vor Argumenten, beginnt mit Zeichen: zwischen Argumenten (als

Operator)

Operator als Funktion:(&&) False True, Funktion als Operator:True ‘f‘ False.

•Listen von Wahrheitswerten:

– Typeset by FoilTEX 58

data List = Nil | Cons Bool List deriving Prelude.Show and :: List -> Bool

and l = case l of ...

entsprechendor :: List -> Bool

•(Wdhlg.) welche Signatur beschreibt bin¨are B¨aume (jeder Knoten hat 2 oder 0 Kinder, die B¨aume sind; es gibt keine Schl¨ussel)

•geben Sie die dazu ¨aquivalentedata-Deklaration an:

data T = ...

•implementieren Sie daf¨ur die Funktionen size :: T -> Prelude.Int

– Typeset by FoilTEX 59

depth :: T -> Prelude.Int benutzePrelude.+(das ist Operator), Prelude.min, Prelude.max

•f¨ur Peano-Zahlendata N = Z | S N implementieren Sieplus, mal, min, max

– Typeset by FoilTEX 60

Hausaufgaben (f ¨ur KW 19)

1. F¨ur die Deklarationen

-- data Bool = False | True (aus Prelude) data S = A Bool | B | C S S

entscheide/bestimme f¨ur jeden der folgenden Ausdr¨ucke:

•syntaktisch korrekt?

•Resultat-Typ (statische Semantik)

•Resultat-Wert (dynamische Semantik)

•Menge der Muster ist: disjunkt? vollst¨andig?

1. case False of { True -> B } 2. case False of { B -> True } 3. case C B B of { A x -> x }

– Typeset by FoilTEX 61

4. case A True of { A x -> False }

5. case A True of { A x -> False ; True -> False } 6. case True of { False -> A ; True -> A False } 7. case True of { False -> B ; False -> A False } 8. case B of { C x y -> False; A x -> x; B -> True } 2. f¨ur selbst definierte Wahrheitswerte (vgl.

¨Ubungsaufgabe): deklarieren, implementieren und testen Sie eine zweistellige Funktion ”exclusiv-oder“ (mit Namen xor)

3. f¨ur bin¨are B¨aume ohne Schl¨ussel (vgl. ¨Ubungsaufgabe):

deklarieren, implementieren und testen Sie ein einstelliges Pr¨adikat ¨uber solchen B¨aumen, das genau

– Typeset by Foil 62

dann wahr ist, wenn das Argument eine gerade Anzahl von Bl¨attern enth¨alt.

4. Peano-Zahlen: siehe autotool und:

Beweisen Sie, daß unsere Implementierung der Addition kommutativ ist. Hinweis: dazu ist ein Hilfssatz n¨otig, in dessen BehauptungZvorkommt.

bg

– Typeset by Foil 63

(9)

Polymorphie

Definition, Motivation

•Beispiel: bin¨are B¨aume mit Schl¨ussel vom Type data Tree e = Leaf

| Branch (Tree e) e (Tree e) Branch Leaf True Leaf :: Tree Bool

Branch Leaf 42 Leaf :: Tree Int

•Definition:

ein polymorpher Datentyp ist einTypkonstruktor (=eine Funktion, die Typen auf einen Typ abbildet)

•unterscheide:Treeist der Typkonstruktor,Branchist ein Datenkonstruktor

– Typeset by FoilTEX 64

Beispiele f. Typkonstruktoren (I)

•Kreuzprodukt:

data Pair a b = Pair a b

•disjunkte Vereinigung:

data Either a b = Left a | Right b

•data Maybe a = Nothing | Just a

•Haskell-Notation f¨ur Produkte:

(1,True)::(Int,Bool) f¨ur0,2,3, . . . Komponenten

– Typeset by FoilTEX 65

Beispiele f. Typkonstruktoren (II)

•bin¨are B¨aume (Schl¨ussel in der Verzweigungsknoten) data Bin a = Leaf

| Branch (Bin a) a (Bin a)

•einfach (vorw¨arts) verkettete Listen data List a = Nil

| Cons a (List a)

•B¨aume mit Knoten beliebiger Stelligkeit, Schl¨ussel in jedem Knoten

data Tree a = Node a (List (Tree a))

– Typeset by FoilTEX 66

Polymorphe Funktionen

Beispiele:

•Spiegelbild einer Liste:

reverse :: forall e . List e -> List e

•Verkettung von Listen mit gleichem Elementtyp:

append :: forall e . List e -> List e -> List e

Knotenreihenfolge eines Bin¨arbaumes:

preorder :: forall e . Bin e -> List e Def: der Typ einer polymorphen Funktion beginnt mit All-Quantoren f¨ur Typvariablen.

Bsp: Datenkonstruktoren polymorpher Typen.

– Typeset by FoilTEX 67

Bezeichnungen f. Polymorphie

data List e = Nil | Cons e (List e)

•Listist einTypkonstruktor

•List eist einpolymorpher Typ (ein Typ-Ausdruck mitTyp-Variablen)

•List Boolist einmonomorpherTyp

(entsteht durchInstantiierung: Substitution der Typ-Variablen durch Typen)

•polymorphe Funktion:

reverse:: forall e . List e -> List e monomorphe Funktion:xor:: List Bool -> Bool polymorphe Konstante:Nil::forall e. List e

– Typeset by FoilTEX 68

Operationen auf Listen (I)

data List a = Nil | Cons a (List a)

•append xs ys = case xs of

Nil ->

Cons x xs’ ->

• ¨U: formuliere, teste und beweise:appendist assoziativ.

•reverse xs = case xs of

Nil ->

Cons x xs’ ->

• ¨U: beweise:

forall xs ys : reverse (append xs ys)

== append (reverse ys) (reverse xs)

– Typeset by FoilTEX 69

Von der Spezifikation zur Implementierung (II)

Bsp: homogene Listen

data List a = Nil | Cons a (List a) Aufgabe: implementieremaximum :: List N -> N Spezifikation:

maximum (Cons x1 Nil) = x1

maximum (append xs ys) = max (maximum xs) (maximum ys)

•substitutierexs = Nil, erhalte

maximum (append Nil ys) = maximum ys

= max (maximum Nil) (maximum ys)

d.h.maximum Nilsollte das neutrale Element f¨urmax (auf nat¨urlichen Zahlen) sein, also0(geschriebenZ).

– Typeset by Foil 70

•substitutierexs = Cons x1 Nil, erhalte maximum (append (Cons x1 Nil) ys)

= maximum (Cons x1 ys)

= max (maximum (Cons x1 Nil)) (maximum ys)

= max x1 (maximum ys)

Damit kann der aus dem Typ abgeleitete Quelltext maximum :: List N -> N

maximum xs = case xs of

Nil ->

Cons x xs’ ->

erg¨anzt werden.

Vorsicht: f¨urmin, minimumfunktioniert das nicht so, denn minhat f¨urNkein neutrales Element.

– Typeset by Foil 71

(10)

Operationen auf Listen (II)

•Die vorige Implementierung vonreverseist (f¨ur einfach verkettete Listen) nicht effizient (sondern quadratisch, vgl.

https://accidentallyquadratic.tumblr.com/)

•Besser ist Verwendung einer Hilfsfunktion reverse xs = rev_app xs Nil mit Spezifikation

rev_app xs ys = append (reverse xs) ys

•noch besser ist es,keineListen zu verwenden

https://www.imn.htwk-leipzig.de/˜waldmann/

etc/untutorial/list-or-not-list/

– Typeset by FoilTEX 72

Operationen auf B¨aumen

data List e = Nil | Cons e (List e)

data Bin e = Leaf | Branch (Bin e) e (Bin e) Knotenreihenfolgen

•preorder :: forall e . Bin e -> List e preorder t = case t of ...

•entsprechendinorder,postorder

•und Rekonstruktionsaufgaben

Adressierug von Knoten (False=links,True=rechts)

•get :: Bin e -> List Bool -> Maybe e

•positions :: Bin e -> List (List Bool)

– Typeset by FoilTEX 73

Statische Typisierung und Polymorphie

•Def: dynamische Typisierung:

die Daten (zur Laufzeit des Programms, im Hauptspeicher) haben einen Typ

•Def: statische Typisierung:

Bezeichner, Ausdr¨ucke (im Quelltext) haben einen Typ, dieser wird zur ¨Ubersetzungszeit (d.h., ohne

Programmausf¨uhrung) bestimmt

f¨urjedeAusf¨uhrung des Programms gilt:

der statische Typ eines Ausdrucks ist gleich dem dynamischen Typ seines Wertes

– Typeset by FoilTEX 74

Bsp. f ¨ur Programm ohne statischen Typ

•Javascript

function f (x) { if (x > 0) {

return function () { return 42; } } else { return "foobar"; } } }

Dann: Auswertung vonf(1)()ergibt 42, Auswertung vonf(0)()ergibt Laufzeit-Typfehler.

•entsprechendes Haskell-Programm ist statisch fehlerhaft f x = case x > 0 of

True -> \ () -> 42 False -> "foobar"

– Typeset by FoilTEX 75

Nutzen der stat. Typisierung und Polymorphie

•Nutzen der statischen Typisierung:

beim Programmieren: Entwurfsfehler werden zu Typfehlern, diese werden zur Entwurfszeit automatisch erkannt⇒fr¨uher erkannte Fehler lassen sich leichter beheben

beim Ausf¨uhren: keine Lauzeit-Typfehler⇒keine Typpr¨ufung zur Laufzeit n¨otig, effiziente Ausf¨uhrung

•Nutzen der Polymorphie:

Flexibilit¨at, nachnutzbarer Code, z.B. Anwender einer Collection-Bibliothek legt Element-Typ fest (Entwickler der Bibliothek kennt den Element-Typ nicht)

gleichzeitig bleibt statische Typsicherheit erhalten

– Typeset by FoilTEX 76

Konstruktion von Objekten eines Typs

Aufgabe (Bsp):

x :: Either (Maybe ()) (Pair Bool ()) L¨osung (Bsp):

•der TypEither a bhat Konstruktoren Left a | Right b. W¨ahleRight b.

Die Substitution f¨ur die Typvariablen ist a = Maybe (), b = Pair Bool ().

x = Right ymity :: Pair Bool ()

•der TypPair a bhat KonstruktorPair a b.

die Substitution f¨ur diese Typvariablen ist a = Bool, b = ().

– Typeset by FoilTEX 77

y = Pair p qmitp :: Bool, q :: ()

•der TypBoolhat KonstruktorenFalse | True, w¨ahle p = False. der Typ()hat Konstruktor(), alsoq=() Insgesamtx = Right y = Right (Pair False ()) Vorgehen (allgemein)

•bestimme den Typkonstruktor

•bestimme die Substitution f¨ur die Typvariablen

•w¨ahle einen Datenkonstruktor

•bestimme Anzahl und Typ seiner Argumente

•w¨ahle Werte f¨ur diese Argumente nach diesem Vorgehen.

– Typeset by Foil 78

Bestimmung des Typs eines Bezeichners

Aufgabe (Bsp.) bestimme Typ vonx(erstes Arg. vonget):

at :: Position -> Tree a -> Maybe a at p t = case t of

Node f ts -> case p of Nil -> Just f

Cons x p’ -> case get x ts of Nothing -> Nothing

Just t’ -> at p’ t’

L¨osung:

•bestimme das Muster, durch welchesxdeklariert wird.

L¨osung:Cons x p’ ->

•bestimme den Typ diese Musters

– Typeset by Foil 79

(11)

L¨osung: ist gleich dem Typ der zugeh¨origen Diskriminantep

•bestimme das Muster, durch daspdeklariert wird L¨osung:at p t =

•bestimme den Typ vonp

L¨osung: durch Vergleich mit Typdeklaration vonat(pist das erste Argument)p :: Position, also

Cons x p’ :: Position = List N, alsox :: N.

Vorgehen zur Typbestimmung eines Namens:

•finde die Deklaration (Muster einer Fallunterscheidung oder einer Funktionsdefinition)

•bestimme den Typ des Musters (Fallunterscheidung: Typ der Diskriminante, Funktion: deklarierter Typ)

– Typeset by FoilTEX 80

¨Ubung Polymorphie

Geben Sie alle Elemente dieser Datentypen an:

•Maybe ()

•Maybe (Bool, Maybe ())

•Either ((),Bool) (Maybe (Maybe Bool)) Operationen auf Listen:

•append, reverse, rev app Operationen auf B¨aumen:

•preorder, inorder

•get, (positions)

– Typeset by FoilTEX 81

Quelltexte aus Vorlesung:https://gitlab.imn.

htwk-leipzig.de/waldmann/fop-ss18

– Typeset by FoilTEX 82

Hausaufgaben (f ¨ur KW 20)

1. f¨ur die folgenden Datentypen: geben Sie einige Elemente an (ghci), geben Sie die Anzahl aller Elemente an (siehe auch autotool-Aufgabe)

(a)Maybe (Maybe Bool)

(b)Either (Bool, ()) (Maybe ()) (c)Foo (Maybe (Foo Bool))mit

data Foo a = C a | D (d) (Zusatz)T ()mit

data T a = L a | B (T (a,a))

2. Implementieren Sie die Post-Order Durchquerung von Bin¨arb¨aumen.

– Typeset by FoilTEX 83

(Zusatz: Level-Order. Das ist schwieriger.) 3. Beweisen Sie

forall xs . reverse (reverse xs) == xs Sie d¨urfen

reverse (append xs ys)

== append (reverse ys) (reverse xs) ohne Beweis verwenden.

– Typeset by FoilTEX 84 – Typeset by FoilTEX 85

– Typeset by Foil 86 – Typeset by Foil 87

(12)

Funktionen

Funktionen als Daten

•bisher: Funktion beschrieben durch Regel(menge) dbl x = plus x x

•jetzt: durch Lambda-Termdbl = \ x -> plus x x λ-Terme: mit lokalen Namen (hier:x)

•Funktionsanwendung:(λx.B)A→B[x:=A]

freie Vorkommen vonxin B werden durchAersetzt

•Funktionen sind Daten (Bsp:Cons dbl Nil)

•λ-Kalk¨ul: Alonzo Church 1936, Henk Barendregt 198*

Henk Barendregt:The Impact of the Lambda Calculus in Logic and Computer Science, Bull. Symbolic Logic, 1997.

– Typeset by FoilTEX 88

https://citeseerx.ist.psu.edu/viewdoc/summary?doi=

10.1.1.25.9348

– Typeset by FoilTEX 89

Der Lambda-Kalk ¨ul

•ist ein Berechnungsmodell,

vgl. Termersetzungssysteme, Turingmaschine, Random-Access-Maschine (=Goto-Programme)

•Syntax: die Menge der Lambda-TermeΛist jede Variable ist ein Term:v∈V ⇒v∈Λ Funktionsanwendung (Applikation):

F ∈Λ, A∈Λ⇒(F A)∈Λ Funktionsdefinition (Abstraktion):

v∈V, B∈Λ⇒(λv.B)∈Λ

•Semantik: eine Relation→βaufΛ (vgl.→Rf¨ur TermersetzungssystemR)

– Typeset by FoilTEX 90

Freie und gebundene Variablen(vorkommen)

•Das Vorkommen vonv∈V an Positionpin Termtheißt frei, wenn ”dar¨uber keinλv. . . . steht“

•Def.fvar(t) =Menge der intfrei vorkommenden Variablen (definiere durch strukturelle Induktion)

•Eine Variablexheißt inAgebunden, fallsAeinen Teilausdruckλx.Benth¨alt.

•Def.bvar(t) =Menge der intgebundenen Variablen Bsp:fvar(x(λx.λy.x)) ={x},bvar(x(λx.λy.x)) ={x, y},

– Typeset by FoilTEX 91

Semantik des Lambda-Kalk ¨uls: Reduktion

β

Relation→βaufΛ(ein Reduktionsschritt) Es giltt→βt0, falls

• ∃p∈Pos(t), so daß

•t[p] = (λx.B)Amitbvar(B)∩fvar(A) =∅

•t0=t[p:=B[x:=A]]

dabei bezeichnetB[x:=A]ein Kopie vonB, bei der jedes freie Vorkommen vonxdurchAersetzt ist Ein (Teil-)Ausdruck der Form(λx.B)AheißtRedex.

(Dort kann weitergerechnet werden.) Ein Term ohne Redex heißtNormalform.

(Normalformen sind Resultate von Rechnungen.)

– Typeset by FoilTEX 92

Falsches Binden lokaler Variablen

•dieser Ausdruck hat den Wert 15:

(\x->(((\f->\x->x + f 8) (\y-> x+y)) 4)) 3

•Redex(λf.B)AmitB=λx.x+f8undA=λy.x+y:

•dort keine→β-Reduktion,bvar(B)∩fvar(A) ={x} 6=∅.

•falls wir die Nebenbedingung ignorieren, erhalten wir (\x->(( \x->x + (\y-> x+y) 8) 4)) 3 mit Wert 16.

•dieses Beispiel zeigt, daß die Nebenbedingung semantische Fehler verhindert

– Typeset by FoilTEX 93

Semantik . . . : gebundene Umbenennung

α

•falls wir einen Redex(λx.B)Areduzieren m¨ochten, f¨ur denbvar(B)∩fvar(A) =∅nicht gilt,

dann vorher dort die lokale Variablexumbenennen (hinter demλund jedes freie Vorkommen vonxinB)

•Relation→αaufΛ, beschreibtgebundene Umbenennung einer lokalen Variablen.

•Beispielλx.f xz→αλy.f yz.

(fundzsind frei, k¨onnen nicht umbenannt werden)

•Definitiont→αt0:

∃p∈Pos(t), so daßt[p] = (λx.B) y /∈bvar(B)∪fvar(B)

t0=t[p:=λy.B[x:=y]]

– Typeset by Foil 94

Umbenennung von lokalen Variablen

int x = 3;

int f(int y) { return x + y; }

int g(int x) { return (x + f(8)); } // g(5) => 16

•Darff(8)ersetzt werden durchf[y:= 8]? - Nein:

int x = 3;

int g(int x) { return (x + (x+8)); } // g(5) => 18

Das freiexin(x+y)wird f¨alschlich gebunden.

•L¨osung: lokal umbenennen

int g(int z) { return (z + f(8)); }

dann ist Ersetzung erlaubt

int x = 3;

int g(int z) { return (z + (x+8)); } // g(5) => 16

– Typeset by Foil 95

(13)

Lambda-Terme: verk ¨urzte Notation

•Applikation ist links-assoziativ, Klammern weglassen:

(. . .((F A1)A2). . . An)∼F A1A2. . . An

Beispiel:((xz)(yz))∼xz(yz) Wirkt auch hinter dem Punkt:

(λx.xx)bedeutet(λx.(xx))— und nicht((λx.x)x)

•geschachtelte Abstraktionen unter ein Lambda schreiben:

(λx1.(λx2. . . .(λxn.B). . .))∼λx1x2. . . xn.B Beispiel:λx.λy.λz.B∼λxyz.B

– Typeset by FoilTEX 96

Ein- und mehrstellige Funktionen

eine einstellige Funktion zweiter Ordnung:

f = \ x -> ( \ y -> ( x*x + y*y ) ) Anwendung dieser Funktion:

(f 3) 4 = ...

Kurzschreibweisen (Klammern weglassen):

f = \ x y -> x * x + y * y ; f 3 4

¨Ubung:

gegebent = \ f x -> f (f x) bestimmet succ 0, t t succ 0, t t t succ 0, t t t t succ 0, ...

– Typeset by FoilTEX 97

Typen

f¨ur nicht polymorphe Typen: tats¨achlicher Argumenttyp muß mit deklariertem Argumenttyp ¨ubereinstimmen:

wennf::A→Bundx::A, dann(f x) ::B.

bei polymorphen Typen k¨onnen der Typ vonf ::A→B und der Typ vonx::A0Typvariablen enthalten.

Beispiel:λx.x::∀t.t→t.

Dann m¨ussenAundA0nicht ¨ubereinstimmen, sondern nur unifizierbar sein (eine gemeinsame Instanz besitzen).

Beispiel:(λx.x)True

benutze Typ-Substitutionσ={(t,Bool)}.

Bestimme allgemeinsten Typ vont=λf x.f(f x)), von(tt).

– Typeset by FoilTEX 98

Beispiel f ¨ur Typ-Bestimmung

Aufgabe: bestimme den allgemeinsten Typ vonλf x.f(f x)

•Ansatz mit Typvariablenf::t1, x::t2

•betrachte(f x): der Typ vonfmuß ein Funktionstyp sein, alsot1= (t11→t12)mit neuen Variablent11, t12.

Dann giltt11=t2und(f x) ::t12.

•betrachtef(f x). Wir habenf::t11→t12und(f x) ::t12, also folgtt11=t12. Dannf(f x) ::t12.

•betrachteλx.f(f x).

Ausx::t12undf(f x) ::t12folgtλx.f(f x) ::t12→t12.

•betrachteλf.(λx.f(f x)).

Ausf ::t12→t12undλx.f(f x) ::t12→t12

folgtλf x.f(f x) :: (t12→t12)→(t12→t12)

– Typeset by FoilTEX 99

Verk ¨urzte Notation f ¨ur Typen

•Der Typ-Pfeil istrechts-assoziativ:

T1→T2→ · · · →Tn→T bedeutet (T1→(T2→ · · · →(Tn→T)· · ·))

•das paßt zu den Abk¨urzungen f¨ur mehrstellige Funktionen:

λ(x::T1).λ(x::T2).(B::T) hat den Typ(T1→(T2→T)), mit o.g. Abk¨urzungT1→T2→T.

– Typeset by FoilTEX 100

Lambda-Ausdr ¨ucke in C#

•Beispiel (Fkt. 1. Ordnung)

Func<int,int> f = (int x) => x*x;

f (7);

• ¨Ubung (Fkt. 2. Ordnung) — erg¨anze alle Typen:

??? t = (??? g) => (??? x) => g (g (x));

t (f)(3);

•Anwendungen bei Streams, sp¨ater mehr

(new int[]{3,1,4,1,5,9}).Select(x => x * 2);

(new int[]{3,1,4,1,5,9}).Where(x => x > 3);

• ¨Ubung: Diskutiere statische/dynamische Semantik von

(new int[]{3,1,4,1,5,9}).Select(x => x > 3);

(new int[]{3,1,4,1,5,9}).Where(x => x * 2);

– Typeset by FoilTEX 101

Lambda-Ausdr ¨ucke in Java

•funktionalesInterface (FI): hat genau eine Methode

•Lambda-Ausdruck (”burger arrow“) erzeugt Objekt eineranonymen Klasse, die FI implementiert.

interface I { int foo (int x); } I f = (x)-> x+1;

System.out.println (f.foo(8));

•vordefinierte FIs:

import java.util.function.*;

Function<Integer,Integer> g = (x)-> x*2;

System.out.println (g.apply(8));

Predicate<Integer> p = (x)-> x > 3;

if (p.test(4)) { System.out.println ("foo"); }

– Typeset by Foil 102

Lambda-Ausdr ¨ucke in Javascript

$ node

> let f = function (x){return x+3;}

undefined

> f(4) 7

> ((x) => (y) => x+y) (3) (4) 7

> ((f) => (x) => f(f(x))) ((x) => x+1) (0) 2

– Typeset by Foil 103

(14)

Beispiele Fkt. h ¨oherer Ord.

•Haskell-Notation f¨ur Listen:

data List a = Nil | Cons a (List a) data [a] = [] | a : [a]

•Verarbeitung von Listen:

filter :: (a -> Bool) -> [a] -> [a]

takeWhile :: (a -> Bool) -> [a] -> [a]

partition :: (a -> Bool) -> [a] -> ([a],[a])

•Vergleichen, Ordnen:

nubBy :: (a -> a -> Bool) -> [a] -> [a]

data Ordering = LT | EQ | GT minimumBy

:: (a -> a -> Ordering) -> [a] -> a

– Typeset by FoilTEX 104

Fkt. h ¨oherer Ord. f ¨ur Folgen

•vgl.https://www.imn.htwk-leipzig.de/

˜waldmann/etc/untutorial/list-or-not-list/

•Folgen, repr¨asentiert als balancierte B¨aume:

module Data.Sequence where data Seq a = ...

-- keine sichtbaren Konstruktoren!

fromList :: [a] -> Seq a

filter :: (a -> Bool) -> Seq a -> Seq a takeWhile :: (a -> Bool) -> Seq a -> Seq a

•Anwendung:

import qualified Data.Sequence as Q xs = Q.fromList [1, 4, 9, 16]

ys = Q.filter (\x -> 0 == mod x 2) xs

– Typeset by FoilTEX 105

Fkt. h ¨oherer Ord. f ¨ur Mengen

•Mengen, repr¨asentiert als balancierte Such-B¨aume:

module Data.Set where data Set a = ...

-- keine sichtbaren Konstruktoren!

fromList :: Ord a => [a] -> Set a

filter :: Ord a => (a -> Bool) -> Set a -> Set a dasTyp-ConstraintOrd aschr¨ankt die Polymorphie ein

(der Typ, durch den die Typ-Variableainstantiiert wird, muß eine Vergleichsmethode haben)

•Anwendung:

import qualified Data.Set as S xs = S.fromList [1, 4, 9, 16]

ys = S.filter (\x -> 0 == mod x 2) xs

– Typeset by FoilTEX 106

N ¨utzliche Funktionen h ¨oherer Ordnung

•compose :: (b -> c) -> (a -> b) -> a -> c aus dem Typ folgt schon die Implementierung!

compose f g x = ...

diese Funktion in der Standard-Bibliothek:

der Operator.(Punkt)

•apply :: (a -> b) -> a -> b apply f x = ...

das ist der Operator$(Dollar) . . . istrechts-assoziativ

•flip :: ...

flip f x y = f y x

wie lautet der (allgemeinste) Typ?

– Typeset by FoilTEX 107

Stelligkeit von Funktionen

•istplusinfliprichtig benutzt? Ja!

flip :: (a -> b -> c) -> b -> a -> c data N = Z | S N

plus :: N -> N -> N

plus (S Z) (S (S Z)) ; flip plus (S Z) (S (S Z))

•beachte Unterschied zwischen:

Term-Ersetzung: Funktionssymbol→Stelligkeit abstrakter Syntaxbaum: Funktionss. ¨uber Argumenten Lambda-Kalk¨ul: nur einstellige Funktionen

AST: Applikationsknoten, Funkt.-Symb. links unten.

Simulation mehrstelliger Funktionen wegen

Isomorphie zwischen(A×B)→CundA→(B→C)

•case: Diskriminante u. Muster m¨ussendata-Typ haben

– Typeset by FoilTEX 108

¨Ubung Lambda-Kalk¨ul

•abstrakten Syntaxbaum und Normalform vonSKKc, wobeiS=λxyz.xz(yz), K=λab.a,

•(mitdata N=Z|S N) bestimme Normalform vonttSZ f¨urt=λf x.f(f x),

•definiereΛals algebraischen Datentyp

data L = Var String | App L L | Abs String L

implementieresize :: L -> Int, depth :: L -> Int.

implementierebvar :: L -> S.Set String, fvar :: L -> S.Set String,

– Typeset by FoilTEX 109

siehe Folie mit Definitionen und dort angegebene Testf¨alle

benutzeimport qualified Data.Set as S, API-Dokumentation:https://hackage.haskell.

org/package/containers/docs/Data-Set.html Teill¨osung:

bvar :: L -> S.Set String bvar t = case t of

Var v -> S.empty v

App l r -> S.union (bvar l) (bvar r) Abs v b -> S.insert v (bvar b)

•den allgemeinsten Typ eines Lambda-Ausdrucks bestimmen, Beispiel

– Typeset by Foil 110

compose ::

compose = \ f g -> \ x -> f (g x) Musterl¨osung:

wegeng xmußg :: a -> bgelten, dannx :: aundg x :: b

wegenf (g x)mußf :: b -> cgelten, dannf (g x):: c

dann\ x -> f (g x) :: a -> c dann

\ f g -> .. :: (b->c) -> (a->b) -> (a->c)

•Implementierung von takeWhile

takeWhile :: (a -> Bool) -> List a -> List a

– Typeset by Foil 111

(15)

takeWhile p xs = case xs of Nil -> Nil

Cons x xs’ -> case p x of False -> Nil

True -> Cons x (takeWhile p xs’)

– Typeset by FoilTEX 112

Hausaufgaben f ¨ur KW 21

1. (autotool) Reduktion im Lambda-Kalk¨ul 2.fvarimplementieren (vgl. ¨Ubungsaufgabe)

3. Normalform eines Lambda-Ausdrucks berechnen (an der Tafel, der Ausdruck wird erst dann gegeben)

4. den allgemeinsten Typ eines Lambda-Ausdrucks bestimmen (an der Tafel, der Ausdruck wird erst dann gegeben)

5. (autotool) Implementierung vondropWhileo.¨a.

6. Beweisen Sie f¨ur diese Implementierung

xs=append (takeWhile p xs) (dropWhile p xs)

– Typeset by FoilTEX 113

Rekursionsmuster

Rekursion ¨uber B¨aume (Beispiele)

data Tree a = Leaf

| Branch (Tree a) a (Tree a) summe :: Tree Int -> Int

summe t = case t of Leaf -> 0

Branch l k r -> summe l + k + summe r preorder :: Tree a -> List a

preorder t = case t of Leaf -> Nil

Branch l k r ->

Cons k (append (preorder l) (preorder r))

– Typeset by FoilTEX 114

Rekursion ¨uber B¨aume (Schema)

f :: Tree a -> b

f t = case t of Leaf -> ...

Branch l k r -> ... (f l) k (f r) dieses Schemaisteine Funktion h¨oherer Ordnung:

fold :: ( ... ) -> ( ... ) -> ( Tree a -> b ) fold leaf branch = \ t -> case t of

Leaf -> leaf Branch l k r ->

branch (fold leaf branch l) k (fold leaf branch r)

summe = fold 0 ( \ l k r -> l + k + r )

– Typeset by FoilTEX 115

Rekursion ¨uber Listen

and :: List Bool -> Bool

and xs = case xs of

Nil -> True ; Cons x xs’ -> x && and xs’

length :: List a -> N length xs = case xs of

Nil -> Z ; Cons x xs’ -> S (length xs’) fold :: b -> ( a -> b -> b ) -> List a -> b fold nil cons xs = case xs of

Nil -> nil

Cons x xs’ -> cons x ( fold nil cons xs’ ) and = fold True (&&)

length = fold Z ( \ x y -> S y)

– Typeset by FoilTEX 116

Rekursionsmuster (Prinzip)

data List a = Nil | Cons a (List a) fold ( nil :: b ) ( cons :: a -> b -> b )

:: List a -> b Rekursionsmuster anwenden

=jeden Konstruktor durch eine passende Funktion ersetzen

=(Konstruktor-)Symboleinterpretieren(durch Funktionen)

=eineAlgebraangeben.

length = fold Z ( \ _ l -> S l ) reverse = fold Nil ( \ x ys -> )

– Typeset by FoilTEX 117

Rekursionsmuster (Merks¨atze)

aus dem Prinzipein Rekursionsmuster anwenden=jeden Konstruktor durch eine passende Funktion ersetzenfolgt:

•Anzahl der Muster-Argumente=Anzahl der Konstruktoren (plus eins f¨ur das Datenargument)

•Stelligkeit eines Muster-Argumentes=Stelligkeit des entsprechenden Konstruktors

•Rekursion im Typ⇒Rekursion im Muster (Bsp: zweites Argument vonCons)

•zu jedem rekursiven Datentyp gibt esgenau ein passendes Rekursionsmuster

– Typeset by Foil 118

Rekursion ¨uber Listen ( ¨Ubung)

das vordefinierte Rekursionsschema ¨uber Listen ist:

foldr :: (a -> b -> b) -> b -> ([a] -> b) length = foldr ( \ x y -> 1 + y ) 0 Beachte:

•Argument-Reihenfolge (erst cons, dann nil)

•foldr nicht mit foldl verwechseln (foldr ist das ”richtige“) Aufgaben:

•append, reverse, concat, inits, tails mitfoldr(d. h., ohne Rekursion)

– Typeset by Foil 119

(16)

Weitere Beispiele f ¨ur Folds

data Tree a

= Leaf a | Branch (Tree a) (Tree a) fold :: ...

•Anzahl der Bl¨atter

•Anzahl der Verzweigungsknoten

•Summe der Schl¨ussel

•die Tiefe des Baumes

•der gr¨oßte Schl¨ussel

– Typeset by FoilTEX 120

Rekursionsmuster (Peano-Zahlen)

data N = Z | S N

fold :: ...

fold z s n = case n of Z ->

S n’ ->

plus = fold ...

times = fold ...

– Typeset by FoilTEX 121

Spezialf¨alle des Fold

•jeder Konstruktor durch sich selbst ersetzt, mit unver¨anderten Argumenten:identischeAbbildung data List a = Nil | Cons a (List a) fold :: r -> (a -> r -> r) -> List a -> r fold Nil Cons (Cons 3 (Cons 5 Nil))

•jeder Konstruktor durch sich, mit transformierten Argumenten:

fold Nil (\x y -> Cons (not x) y) (Cons True (Cons False Nil)) struktur-erhaltendeAbbildung. Diese heißtmap.

– Typeset by FoilTEX 122

Argumente f ¨ur Rekursionsmuster finden

Vorgehen zur L¨osung der Aufgabe:

”Schreiben Sie Funktionf :T →Ralsfold“

•eine Beispiel-Eingabe (t∈T) notieren (Baum zeichnen)

•f¨ur jeden Teilbaumsvont, der den TypT hat:

den Wert vonf(s)in (neben) Wurzel vonsschreiben

•daraus Testf¨alle f¨ur die Funktionen ableiten, die Argumente des Rekursionsmusters sind.

Beispiel:data N = Z | S N, f:N→Bool,f(x) =”xist ungerade“

– Typeset by FoilTEX 123

Nicht durch Rekursionmuster darstellbare Fkt.

•Beispiel:data N = Z | S N,

f:N→Bool,f(x) =”xist durch 3 teilbar“

•wende eben beschriebenes Vorgehen an,

•stelle fest, daß die durch Testf¨alle gegebene Spezifikation nicht erf¨ullbar ist

•Beispiel: bin¨are B¨aume mit Schl¨ussel in Verzweigungsknoten,

f:Tree k→Bool,

f(t) =”tist h¨ohen-balanciert (erf¨ullt die AVL-Bedingung)“

– Typeset by FoilTEX 124

Darstellung als fold mit Hilfswerten

•f:Tree k→Bool,

f(t) =”tist h¨ohen-balanciert (erf¨ullt die AVL-Bedingung)“

ist nicht als fold darstellbar

•g:Tree k→Pair Bool Int g(t) = (f(t),height(t))

ist als fold darstellbar

– Typeset by FoilTEX 125

Weitere ¨Ubungsaufgaben zu Fold

•data List a = Nil | Cons a (List a) fold :: r -> (a -> r -> r) -> List a -> r

•schreibe mittelsfold(ggf. verwendemap)

inits, tails :: List a -> List (List a) inits [1,2,3] = [[],[1],[1,2],[1,2,3]]

tails [1,2,3] = [[1,2,3],[2,3],[3],[]]

filter :: (a -> Bool) -> List a -> List a filter odd [1,8,2,7,3] = [1,7,3]

partition :: (a -> Bool) -> List a -> Pair (List a) (List a) partition odd [1,8,2,7,3]

= Pair [1,7,3] [8,2]

– Typeset by Foil 126

¨Ubung Rekursionsmuster

•Rekursionsmusterfoldrf¨ur Listen benutzen (filter, takeWhile, append, reverse, concat, inits, tails)

•Rekursionmuster f¨ur Peano-Zahlen hinschreiben und benutzen (plus, mal, hoch, Nachfolger, Vorg¨anger, minus)

•Rekursionmuster f¨ur bin¨are B¨aume mit Schl¨usselnnur in den Bl¨atternhinschreiben und benutzen

•Rekursionmuster f¨ur bin¨are B¨aume mit Schl¨usselnnur in den Verzweigungsknotenbenutzen f¨ur rekursionslose Programme f¨ur:

Anzahl der Branch-Knoten ist ungerade (nicht z¨ahlen!) Baum (Tree a) erf¨ullt die AVL-Bedingung

– Typeset by Foil 127

Referenzen

ÄHNLICHE DOKUMENTE

[r]

[r]

Eine selbstorganisierende Liste ist eine Liste, bei der kurz aufeinanderfolgende Zugriffe auf dieselben Elemente sehr effizient sind: Wird ein Element in der Liste gefunden,

Eine selbstorganisierende Liste ist eine Liste, bei der kurz aufeinanderfolgende Zugriffe auf dieselben Elemente sehr effizient sind: Wird ein Element in der Liste gefunden,

Hinweis: Die Methode equals(Struct other) der Klasse Struct wird nur von den JUnit-Testfällen verwendet und prüft nicht die korrekte

Verfahren nach einem der Ansprüche 1 bis 8, da- durch gekennzeichnet, dass das Schüttgut mit dem überhitzten Wasserdampf auf eine Temperatur von vorzugsweise über 140 °C,

Bei den derzeitigen automatischen Lagerungs- systemen lassen sich zwei hauptsächliche Arten jnterscheiden (siehe Fig. Palette durch ein besonderes Gerät

Stelligkeit des entsprechenden Konstruktors I Rekursion im Typ → Rekursion im Muster I zu jedem rekursiven Datentyp gibt es genau ein.