Grundlagen der Programmierung 2 (2.A) Prof. Dr. Manfred Schmidt-Schauß Künstliche Intelligenz und Softwaretechnologie 5. Mai 2011 Listen und Listenfunktionen Listen modellieren Folgen von gleichartigen, gleichgetypten Objekten. Ausdruck im Programm [0,1,2,3,4,5,6,7,8,9] Erklärung Typ: [Integer]; d.h. Liste von Integer. [] leere Liste, (Nil) [’a’, ’b’, ’c’] Typ: [Char]; abgekürzt als String Druckbild: "abc" [[], [0], [1, 2]] Liste von Listen; Typ [[Integer]], d.h. eine Liste von Listen von Integer-Objekten. [1..] potentiell unendliche Liste der Zahlen 1, 2, 3, ... Grundlagen der Programmierung 2 (2.A) SS 2011 - 1 - Listen und Listenfunktionen zwei Schreibweisen für Listen: [0, 1, 2] schöne Darstellung Druckbild einer Liste (0 : (1 : (2 : []))) interne Darstellung mit zweistelligem Infix-Listen-Konstruktor : “ ” und dem Konstruktor [] Eingebaute, listenerzeugende Funktionen: Ausdruck im Programm [n..] [n..m] [1..10] [n,m..k] Grundlagen der Programmierung 2 (2.A) SS 2011 Erklärung erzeugt die Liste der Zahlen ab n. erzeugt die Liste von n bis m ergibt [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] erzeugt die Liste von n bis k mit Schritten m − n - 2 - Listen • Listendarstellung • Listenerzeugung • Listen-Druckbild Grundlagen der Programmierung 2 (2.A) SS 2011 - 3 - Darstellung von Listen Listen sind aufgebaut mittels zwei Konstruktoren: [] : Konstante für die leere Liste Zweistelliger Infix-Konstruktor a : b Linkes Argument a: erstes Element der Liste Rechtes Argument b: Restliste Beispiel für Haskells Listenerzeugung: 8:[] 9:(8:[]) 10:(9:(8:[])) Liste [8] mit dem Element 8 Liste [9,8] mit zwei Elementen 8,9 Liste [10,9,8] mit drei Elementen Grundlagen der Programmierung 2 (2.A) SS 2011 - 4 - Baum-Bild einer Liste || || | || || | || || | | |~ | : ??? 10 9 ?? ?? ?? ?? ?? ?? ?? ? : >>> 8 >> >> >> >> >> >> >> > : === == == == == == == == [] entspricht [10, 9, 8] Grundlagen der Programmierung 2 (2.A) SS 2011 - 5 - Einfache Listenfunktionen Definitionen head (x:xs) = x tail (x:xs) = xs Extrahiere das erste Element Extrahiere die Restliste Auswertungen Prelude> ????? Prelude> ????? Prelude> ????? Prelude> ????? head [] ←head [1] ←tail [] ←tail [1] ←- Grundlagen der Programmierung 2 (2.A) SS 2011 - 6 - Beispiel lengthr lengthr [] = 0 lengthr (x:xs) = 1 + (lengthr xs) Auswertung bei bereits ausgewerteter Liste lengthr (10:(9:(8:[]))) 1+ (lengthr (9:(8:[]))) 1+(1+ (lengthr (8:[]))) 1+(1+ (1+ (lengthr []))) 1+(1+ (1+ (0))) 3 Grundlagen der Programmierung 2 (2.A) SS 2011 Zweiter Fall; [10/x, (9:(8:[]))/xs] Zweiter Fall; [9/x, (8:[])/xs] Zweiter Fall; [8/x, ([])/xs] Erster Fall; 3 × Addition - 7 - Funktionen auf Listen: map map :: (a -> b) -> [a] -> [b] map f [] = [] map f (x:xs) = (f x) : (map f xs) map wendet eine Funktion f auf alle Elemente einer Liste an und konstruiert die Liste der Ergebnisse. [] und (x:xs) links von =“ sind Muster(Pattern) ” Z.B. Muster (x:xs) und Argument (s:t) ergibt die Ersetzung: [s/x, t/xs] Grundlagen der Programmierung 2 (2.A) SS 2011 - 8 - Funktionen auf Listen: Beispiele map f [] map f (x:xs) = [] = (f x) : (map f xs) Auswertung von map quadrat (1:(2:[])): map quadrat (1:(2:[])) quadrat 1 : map quadrat (2:[]) 1*1 : map quadrat (2:[]) 1 : map quadrat (2:[]) 1 : (quadrat 2 : map quadrat []) 1 : (2*2 : map quadrat []) 1 : (4 : map quadrat []) 1 : (4 : []) Grundlagen der Programmierung 2 (2.A) SS 2011 [quadrat/f, 1/x, (2:[])/xs] bei vollst. Auswertung: Zweite Gleichung wg Interpreter Erste Gleichung = [1,4] - 9 - Auswertung: Wieviel ist nötig? istLeer [] = True istLeer (x:xs) = False zahlenAb n Auswertung = n: zahlenAb (n+1) (mit Listenerzeuger als Argument) istLeer [1..] istLeer (zahlenAb 1) istLeer (1: zahlenAb (1+1)) False Grundlagen der Programmierung 2 (2.A) SS 2011 verwende zahlenAb Zweite Gleichung von istLeer - 10 - Listenfunktionen und Listenerzeuger *Main> map quadrat [1..10] ←[1,4,9,16,25,36,49,64,81,100] *Main> map quadrat [1..] ←[1,4,9,16,25,36,49,64,81,100,121, .... Der Listenerzeuger [1..] erzeugt soviel von der Liste [1,2,3,4,5, usw. wie von der Listenfunktion benötigt wird. Grundlagen der Programmierung 2 (2.A) SS 2011 - 11 - Typen von Listenausdrücken mapQuadrat xs = map quadrat xs *Main> :t mapQuadrat ←mapQuadrat :: forall a. (Num a) => [a] -> [a] mapLength xs = map length xs *Main> :t mapLength ←mapLength :: forall a. [[a]] -> [Int] Grundlagen der Programmierung 2 (2.A) SS 2011 - 12 - Listenfunktion append Die folgende Funktion hängt zwei Listen zusammen (genauer: sie konstruiert diese Liste) append :: [a] -> [a] -> [a] append [] ys = ys append (x:xs) ys = x : (append xs ys) Haskell-Operator für append: ++ (Infix-Operator) Haskell-Schreibweise: [1,2,3] ++ [4,5,6,7] Grundlagen der Programmierung 2 (2.A) SS 2011 - 13 - Beispiele *Main> [] ++ [3,4,5] ←[3,4,5] *Main> [0,1,2] ++ [] ←[0,1,2] *Main> [0,1,2] ++ [3,4,5] ←[0,1,2,3,4,5] *Main> [0..10000] ++ [10001..20000] == [0..20000] ←True Grundlagen der Programmierung 2 (2.A) SS 2011 - 14 - Funktionen auf Listen (2) Filtern von Elementen aus einer Liste: filter :: (a -> Bool) -> [a] -> [a] filter f [] = [] filter f (x:xs) = if (f x) then x : filter f xs else filter f xs Beispiele: *Main> filter (< 5) [1..10] ←[1,2,3,4] *Main> filter primzahlq [2..] ←[2,3,5,7,11,13,17,19,23,29,31,37,41,43,47,53,59,61,67,71, 73,79,83,89,97,101,103,107,109,113,127,131,137,139, 149,151,157,163,167,173,179,181,191,193,197,199,211, Grundlagen der Programmierung 2 (2.A) SS 2011 - 15 - Funktionen auf Listen Die ersten n Elemente der Liste xs: take take take take :: Int -> [a] -> [a] 0 _ = [] n [] = [] n (x:xs) = x : (take (n-1) xs) *Main> take 10 [20..40] ←[20,21,22,23,24,25,26,27,28,29] *Main> take 10 [20,23..] ←[20,23,26,29,32,35,38,41,44,47] Grundlagen der Programmierung 2 (2.A) SS 2011 - 16 - Auswertungsreihenfolge, Definitionseinsetzung Auswertung von f s1 . . . sn wenn Muster verwendet wurden: Vor Definitionseinsetzung diejenigen Argumente auswerten, die für die Fallunterscheidung benötigt werden. Aber nur soviel wie nötig Zuordnung: Mustervariablen zu Ausdruck analog wie Zuordnung: formale Parameter zu Argumenten. Grundlagen der Programmierung 2 (2.A) SS 2011 - 17 - Geschachtelte Pattern nub (aus Module List) eliminiert doppelte, benachbarte Vorkommen von Elementen aus Listen: nub [] = [] nub [x] = [x] nub (x:(y:r)) = if x == y then nub (y:r) else x : nub (y:r) Beachte das Pattern (x:(y:r)) Grundlagen der Programmierung 2 (2.A) SS 2011 - 18 - Listen: Auswertung Listen (bzw. Listenargumente) nennt man: einfach ausgewertet: wenn Listen-Fallunterscheidung möglich ist, d.h. [] oder von der Form s : t vollständig ausgewertet: wenn Liste endlich ist und jeder Tail der Liste mindestens einfach ausgewertet und alle Elemente ebenfalls vollständig ausgewertet sind, Grundlagen der Programmierung 2 (2.A) SS 2011 - 19 - Iterative Prozesse mit Listenargumenten Bei Verwendung von Listenargumenten: Die folgenden Begriffe sind unverändert: linear rekursiv, end-rekursiv (= tail-recursive) Baum-rekursiv geschachtelt Baum-rekursiv (Bei applikativer Reihenfolge der Auswertung) iterativ muss angepasst werden. Grundlagen der Programmierung 2 (2.A) SS 2011 - 20 - Iterativer Auswertungsprozess zu f Ein iterativer Auswertungsprozess liegt vor, bei rekursiver Funktion f , wenn: (f a1 . . . an) ∗ − → (3) (f a1 (3) . . . an ) (j) und alle ai ∗ − → sind (f a01 . . . a0n) ∗ − → ...... ∗ − → ∗ − → (2) (f a1 (m) (f a1 (2) . . . an ) (m) . . . an ) ∗ − → ... Basiswerte oder komplett ausgewertete, endliche Listen bei applikativer Reihenfolge der Auswertung Grundlagen der Programmierung 2 (2.A) SS 2011 - 21 - iterative Version fiter von f fiter ist iterative Version von f Wenn: f und fiter das gleiche berechnen und fiter einen iterativen Prozess erzeugt (unter applikativer R.) für alle Basiswerte und alle komplett ausgewerteten endlichen Listen als Eingaben Grundlagen der Programmierung 2 (2.A) SS 2011 - 22 - Beispiel: iterative Version von lengthr: length_lin xs = length_linr 0 xs length_linr s [] = s length_linr s (x:xs) = length_linr (s+1) xs nicht-iterative Version: lengthr [] = 0 lengthr (x:xs) = 1 + lengthr xs Grundlagen der Programmierung 2 (2.A) SS 2011 - 23 - Linearer (Nicht-iterativer) Prozess zu lengthr lengthr (9:(8:(7:(6:...(1:[]))))) 1+(lengthr (8:(7:(6:...(1:[])))) 1+(1+(lengthr (7:(6:...(1:[]))) 1+(1+(1+(lengthr (6:...(1:[])) .......... (1+(1+(1+(1+(1+(1+(1+(1+(1+0)))))))))) ..... 9 Grundlagen der Programmierung 2 (2.A) SS 2011 - 24 - Beispiel: iterativer Prozess Beachte: wir benutzen hier die applikative Reihenfolge der Auswertung length_lin length_linr length_linr length_linr length_linr .......... length_linr 0 1 2 3 (9:(8:(7:(6:...(1:[]))))) (9:(8:(7:(6:...(1:[]))))) (8:(7:(6:...(1:[])))) (7:(6:...(1:[]))) (6:...(1:[])) 9 Grundlagen der Programmierung 2 (2.A) SS 2011 [] - 25 - Allgemeine Funktionen auf Listen Allgemeine Funktionen (Methoden): foldl und foldr Links-Faltung und Rechts-Faltung Die 3 Argumente sind: • • • eine zweistellige Operation, ein Anfangselement (Einheitselement) und die Liste. foldl ⊗ e [a1, . . . , an] entspricht ((. . . ((e ⊗ a1) ⊗ a2) . . . ) ⊗ an). foldr ⊗ e [a1, . . . , an] entspricht a1 ⊗ (a2 ⊗ (. . . (an ⊗ e))) Grundlagen der Programmierung 2 (2.A) SS 2011 - 26 - Definitionen der fold-Funktionen foldl (Linksfaltung) foldr (Rechtsfaltung) foldl :: (a -> b -> a) -> a -> [b] -> a foldl f z [] = z foldl f z (x:xs) = foldl f (f z x) xs foldr :: (a -> b -> b) -> b -> [a] -> b foldr f z [] = z foldr f z (x:xs) = f x (foldr f z xs) Grundlagen der Programmierung 2 (2.A) SS 2011 - 27 - Fold-Verwendungen Summe bzw. Produkt einer Liste von Zahlen: sum xs = foldl (+) 0 xs produkt xs = foldl (*) 1 xs concat xs = foldr (++) [] xs foldl (+) 0 [1,2,3,4] foldr (++) [] [[0],[2,3],[5]] ≡ ≡ ((((0+1)+2)+3)+4) [0] ++ ([2,3] ++ ([5] ++ [])) Je nach Operator ist foldl, oder foldr besser geeignet. Grundlagen der Programmierung 2 (2.A) SS 2011 - 28 - Lokale Funktionsdefinitionen, anonyme Funktionen, Lambda-Ausdrücke Lambda-Ausdruck \x1 . . . xn -> hAusdrucki x1, x2, ... sind die formalen Parameter Beispiel quadrat = \x -> x*x Der Lambdaausdruck kann wie eine Funktion verwendet werden Grundlagen der Programmierung 2 (2.A) SS 2011 - 29 - let und lokale Bindungen let {x1 = s1; . . . ; xn = sn} in t {x1 = s1; . . . ; xn = sn} ist eine lokale Umgebung die Variablen xi können in t vorkommen mit der Bedeutung: Wert von si “ ” t der eigentliche Ausdruck let x1 = 5 x2 = "abc" x3 = 7*x1 in (x1,x2,x3) Grundlagen der Programmierung 2 (2.A) SS 2011 - 30 - let und lokale Bindungen In Haskell: rekursives let. D.h. xi kann in jedem sj vorkommen Beachte im ghci-Interpreter: Spezielle Verwendung des let Grundlagen der Programmierung 2 (2.A) SS 2011 - 31 - Erweiterungen des let Funktionen sind definierbar direkt in einem rekursiven let: let {f x1 . . . xn = s; . . .} in t Zum Beispiel: let hochdrei x = x*x*x, a = 3 in hochdrei a Grundlagen der Programmierung 2 (2.A) SS 2011 - 32 - Freie und Gebundene Variablen, Gültigkeitsbereiche Um Definitionen von lokalen Namen korrekt zu handhaben braucht man neue Begriffe: Gültigkeitsbereich einer Variablen x freie Variablen eines Ausdrucks gebundene Variablen eines Ausdrucks Text-Fragment(e) des Programms in dem dieses x gemeint ist. Variablen, deren Bedeutung außerhalb des Ausdrucks festgelegt wird. Variablen, deren Bedeutung innerhalb des Ausdrucks festgelegt wird. Problem: Variablen können mit gleichem Namen, aber verschiedener Bedeutung in einem Ausdruck vorkommen: Lösung: • • Exakte Festlegung der Gültigkeitsbereiche für jedes syntaktische Konstrukt Umbenennen von gebundenen Variablennamen, falls nötig Grundlagen der Programmierung 2 (2.A) SS 2011 - 33 - Beispiel \x-> x*x Gültigkeitsbereich von x: der Ausdruck x*x die Variable x ist gebunden von \x x*x in diesem Ausdruck ist x frei Grundlagen der Programmierung 2 (2.A) SS 2011 - 34 - Definition von FV FV: ergibt Menge von Variablen-Namen. • • • • • • F V (x) := {x} , wenn x ein Variablenname ist F V ((s t)) := F V (s) ∪ F V (t) F V (if t1 then t2 else t3) := F V (t1) ∪ F V (t2) ∪ F V (t3) F V (\x1 . . . xn -> t) := F V (t) \ {x1, . . . , xn} F V (let x1 = s1, . . . , xn = sn in t) := (F V (t) ∪ F V (s1) ∪ . . . ∪ F V (sn)) \ {x1, . . . , xn} F V (let f x1 . . . xn = s in t) := F V (let f = \x1 . . . xn -> s in t) Beachte: FV ist eine Funktion auf dem Syntaxbaum keine Haskell-Funktion Grundlagen der Programmierung 2 (2.A) SS 2011 - 35 - Beispiel: freie Variablen F V (\x -> (f x y)) = = = = Grundlagen der Programmierung 2 (2.A) SS 2011 F V (f x y ) \ {x} ... {x, f, y} \ {x} {f, y} - 36 - Gebundene Variablen GV (t) Entsprechend der F V -Definition: • • • • • • GV (x) := ∅ GV ((s t)) := GV (s) ∪ GV (t) GV (if t1 then t2 else t3) := GV (t1) ∪ GV (t2) ∪ GV (t3) GV (\x1 . . . xn -> t) := GV (t) ∪ {x1, . . . , xn} GV (let x1 = s1, . . . , xn = sn in t) := (GV (t) ∪ GV (s1) ∪ . . . ∪ GV (sn) ∪ {x1, . . . , xn}}) GV (let f x1 . . . xn = s in t) := GV (let f = \x1 . . . xn -> s in t) = {f, x1, . . . , xn} ∪ GV (s) ∪ GV (t) Grundlagen der Programmierung 2 (2.A) SS 2011 - 37 - Beispiel : Berechnung von gebundenen Variablen GV(\x -> (f x y)) = = = = Grundlagen der Programmierung 2 (2.A) SS 2011 GV (f x y) ∪ {x} ... ∅ ∪ {x} {x} - 38 - Lexikalischer Gültigkeitsbereich einer Variablen • let x = s • \x y z -> t in t die Vorkommen der freien Variablen x in s, t werden gebunden. s, t ist der lexikalische Gültigkeitsbereich der Variablen x die freien Variablen x, y, z in t werden gebunden. t ist der lexikalische Gültigkeitsbereich von x, y, z. Grundlagen der Programmierung 2 (2.A) SS 2011 - 39 - Beispiel Ausdruck t = \x -> (x (\x -> x*x)) x ist in t gebunden, aber in zwei Bindungsbereichen: \x -> (x (\x -> x*x)) In (x (\x -> x*x)) kommt x frei und gebunden vor. Umbenennen des gebundenen x in y ergibt: (x (\y -> y*y)) Grundlagen der Programmierung 2 (2.A) SS 2011 - 40 - Beispiele Zwei Bindungsbereiche für x in einem let-Ausdruck: let x = 10 in (let x = 100 in (x+x)) + x Umbenennung ergibt: let x1 = 10 in (let x2 = 100 in (x2+x2)) + x1 Dieser Term wertet zu 210 aus. Grundlagen der Programmierung 2 (2.A) SS 2011 - 41 - Beispiele Der Ausdruck let x = (x*x) in (x+x) führt zu Nichtterminierung. Grundlagen der Programmierung 2 (2.A) SS 2011 - 42 - Beispiel: Reihenfolgenunabhängigkeit der Bindungen let y = 20*z x = 10+y z = 15 in x Wertet aus zu : 310. Grundlagen der Programmierung 2 (2.A) SS 2011 - 43 - Beispiel geschachtelte Bindungsbereiche let {x = 1;y = 7} in (let {y = 2; z = 4} in (let z = 5 in (x+y+z))) x = 1;y = 7 y = 2; z = 4 z = 5 x+y+z Grundlagen der Programmierung 2 (2.A) SS 2011 - 44 - Programm als let Ein Programm mit den Definitionen fi = ei | i = 1, . . . , n und dem auszuwertenden Ausdruck main kann als großes“ let betrachtet werden: ” let {f1 = e1; . . . ; fn = en} in main Grundlagen der Programmierung 2 (2.A) SS 2011 - 45 - Optimierung mittels let Vermeidung redundanter Auswertungen mit let f (x, y) := x(1 + xy)2 + y(1 − y) + (1 + xy)(1 − y) optimierbar durch Vermeidung von Doppelauswertungen: Der zugehörige Ausdruck ist: let a = 1 + x*y b = 1 - y in x*a*a + y*b + a*b Grundlagen der Programmierung 2 (2.A) SS 2011 - 46 - Funktionen als Ergebnis: Komposition Beispiel: Komposition von Funktionen: komp::(a -> b) -> (c -> a) -> c -> b komp f g x = f (g x) In Haskell ist komp vordefiniert und wird als .“ geschrieben: ” Beispielaufruf: *Main> suche_nullstelle (sin . quadrat) 1 4 0.00000001 ←1.772453852929175 (sin . quadrat) entspricht sin(x2) und quadrat . sin entspricht (sin(x))2. Grundlagen der Programmierung 2 (2.A) SS 2011 - 47 - Typ der Komposition Erklärung zum Typ von komp, wobei {a,b,c} Typvariablen sind Ausdruck: f ‘komp‘ g bzw. f . g (a->b) -> (c->a) -> c->b (τ1 -> τ2) (τ3 -> τ1) τ3 τ2 Typ von (.) Typ von f Typ von g Typ des Arguments x der Komposition f . g Typ des Resultats der Komposition f (g x) (f . g) :: (τ3 -> τ2) x :: τ3 Grundlagen der Programmierung 2 (2.A) SS 2011 g / τ1 f / τ2 - 48 - Weitere Daten: Zusammengesetzte Daten-Objekte Paar: Beispiele (1, 2) (1, "hallo") (1,(2,"hallo")) Grundlagen der Programmierung 2 (2.A) SS 2011 (x, y) (Paar von Zahl und Paar...) - 49 - n-Tupel von Objekten n-Tupel sind Verallgemeinerung von Paaren (n ≥ 2) (t1, . . . , tn) ist n-Tupel von t1, . . . , tn Beispiele (1,2,3,True) (1,(2,True),3) ("hallo",False) (fakultaet 100,\x-> x) Grundlagen der Programmierung 2 (2.A) SS 2011 - 50 - Zusammengesetzte Objekte: Datentypen Für Datentypen benötigt man: Datenkonstruktor(en) Datenselektor(en) Beispiel Paarkonstruktor Paarselektoren s, t −→ (s, t) fst, snd Eigenschaften: fst(s, t) = s snd(s, t) = t. und Grundlagen der Programmierung 2 (2.A) SS 2011 - 51 - Beispiel n-Tupel n-Tupelkonstruktor t1, . . . , tn −→ (t1, . . . , tn) Tupelselektoren n Selektoren: pro Stelle ein Selektor n-Tupel haben einen impliziten Konstruktor: (., . . . , .) | {z } n Grundlagen der Programmierung 2 (2.A) SS 2011 - 52 - Definition der Selektoren Muster (pattern) statt Selektoren. Muster sind syntaktisch dort erlaubt, wo formale Parameter (Variablen) neu eingeführt werden: • • • in Funktionsdefinitionen, in Lambda-Ausdrücken und in let-Ausdrücken. Beispiel-Definitionen von Selektoren mittels Muster fst (x,y) = x snd (x,y) = y selektiere_erstes_von_3 (x1,x2,x3) = x1 selektiere_zweites_von_3 (x1,x2,x3) = x2 selektiere_drittes_von_3 (x1,x2,x3) = x3 Grundlagen der Programmierung 2 (2.A) SS 2011 - 53 - Beispiel: Typen von Selektoren, Konstruktoren, Tupeln (1, 1) :: (Integer, Integer) (1, (2, True)) :: (Integer, (Integer, Bool)) (., . . . , .) :: a1 → a2 → . . . → an → (a1, a2, . . . , an) :: (a1, a2, a3) → a3 | {z } n selektiere_drittes_von_3 Grundlagen der Programmierung 2 (2.A) SS 2011 - 54 - Benutzerdefinierte Konstruktoren Definierbar in Haskell mittels data-Anweisung Beispiel data Punkt = Punktkonstruktor Double Double data Strecke = Streckenkonstruktor Punkt Punkt Punkt, Strecke: Punktkonstruktor Streckenkonstruktor Double, Punkt (rechts) Typen Konstruktoren Typen der Argumente Grundlagen der Programmierung 2 (2.A) SS 2011 - 55 - Muster (pattern) streckenAnfang (Streckenkonstruktor x y) = x Nutzen der Muster: • • tiefes Selektieren Ersatz für Selektoren Syntax der Muster: hMusteri ::= hVariablei | (hMusteri) | hKonstruktor(n)i hMusteri . . . hMusteri | {z n } | (hMusteri, . . . , hMusteri) Bedingung: in einem Muster darf keine Variable doppelt vorkommen Grundlagen der Programmierung 2 (2.A) SS 2011 - 56 - Auswertung unter Benutzung von Mustern Mustervergleich: Anpassen des Objekts an das Muster gleichzeitige Selektion mittels impliziter let-Bindungen I.a. vorher Auswertung des Objekts erforderlich Beispiele (x,y,(u,v)) anpassen an: (1,2,(3,4)) ergibt: let {x = 1;y = 2;u = 3;v = 4} in ... (x,y,(u,v)) anpassen an: (1,2,True) ergibt: Fehler. Kann nicht vorkommen wegen Typcheck. (x,y,u) anpassen an: (1,2,(4,5)) ergibt: let {x = 1; y = 2;u = (4,5)} Grundlagen der Programmierung 2 (2.A) SS 2011 in ... - 57 - Auswertung unter Benutzung von Mustern (2) Beispiele (x,y) anpassen an: (1,fakt 100) ergibt: {let {x = 1; y = fakt 100} in ... (x,y) anpassen an: (fst (1,2), snd (fakt 100,fakt 200)) ergibt : let {x = fst (1,2); y = snd (fakt 100, fakt 200)} in ... Grundlagen der Programmierung 2 (2.A) SS 2011 - 58 - Benutzerdefinierte Typnamen mit Parametern Beispiel Punkt, Strecke, Polygonzug data data data data Punkt a Strecke a Vektor a Polygon a = = = = Punkt a a Strecke (Punkt a) (Punkt a) Vektor a a Polygon [Punkt a] Typ und Konstruktor können gleiche Namen haben. Der Parameter a kann jeder Typ sein: z.B.: Float, Int, aber auch [[(Int, Char)]] Grundlagen der Programmierung 2 (2.A) SS 2011 - 59 - Funktionen auf Punkt, Strecke, Polygonzug addiereVektoren::Num a => Vektor a -> Vektor a -> Vektor a addiereVektoren (Vektor a1 a2) (Vektor b1 b2) = Vektor (a1 + b1) (a2 + b2) streckenLaenge (Strecke (Punkt a1 a2) (Punkt b1 b2)) = sqrt (fromInteger ( (quadrat (a1-b1)) + (quadrat (a2-b2)))) Grundlagen der Programmierung 2 (2.A) SS 2011 - 60 - Summentypen und Fallunterscheidung Summentypen: diese haben mehr als einen Konstruktor Beispiele: Bool mit True False data Bool = True | False Aufzählungstyp: data Farben = Rot | Gruen | Blau | Weiss | Schwarz data Kontostand = Euro Integer | Dollar | SFranken Integer Grundlagen der Programmierung 2 (2.A) SS 2011 Integer - 61 - Liste als Summentyp selbstdefinierte Listen: Typvariable data Liste a = Nil | Cons a (Liste a) Typkonstruktor Typ erstes Arg Konstruktor1 Typ zweites Arg Konstruktor2 Listen-Definition in Haskell: data [a] = [] | a : [a] Grundlagen der Programmierung 2 (2.A) SS 2011 - 62 - Fallunterscheidung mit case Syntax: case hAusdrucki of {h Musteri -> hAusdrucki; . . . ;hMusteri -> hAusdrucki} Einschränkung: Kontextbedingung: nur einfache Muster: K x1 . . . xn die Muster müssen vom Typ her passen. Beispiel: und4 x y = case x of True -> y; False -> False Grundlagen der Programmierung 2 (2.A) SS 2011 - 63 - case: Gültigkeitsbereich, FV und GV F V (case s of (c1 x11 . . . x1n1 → t1); . . . ; (ck xk1 . . . xknk → tk )) = F V (s) ∪ F V (t1) \ {x11, . . . x1n1 } . . . ∪ F V (tk ) \ {xk1, . . . xknk } GV(.) entsprechend Die Muster binden die Variablen! Grundlagen der Programmierung 2 (2.A) SS 2011 - 64 - case: Gültigkeitsbereich und Beispiel F V (case x of True -> y; False -> False) = {x,y} F V (case x of (Punkt u v) -> u) = {x} GV (case x of (Punkt u v) -> u) = {u,v} Grundlagen der Programmierung 2 (2.A) SS 2011 - 65 - Reduktionsregel zum case case-Reduktion (case (c t1 . . . tn) of . . . (c x1 . . . xn → s) . . .) s[t1/x1, . . . , tn/xn] Alternative Darstellung mit let für verzögerte Reduktion: (case (c t1 . . . tn) of . . . (c x1 . . . xn → s) . . .) let {x1 = t1; . . . ; xn = tn} in s Grundlagen der Programmierung 2 (2.A) SS 2011 - 66 - Bemerkungen zur Auswertung • Auswertung ist erweitert um neue Reduktion: case-Reduktion • Normale Reihenfolge der Auswertung reduziert nicht die Argumente von Konstruktoren Dadurch kann man (potentiell) unendliche Listen verarbeiten Grundlagen der Programmierung 2 (2.A) SS 2011 - 67 - Auswertung in Haskell Kombination von Transformation und Auswertung: Haskell-Programm Entzuckerung Programm in Kernsprache Syntaxanalyse / Kompilieren Syntaxbaum Auswertung (operationelle Semantik) transformierter Syntaxbaum A Auswertung Grundlagen der Programmierung 2 (2.A) SS 2011 - 68 - Entzuckerung: Beispiel map f [] map f (x:xs) = [] = f x : map f xs kann man auch programmieren mittels: map f lst = (case lst of [] -> []; (x:xs) -> f x : map f xs) Grundlagen der Programmierung 2 (2.A) SS 2011 - 69 -