-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy path05-data.scrbl
425 lines (339 loc) · 17.7 KB
/
05-data.scrbl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
#lang scribble/manual
@(require scribble/eval)
@(require "marburg-utils.rkt")
@(require (for-label lang/htdp-beginner))
@(require (for-label (except-in 2htdp/image image?)))
@(require (for-label 2htdp/universe))
@title[#:version "" #:tag "summentypen"]{Datendefinition durch Alternativen: Summentypen}
Die Datentypen, die wir bisher kennengelernt und genutzt haben, umfassen Zahlen, Strings, Wahrheitswerte
sowie in informellen Datendefinitionen definierte Datentypen wie @racket[Temperatur].
Um realistische Anwendungen zu programmieren, ist es hilfreich, ein größeres und flexibel erweiterbares
Repertoire an Datentypen zu haben. In diesem Kapitel schauen wir uns an,
wie man mit Datentypen unterschiedliche Situationen unterscheiden kann und wie diese Datentypen
den Entwurf von Programmen beeinflussen.
@section{Aufzählungstypen}
Häufig hat man den Fall, dass ein Datentyp nur eine endliche aufzählbare Menge von Werten enthält.
Wir nennen solche Datentypen @italic{Aufzählungstypen} oder @italic{Enumerationstypen}.
Hier ist ein Beispiel, wie ein Aufzählungstyp definiert wird:
@#reader scribble/comment-reader
(racketblock
; A TrafficLight is one of:
; – "red"
; – "green"
; – "yellow"
; interp. each element of TrafficLight represents which colored
; bulb is currently turned on
)
Das Programmieren mit Aufzählungstypen ist sehr einfach. Wenn eine Funktion einen Parameter hat, der einen Aufzählungstyp
hat, dann enthält das Programm typischerweise eine Fallunterscheidung, in denen alle Alternativen des Aufzählungstyps
separat voneinander behandelt werden. Hier ein Beispiel:
@#reader scribble/comment-reader
(racketblock
; TrafficLight -> TrafficLight
; given state s, determine the next state of the traffic light
(check-expect (traffic-light-next "red") "green")
(define (traffic-light-next s)
(cond
[(string=? "red" s) "green"]
[(string=? "green" s) "yellow"]
[(string=? "yellow" s) "red"]))
)
Manchmal ist eine Funktion auch nur an einer Teilmenge der möglichen Werte eines Aufzählungstypen interessiert. In diesem
Fall werden nur die interessanten Fälle abgefragt und der Rest ignoriert. Ein Beispiel dafür ist die
@racket[on-mouse-event] Funktion aus @secref{universeteachpack}, die eine Eingabe vom Aufzählungstypen @racket[MouseEvent]
erhält. @racket[MouseEvent] ist folgendermaßen definiert:
@#reader scribble/comment-reader
(racketblock
; A MouseEvt is one of:
; – "button-down"
; – "button-up"
; – "drag"
; – "move"
; – "enter"
; – "leave"
; interp. the kind of mouse event that has been triggered
)
In der Funktion @racket[on-mouse-event] interessieren wir uns nur für die Alternative @racket["button-down"].
Für alle anderen Alternativen von @racket[MouseEvent], lassen wir den Zustand unverändert.
Es kann auch sein, dass ein Funktionsparameter einen Aufzählungstyp hat, aber die Funktion trotzdem keine Fallunterscheidung vornimmt, weil
hierzu eine Hilfsfunktion aufgerufen wird. Dennoch ist es eine gute Heuristik, im Schritt 3 des Entwurfsrezepts
aus Abschnitt @secref{entwurfsrezept} mit einem Funktionstemplate zu starten, welches alle Fälle des Parameters unterscheidet.
Beispielsweise würde das Template für die @racket[traffic-light-next] Funktion von oben so aussehen:
@racketblock[
(define (traffic-light-next s)
(cond
[(string=? "red" s) ...]
[(string=? "green" s) ...]
[(string=? "yellow" s) ...]))]
Wie sie sehen, können sie einen großen Teil einer Funktionsdefinition quasi mechanisch generieren, indem Sie systematisch den
Schritten aus dem Entwurfsrezept folgen.
@section{Aufzählungstypen mit überprüfbaren Signaturen}
Die Sprache für überprüfbare Signaturen in BSL erlaubt es uns, Aufzählungstypen zu definieren und ihnen einen Namen zu geben.
Mit Hilfe von @racket[enum] kann ein Aufzählungstyp definiert und benutzt werden:
@#reader scribble/comment-reader
(racketblock
(: traffic-light-next ((enum "red" "green" "yellow") -> (enum "red" "green" "yellow")))
)
Allerdings ist es, wie so oft in der Programmierung, sinnvoll, solchen wiederkehrenden Konzepten einen Namen geben. Mit
Hilfe der @racket[signature] Form kann ein Name definiert werden, der im folgenden dann für die angegebene Signatur steht.
@#reader scribble/comment-reader
(racketblock
(define TrafficLight (signature (enum "red" "green" "yellow")))
; interp. each element of TrafficLight represents which colored
; bulb is currently turned on
)
Mit Hilfe dieser Definition kann die @racket[traffic-light-next] Funktion dann wie folgt definiert werden.
@#reader scribble/comment-reader
(racketblock
(: traffic-light-next (TrafficLight -> TrafficLight))
; given state s, determine the next state of the traffic light
(check-expect (traffic-light-next "red") "green")
(define (traffic-light-next s)
(cond
[(string=? "red" s) "green"]
[(string=? "green" s) "yellow"]
[(string=? "yellow" s) "red"]))
)
@section{Intervalltypen}
Betrachten Sie ein Programm, welches ein Ufo beim landen zeigen soll.
Ein Programm hierzu könnte wie folgt aussehen:
@#reader scribble/comment-reader
(racketblock
; constants:
(define WIDTH 300)
(define HEIGHT 100)
; visual constants:
(define MT (empty-scene WIDTH HEIGHT))
(define UFO
(overlay (circle 10 "solid" "green")
(rectangle 40 2 "solid" "green")))
(define BOTTOM (- HEIGHT (/ (image-height UFO) 2)))
; A WorldState is a number.
; interp. height of UFO (from top)
; WorldState -> WorldState
; compute next location of UFO
(define (nxt y)
(+ y 1))
; WorldState -> Image
; place UFO at given height into the center of MT
(define (render y)
(place-image UFO (/ WIDTH 2) y MT))
; WorldState -> Boolean
; returns #true when UFO has reached the bottom, i.e. y > BOTTOM
(define (end-of-the-world y) (> y BOTTOM))
; wire everything together; start descend at position 0
(big-bang 0 (on-tick nxt) (to-draw render) (stop-when end-of-the-world))
)
Betrachten Sie nun folgende Erweiterung der Problemstellung:
@italic{The status line should say "descending" when the UFO’s height is above one third of the height of the canvas.
It should switch to "closing in" below that. And finally, when the UFO has reached the bottom of the canvas,
the status should notify the player that the UFO has "landed."}
Der Datentyp, der sich in dieser Problemstellung versteckt, ist kein Enumerationstyp, denn es gibt eine unendliche
Zahl unterschiedlicher Höhen (oder, sofern wir nur die ganzen Zahlen zählen, so viele unterschiedliche Höhen, dass
es unpraktisch wäre, hierfür einen Aufzählungstypen zu definieren).
Daher verwenden wir @italic{Intervalle}, um zusätzliche Struktur auf geordneten Datentypen wie Zahlen oder Strings zu definieren.
Im folgenden konzentrieren wir uns auf (ganzzahlige oder nicht-ganzzahlige) Zahlen. Ein Intervall wird durch seine @italic{Grenzen}
definiert. Ein Intervall hat entweder eine untere und obere Grenze, oder es hat nur eine dieser Grenzen und ist zur anderen Seite offen.
In unserem Beispiel können wir die Datendefinition für den WorldState als Intervall ausdrücken, um eine Datendefinition zu
haben, die die Intention der Problemstellung besser erfasst.
@#reader scribble/comment-reader
(racketblock
; constants:
(define CLOSE (* 2 (/ HEIGHT 3)))
; A WorldState is one of:
; – Number between 0 and CLOSE
; – Number between CLOSE and BOTTOM
; – BOTTOM
; interp. height of UFO (from top)
)
Nicht alle Funktionen, die einen Intervalltypen als Argument bekommen, nutzen diese zusätzliche Struktur.
So kann zum Beispiel die @racket[render] Funktion von oben so bleiben wie sie ist, weil das Intervall nur für die
Statusanzeige aber nicht für das Ufo relevant ist.
Funktionen, die jedoch die zusätzliche Struktur des Intervalls benötigen, enthalten typischerweise einen
@racket[cond] Ausdruck, der die unterschiedlichen Intervalle unterscheidet.
@#reader scribble/comment-reader
(racketblock
; WorldState -> Image
; add a status line to the scene created by render
(define (render/status y)
(cond
[(<= 0 y CLOSE) (above (text "descending" 12 "black") (render y))]
[(< CLOSE y BOTTOM) (above (text "closing in" 12 "black") (render y))]
[(= y BOTTOM) (above (text "landed" 12 "black") (render y))]))
)
Es empfiehlt sich, diesen @racket[cond] Ausdruck im Rahmen der Templatekonstruktion aus dem Entwurfsrezept direkt in das Template mit
aufzunehmen. Allerdings findet die Fallunterscheidung nicht notwendigerweise als erstes statt sondern kann auch tiefer
im Funktionsbody stattfinden. Dies bietet sich in unserem Beispiel an, denn wir haben gegen das DRY Prinzip verstoßen (Wenn wir
die Farbe des Textes beispielsweise auf "red" ändern möchten, müssten wir drei Zeilen ändern). Deshalb ist es vorteilhaft,
den konditionalen Ausdruck nach innen zu ziehen:
@#reader scribble/comment-reader
(racketblock
; WorldState -> Image
; add a status line to the scene created by render
(define (render/status y)
(above
(text
(cond
[(<= 0 y CLOSE) "descending"]
[(< CLOSE y BOTTOM) "closing in"]
[(= y BOTTOM) "landed"])
12 "black")
(render y)))
)
Nun muss nur noch der @racket[big-bang] Ausdruck angepasst werden, so dass @racket[render/status] und nicht mehr @racket[render] zum Zeichnen verwendet wird.
@#reader scribble/comment-reader
(racketblock
; wire everything together; start descend at position 0
(big-bang 0 (on-tick nxt) (to-draw render/status) (stop-when end-of-the-world))
)
Intervalle liefern uns außer neuen Funktionstemplates auch Anhalte zum Testen: Typischerweise möchte man einen Testcase für jedes Intervall
und insbesondere für die Intervallgrenzen haben.
@section[#:tag "sums"]{Summentypen}
Intervalltypen unterscheiden unterschiedliche Teilmengen der Zahlen (oder anderer geordneter Werte).
Aufzählungstypen zählen die unterschiedlichen Elemente des Typs Wert für Wert auf.
@italic{Summentypen} verallgemeinern Intervalltypen und Aufzählungstypen.
Mit Summentypen können existierende Datentypen und individuelle
Werte beliebig miteinander kombiniert werden. Ein Summentyp gibt verschiedene
Alternativen an, von denen jeder Wert dieses Typs genau eine Alternative erfüllt.
Betrachten wir als Beispiel die @racket[string->number] Funktion, die einen
String in eine Zahl konvertiert, falls dies möglich ist, und andernfalls @racket[#false]
zurückgibt.
Hier ist die Definition eines Summentyps, der das Verhalten dieser Funktion beschreibt:
@#reader scribble/comment-reader
(racketblock
; A MaybeNumber is one of:
; – #false
; – Number
; interp. a number if successful, else false.
)
Damit können wir für @racket[string->number] folgende Signatur definieren:
@#reader scribble/comment-reader
(racketblock
; String -> MaybeNumber
; converts the given string into a number;
; produces #false if impossible
(define (string->number str) ...)
)
Summentypen werden manchmal auch Vereinigungstypen (union types) genannt.
In HTDP/2e werden sie @italic{Itemizations} genannt.
In der Dokumentation der HTDP-Sprachen ist die Signatur von
@racket[string->number] so angegeben:
@racketblock[String -> (union Number #false)]
Der Operator @racket[union] steht für die ``on-the-fly'' Konstruktion eines
anonymen Summentyps mit der gleichen Bedeutung wie unser @racket[MaybeNumber] oben.
Was macht man mit einem Wert, der einen Summentyp hat? Wie bei Aufzählungs- und Intervalltypen
auch ist typischerweise die einzige sinnvolle Operation die, welche die unterschiedlichen
Alternativen voneinander trennt, beispielsweise im Rahmen eines @racket[cond] Ausdrucks.
Hier ein Beispiel:
@#reader scribble/comment-reader
(racketblock
; MaybeNumber -> MaybeNumber
; adds 3 to a if it is a number; returns #false otherwise
(check-expect (add3 5) 8)
(check-expect (add3 #false) #false)
(define (add3 a)
(cond [(number? a) (+ a 3)]
[else #false]))
)
Funktionen mit Summentypen unterscheiden sich in ihrer Entwurfsmethodik etwas von den Funktionen, die
wir bisher kennengelernt haben. Deswegen gehen wir nochmal durch unser Entwurfsrezept (@secref{entwurfsrezept}) und beschreiben,
wie sich der Entwurf von Funktionen mit Summentypen vom allgemeinen Entwurfsrezept unterscheidet.
@italic{The tax on an item is either an absolute tax of 5 or 10 currency units, or a linear tax. Design a function that computes the price
of an item after applying its tax.}
@subsection[#:tag "entwurfsrezept-summen"]{Entwurf mit Summentypen}
Das Entwurfsrezept für den Entwurf von Funktionen ergänzen wir wie folgt:
@itemize[#:style 'ordered
@item{Falls die Problemstellung Werte in unterschiedliche Klassen unterteilt,
so sollten diese Klassen durch eine Datendefinition explizit definiert werden.
Beispiel: Die Problemstellung oben unterscheidet drei verschiedene Arten
der Steuer. Dies motiviert folgende Datendefinition:
@#reader scribble/comment-reader
(racketblock
; A Tax is one of
; - "absolute5"
; - "absolute10"
; - a Number representing a linear tax rate in percent
) }
@item{Im zweiten Schritt ändert sich nichts, außer dass typischerweise der
definierte Datentyp in der Signatur verwendet wird.
Beispiel:
@#reader scribble/comment-reader
(racketblock
; Number Tax -> Number
; computes price of an item after applying tax
(define (total-price itemprice tax) 0)
)
}
@item{Bzgl. der Tests ist es ratsam, pro Alternative des Datentypen mindestens
ein Beispiel zu haben. Bei Intervallen sollten die Grenzen der Intervalle
getestet werden. Gibt es mehrere Parameter mit Summentypen, so sollten
alle Kombinationen der Alternativen getestet werden.
Beispiel:
@#reader scribble/comment-reader
(racketblock
(check-expect (total-price 10 "absolute5") 15)
(check-expect (total-price 10 "absolute10") 20)
(check-expect (total-price 10 25) 12.50)
)
}
@item{Die größte Neuerung ist die des Templates für die Funktion. Im Allgemeinen
@italic{folgt die Struktur der Funktionen aus der Struktur der Daten}.
Dies bedeutet, dass in den meisten Fällen der Funktionskörper mit
einem @racket[cond] Ausdruck startet, der die unterschiedlichen Fälle
des Summentypen unterscheidet.
@margin-note{Wieso ist in diesem Beispiel die Reihenfolge der Zweige des @racket[cond]
Ausdrucks wichtig?}
@#reader scribble/comment-reader
(racketblock
(define (total-price itemprice tax)
(cond [(number? tax) ...]
[(string=? tax "absolute5") ...]
[(string=? tax "absolute10") ...]))
)}
@item{@margin-note{Diese Unterscheidung der Fälle ist ein Beispiel
für ein allgemeineres Entwurfskonzept für komplexe Systeme,
welches sich @italic{separation of concerns} (Trennung der Belange) nennt.}
Im fünften Schritt werden die @racket[...] Platzhalter durch den korrekten Code
ersetzt. Hier ist es sinnvoll, jeden Zweig des @racket[cond] Ausdrucks einzeln
und nacheinander durchzugehen und zu implementieren. Der Vorteil ist, dass Sie
sich bei dieser Art der Implementierung immer nur um einen der Fälle kümmern müssen
und alle anderen Fälle ignorieren können.
Möglicherweise stellen Sie während dieses Schritts fest, dass es sinnvoll ist,
den @racket[cond] Ausdruck nach innen zu ziehen (ähnlich wie
in @racket[create-rocket-scene-v6] in Abschnitt @secref{dryredux}), oder
vielleicht müssen Sie gar nicht alle Fälle unterscheiden, oder vielleicht
möchten Sie die Unterscheidung auch in eine Hilfsfunktion auslagern --
dennoch ist es in der Regel sinnvoll, mit diesem Template zu starten, selbst
wenn am Ende ihre Funktion anders aussieht.
Beispiel:
@#reader scribble/comment-reader
(racketblock
(define (total-price itemprice tax)
(cond [(number? tax) (* itemprice (+ 1 (/ tax 100)))]
[(string=? tax "absolute5") (+ itemprice 5)]
[(string=? tax "absolute10") (+ itemprice 10)]))
)
}
@item{Im letzten Schritt ändert sich nichts, aber überprüfen Sie,
dass Sie Testfälle für alle Alternativen definiert haben.}
]
@section[#:tag "disjoint"]{Unterscheidbarkeit der Alternativen}
Was wäre, wenn wir in unserem letzten Beispiel statt zwei absoluten Steuersätzen ein kontinuierliches
Spektrum hätten? Wir könnten unsere Datendefinition wie folgt abändern:
@#reader scribble/comment-reader
(racketblock
; A Tax is one of
; - a Number representing an absolute tax in currency units
; - a Number representing a linear tax rate in percent
)
So weit so gut -- aber wie können wir in einem @racket[cond] Ausdruck diese Fälle unterscheiden? Wir haben
zwar das Prädikat @racket[number?], aber damit können wir nicht zwischen diesen beiden Fällen unterscheiden.
In unseren Summentypen ist es wichtig, dass man eindeutig unterscheiden kann, welche Alternative in einem Wert, der
einen Summentyp hat, gewählt wurde. Daher müssen die Mengen der möglichen Werte für jede Alternative disjunkt sein.
@margin-note{Die Variante der Summentypen, die wir verwenden, bezeichnet man deshalb auch als @italic{untagged unions}.}
Falls sie nicht disjunkt sind, muss man zu jedem Wert eine zusätzliche Information abspeichern, die aussagt, zu welcher
Alternative dieser Wert gehört: ein sogenanntes @italic{tag} ("Etikett", "Anhänger").
Mit den Mitteln, die wir bisher kennengelernt haben,
können wir den oben gewünschten Datentyp nicht sinnvoll ausdrücken. Im nächsten Kapitel werden wir jedoch
ein Sprachkonstrukt kennenlernen, mit dem man solche @italic{tags} und damit auch solche Datentypen strukturiert
definieren kann.