-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy path09-recdatastructures.scrbl
989 lines (757 loc) · 47.8 KB
/
09-recdatastructures.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
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
#lang scribble/manual
@(require scribble/eval)
@(require scribble/core)
@(require "marburg-utils.rkt")
@(require (for-label lang/htdp-beginner))
@(require (for-label (except-in 2htdp/image image?)))
@(require (for-label 2htdp/universe))
@(require scriblib/footnote)
@(require scribble-math)
@; (require "math-utilities.rkt")
@;setup-math
@title[#:version ""]{Daten beliebiger Größe}
Die Datentypen, die wir bisher gesehen haben, repräsentieren grundsätzlich
Daten, die aus einer festen Anzahl von atomaren Daten bestehen. Dies
liegt daran, dass jede Datendefinition nur andere Datendefinitionen verwenden
darf, die vorher bereits definiert wurden. Betrachten wir zum Beispiel
@#reader scribble/comment-reader
(racketblock
(define-struct gcircle (center radius))
; A GCircle is (make-gcircle Posn Number)
; interp. the geometrical representation of a circle
)
so wissen wir, dass eine @racket[Posn] aus zwei und eine @racket[Number]
aus einem atomaren Datum (jeweils Zahlen) besteht und damit
ein @racket[GCircle] aus genau drei atomaren Daten.
In vielen Situationen wissen wir jedoch nicht zum Zeitpunkt des Programmierens, aus
wievielen atomaren Daten ein zusammengesetztes Datum besteht. In diesem Kapitel befassen
wir uns damit, wie wir Daten beliebiger (und zum Zeitpunkt des Programmierens unbekannter)
Größe repräsentieren und Funktionen, die solche Daten verarbeiten, programmieren können.
@section[#:tag "rekursivedatentypen"]{Rekursive Datentypen}
Betrachten wir als Beispiel ein Programm, mit dem Stammbäume von Personen verwaltet werden können.
Jede Person im Stammbaum hat einen Vater und eine Mutter; manchmal sind Vater oder Mutter einer
Person auch unbekannt.
Mit den bisherigen Mitteln könnten wir zum Beispiel so vorgehen, um die Vorfahren einer Person zu repräsentieren:
@#reader scribble/comment-reader
(racketblock
(define-struct parent (name grandfather grandmother))
; A Parent is: (make-parent String String String)
; interp. the name of a person with the names of his/her grandparents
(define-struct person (name father mother))
; A Person is: (make-person String Parent Parent)
; interp. the name of a person with his/her parents
)
Allerdings können wir so nur die Vorfahren bis zu genau den Großeltern repräsentieren. Natürlich könnten wir noch
weitere Definitionen für die Urgroßeltern usw. hinzufügen, aber stets hätten wir eine beim Programmieren festgelegte
Größe. Außerdem verstoßen wir gegen das DRY Prinzip, denn die Datendefinitionen sehen für jede Vorgängergeneration
sehr ähnlich aus.
Was können wir machen um mit diesem Problem umzugehen? Betrachten wir etwas genauer, was für Daten wir hier eigentlich
modellieren wollen. Die Erkenntnis, die wir hier benötigen, ist, dass ein Stammbaum eine rekursive Struktur hat:
Der Stammbaum einer Person besteht aus dem Namen der Person und dem Stammbaum seiner Mutter und dem Stammbaum seiner Eltern.
Ein Stammbaum hat also eine Eigenschaft, die man auch @italic{Selbstähnlichkeit} nennt: Ein Teil eines Ganzen hat die
gleiche Struktur wie das Ganze.
Dies bedeutet, dass wir die Restriktion fallen lassen müssen, dass in einer Datendefinition nur die Namen bereits vorher
definierter Datentypen auftauchen dürfen. In unserem Beispiel schreiben wir:
@#reader scribble/comment-reader
(racketblock
(define-struct person (name father mother))
; A FamilyTree is: (make-person String FamilyTree FamilyTree)
; interp. the name of a person and the tree of his/her parents.
)
Die Definition von @racket[FamilyTree] benutzt also selber wieder @racket[FamilyTree].
Erstmal ist nicht ganz klar, was das bedeuten soll. Vor allen Dingen ist aber auch nicht
klar, wie wir einen Stammbaum erzeugen sollen. Wenn wir versuchen, einen zu erzeugen,
bekommen wir ein Problem:
@racket[(make-person "Bob" (make-person "Horst" (make-person "Joe" ...)))]
Wir können überhaupt gar keinen Stammbaum erzeugen, weil wir zur Erzeugung bereits einen
Stammbaum haben müssen. Ein Ausdruck, der einen Stammbaum erzeugt, wäre also unendlich groß.
Bei genauer Überlegung sind Stammbäume aber niemals unendlich groß, sondern sie enden bei
irgendeiner Generation - zum Beispiel weil die Vorfahren unbekannt oder nicht von Interesse sind.
Diesem Tatbestand können wir dadurch Rechnung tragen, dass wir aus @racket[FamilyTree] einen
Summentyp machen, und zwar wie folgt:
@#reader scribble/comment-reader
(racketblock
(define-struct person (name father mother))
; A FamilyTree is either:
;- (make-person String FamilyTree FamilyTree)
;- #false
; interp. either the name of a person and the tree of its parents,
; or #false if the person is not known/relevant.
)
Dieser neue, nicht-rekursive Basisfall erlaubt es uns nun, Werte dieses Typen zu erzeugen.
Hier ist ein Beispiel:
@racketblock[
(define Bob
(make-person "Bob"
(make-person "Alice" #false #false)
(make-person "Horst"
(make-person "Joe"
#false
(make-person "Rita"
#false
#false))
#false)))]
Was bedeutet also so eine rekursive Datendefinition? Wir haben bisher Datentypen immer
als Mengen von Werten aus dem Datenuniversum interpretiert, und dies ist auch
weiterhin möglich, nur die Konstruktion der Menge, die der Typ repräsentiert, ist nun
etwas aufwendiger:
Sei @${ft_0} die leere Menge, @${ft_1} die Menge @${\{ \mathtt{\# false} \}}, @${ft_2} die Vereinigung
aus @${\{ \mathtt{\# false} \}} und der Menge der @racket[(make-person name #false #false)] für alle
Strings @racket[name]. Im Allgemeinen sei ft@subscript{i+1} die Vereinigung aus @${\{ \mathtt{\# false} \}} und der
Menge der @racket[(make-person name p1 p2)] für alle Strings @racket[name] sowie für alle @racket[p1] und
alle @racket[p2] aus ft@subscript{i}. Beispielsweise ist @racket[Bob] ein Element von ft@subscript{5} (und
damit auch ft@subscript{6}, ft@subscript{7} usw.) aber nicht von ft@subscript{4}.
Dann ist die Bedeutung von @racket[FamilyTree], @${\mathit{ft}}, die Vereinigung aller dieser Mengen, also ft@subscript{0}
vereinigt mit @${ft_1} vereinigt mit @${ft_2} vereinigt mit @${ft_3} vereinigt mit ... .
In mathematischer Schreibweise können wir die Konstruktion so zusammenfassen:
@${
\begin{aligned}
\mathit{ft}_0 & = \emptyset \\
\mathit{ft}_{i+1} & = \{ \mathtt{\# false} \} \cup \{ \mathtt{(make-person}\ \mathit{n}\ \mathit{p}_1\ \mathit{p}_2 \mathtt{)} \ |
\mathit{n} \in \mathit{String}, \mathit{p}_1 \in \mathit{ft}_{i}, \mathit{p}_2 \in \mathit{ft}_{i} \} \\
\mathit{ft} & = \bigcup_{i \in \mathbb{N}}{\mathit{ft}_i}
\end{aligned}
}
Es ist nicht schwer zu sehen, dass stets @${\mathit{ft}_i \subseteq \mathit{ft}_{i+1}}; die nächste Menge umfasst
also stets die vorherige.
Aus dieser Mengenkonstruktion wird klar, wieso rekursive Datentypen es ermöglichen, Daten beliebiger
Größe zu repräsentieren: Jede Menge ft@subscript{i} enthält die Werte, deren maximale Tiefe in Baumdarstellung i ist.
Da wir alle ft@subscript{i} miteinander vereinigen, ist die Tiefe (und damit auch die Größe) unbegrenzt.
Diese Mengenkonstruktion kann für jeden rekursiven Datentyp definiert werden. Falls es mehrere rekursive
Alternativen gibt, so ist die i-te Menge die Vereinigung der i-ten Mengen für jede Alternative. Gäbe es also
beispielsweise noch eine zusätzliche @racket[FamilyTree] Alternative @racket[(make-celebrity String Number FamilyTree FamilyTree)]
bei der neben dem Namen auch noch das Vermögen angegeben wird, so wäre
@$${
\begin{aligned}
\mathit{ft}_{i+1} & = \{ \mathtt{\# false} \} \cup \{ \mathtt{(make-person}\ \mathit{n}\ \mathit{p}_1\ \mathit{p}_2 \mathtt{)} \ |
\mathit{n} \in \mathit{String}, \mathit{p}_1 \in \mathit{ft}_{i}, \mathit{p}_2 \in \mathit{ft}_{i} \} \\
& \cup \{ \mathtt{(make-celebrity}\ \mathit{n}\ \mathit{w}\ \mathit{p}_1\ \mathit{p}_2 \mathtt{)} \ |
\mathit{n} \in \mathit{String}, \mathit{w} \in \mathit{Number}, \mathit{p}_1 \in \mathit{ft}_{i}, \mathit{p}_2 \in \mathit{ft}_{i} \}
\end{aligned}
}
@section[#:tag "programmieren-rekdt"]{Programmieren mit rekursiven Datentypen}
Im letzten Abschnitt haben wir gesehen, wie man rekursive Datentypen definiert, was sie bedeuten, und wie man Instanzen
dieser Datentypen erzeugt. Nun wollen wir überlegen, wie man Funktionen programmieren kann, die Instanzen
solcher Typen als Argumente haben oder als Resultat produzieren.
Betrachten Sie dazu folgende Aufgabe: Programmieren sie eine Funktion, die herausfindet, ob es im Stammbaum
einer Person einen Vorfahren mit einem bestimmten Namen gibt.
Eine Signatur, Aufgabenbeschreibung und Tests sind dazu schnell definiert:
@#reader scribble/comment-reader
(racketblock
; FamilyTree String -> Boolean
; determines whether person p has an ancestor a
(check-expect (person-has-ancestor Bob "Joe") #true)
(check-expect (person-has-ancestor Bob "Emil") #false)
(define (person-has-ancestor p a) ...)
)
Da @racket[FamilyTree] ein Summentyp ist, sagt unser Entwurfsrezept, dass wir eine Fallunterscheidung machen.
In jedem Zweig der Fallunterscheidung können wir die Selektoren für die Felder eintragen. In
unserem Beispiel ergibt sich:
@#reader scribble/comment-reader
(racketblock
(define (person-has-ancestor p a)
(cond [(person? p)
...(person-name p) ...
...(person-father p)...
...(person-mother p) ...]
[else ...]))
)
@margin-note{Wir interpretieren das Wort "Vorfahr" so, dass eine Person ihr eigener Vorfahr ist. Wie
müsste das Programm aussehen, um die nicht-reflexive Interpretation des Worts "Vorfahr"
zu realisieren?}
Die Ausdrücke @racket[(person-father a)] und @racket[(person-mother a)] stehen für Werte, die einen komplexen
Summentypen, nämlich wieder @racket[FamilyTree] haben. Offensichtlich hat eine Person einen Vorfahren
@racket[a], wenn die Person entweder selber @racket[a] ist oder die Mutter oder der Vater einen
Vorfahr mit dem Namen @racket[a] hat.
Unser Entwurfsrezept schlägt für Fälle, in denen Felder einer Strukur selber einen komplexen Summen-/Produkttyp
haben, vor, eigene Hilfsfunktionen zu definieren. Dies können wir wie folgt andeuten:
@#reader scribble/comment-reader
(racketblock
(define (person-has-ancestor p a)
(cond [(person? p)
... (person-name p) ...
... (father-has-ancestor (person-father p) ...)...
... (mother-has-ancestor (person-mother p) ...)...]
[else ...]))
)
Wenn wir uns jedoch etwas genauer überlegen, was @racket[father-has-ancestor] und @racket[mother-has-ancestor] tun sollen,
stellen wir fest, dass sie die gleiche Signatur und Aufgabenbeschreibung haben wie @racket[person-has-ancestor]!
Das bedeutet, dass die Templates für diese Funktionen wiederum jeweils zwei neue Hilfsfunktionen erfordern. Da wir nicht
wissen, wie tief der Stammbaum ist, können wir auf diese Weise das Programm also gar nicht realisieren.
Zum Glück haben wir aber bereits eine Funktion, deren Aufgabe identisch mit der von
@racket[father-has-ancestor] und @racket[mother-has-ancestor] ist, nämlich
@racket[person-has-ancestor] selbst. Dies motiviert das folgende Template:
@#reader scribble/comment-reader
(racketblock
(define (person-has-ancestor p a)
(cond [(person? p)
... (person-name p) ...
... (person-has-ancestor (person-father p) ...)...
... (person-has-ancestor (person-mother p) ...)...]
[else ...]))
)
Die Struktur der Daten diktiert also die Struktur der Funktionen. Dort wo die Daten rekursiv sind, sind auch die Funktionen, die solche
Daten verarbeiten, rekursiv.
Die Vervollständigung des Templates ist nun eifach:
@#reader scribble/comment-reader
(racketblock
(define (person-has-ancestor p a)
(cond [(person? p)
(or
(string=? (person-name p) a)
(person-has-ancestor (person-father p) a)
(person-has-ancestor (person-mother p) a))]
[else #false]))
)
Die erfolgreich ablaufenden Tests illustrieren, dass diese Funktion anscheinend das tut, was sie soll, aber wieso?
Vergleichen wir die Funktion mit einem anderen Ansatz, der nicht so erfolgreich ist. Betrachten wir nochmal
den Anfang der Programmierung der Funktion, diesmal mit anderem Namen:
@#reader scribble/comment-reader
(racketblock
; FamilyTree String -> Boolean
; determines whether person p has an ancestor a
(check-expect (person-has-ancestor-stupid Bob "Joe") #true)
(check-expect (person-has-ancestor-stupid Bob "Emil") #false)
(define (person-has-ancestor-stupid p a) ...)
)
Wenn wir schon eine Funktion hätten, die bestimmen kann, ob eine Person einen bestimmten Vorfahren hat,
könnten wir einfach diese Funktion aufrufen. Aber zum Glück sind wir ja schon gerade dabei, diese Funktion
zu programmieren. Also rufen wir doch einfach diese Funktion auf:
@#reader scribble/comment-reader
(racketblock
; FamilyTree -> Boolean
; determines whether person p has an ancestor a
(check-expect (person-has-ancestor-stupid Bob "Joe") #true)
(check-expect (person-has-ancestor-stupid Bob "Emil") #false)
(define (person-has-ancestor-stupid p a)
(person-has-ancestor-stupid p a))
)
Irgendwie scheint diese Lösung zu einfach zu sein. Ein Ausführen der Tests bestätigt den Verdacht.
Allerdings schlagen die Tests nicht fehl, sondern die Ausführung der Tests terminiert nicht und
wir müssen sie durch Drücken auf die "Stop" Taste abbrechen.
Wieso funktioniert @racket[person-has-ancestor] aber nicht @racket[person-has-ancestor-stupid]?
Zunächst einmal können wir die Programme operationell miteinander vergleichen. Wenn wir den
Ausdruck @racket[(person-has-ancestor-stupid Bob "Joe")] betrachten, so sagt unsere
Reduktionssemantik:
@racket[(person-has-ancestor-stupid Bob "Joe")] @step @racket[(person-has-ancestor-stupid Bob "Joe")]
Es gibt also keinerlei Fortschritt bei der Auswertung. Dies erklärt, wieso die Ausführung nicht stoppt.
Dies ist anders bei @racket[(person-has-ancestor Bob "Joe")]. Die rekursiven Aufrufe rufen
@racket[person-has-ancestor] stets auf Vorfahren der Person auf. Da der Stammbaum nur eine endliche
Tiefe hat, muss irgendwann @racket[(person-has-ancestor #false "Joe")] aufgerufen werden, und wir
landen im zweiten Fall des konditionalen Ausdrucks, der keine rekursiven Ausdrücke mehr enthält.
Wir können das Programm auch aus Sicht der Bedeutung der rekursiven Datentypdefinition betrachten.
Für jede Eingabe @racket[p] gibt es ein minimales i so dass @racket[p] in ft@subscript{i} ist.
Für @racket[Bob] ist dieses i=5. Da die Werte der @racket[mother] und @racket[father] Felder von @racket[p] damit
in @${ft_{i-1}} sind, ist klar, dass der @racket[p] Parameter bei allen rekursiven Aufrufe in @${ft_{i-1}} ist.
Da das Programm offensichtlich für Werte aus @${ft_0} (im Beispiel die Menge {#false}) terminiert, ist
klar, dass dann auch alle Aufrufe mit Werten aus @${ft_1} terminieren müssen, damit dann aber auch
die Aufrufe mit Werten aus @${ft_2} und so weiter. Damit haben wir gezeigt, dass die Funktion
für alle Werte aus @${ft_i} für ein beliebiges i wohldefiniert ist --- mit anderen Worten:
für alle Werte aus @racket[FamilyTree].
Dieses informell vorgetragene Argument aus dem vorherigen Absatz ist mathematisch gesehen ein
Induktionsbeweis.
Bei @racket[person-has-ancestor-stupid] ist die Lage anders, da im rekursiven Aufruf das Argument
eben nicht aus @${ft_{i-1}} ist.
Die Schlussfolgerung aus diesen Überlegungen ist, dass Rekursion in Funktionen unproblematisch
und wohldefiniert ist, solange sie der Struktur der Daten folgt. Da Werte rekursiver
Datentypen eine unbestimmte aber endliche Größe haben, ist diese sogenannte @italic{strukturelle Rekursion}
stets wohldefiniert. Unser Entwurfsrezept schlägt vor, dass immer dort wo Datentypen rekursiv sind,
auch die Funktionen, die darauf operieren, (strukturell) rekursiv sein sollen.
Bevor wir uns das angepasste Entwurfsrezept im Detail anschauen, wollen wir erst noch überlegen, wie
Funktionen aussehen, die Exemplare rekursiver Datentypen @italic{produzieren}.
Als Beispiel betrachten wir eine Funktion, die sehr nützlich ist, um den eigenen Stammbaum etwas
eindrucksvoller aussehen zu lassen, nämlich eine, mit der der Name aller Vorfahren um einen
Titel ergänzt werden kann.
Hier ist die Signatur, Aufgabenbeschreibung und ein Test für diese Funktion:
@#reader scribble/comment-reader
(racketblock
; FamilyTree -> FamilyTree
; prefixes all members of a family tree p with title t
(check-expect
(promote Bob "Dr. ")
(make-person
"Dr. Bob"
(make-person "Dr. Alice" #false #false)
(make-person
"Dr. Horst"
(make-person "Dr. Joe" #false
(make-person "Dr. Rita" #false #false)) #false)))
(define (promote p t) ...)
)
Die Funktion konsumiert und produziert also gleichzeitig einen Wert vom Typ @racket[FamilyTree].
Das Template für solche Funktionen unterscheidet sich nicht von dem für
@racket[person-has-ancestor]. Auch hier wenden wir wieder dort Rekursion an, wo auch der
Datentyp rekursiv ist:
@#reader scribble/comment-reader
(racketblock
(define (promote p t)
(cond [(person? p)
...(person-name p)...
...(promote (person-father p) ...)...
...(promote (person-mother p) ...)...]
[else ...]))
)
Nun ist es recht einfach, die Funktion zu vervollständigen. Um Werte vom Typ @racket[FamilyTree]
zu produzieren, bauen wir die Resultate der rekursiven Aufrufe in einen Aufruf
des @racket[make-person] Konstruktors ein:
@#reader scribble/comment-reader
(racketblock
(define (promote p t)
(cond [(person? p)
(make-person
(string-append t (person-name p))
(promote (person-father p) t)
(promote (person-mother p) t))]
[else p]))
)
@section{Formale Signaturen für rekursive Datentypen}
Wir können formale Signaturen auch für rekursive Datentypen verwenden.
Den äußeren Summentyp können wir über den Signaturkonstruktur @racket[mixed]
abbilden. Wichtig an der formalen Signatur @racket[FamilyTree] ist, dass die
Definition selber rekursiv ist.
@#reader scribble/comment-reader
(racketblock
(define-struct person (name father mother))
(define FamilyTree (signature
(mixed False
(PersonOf String FamilyTree FamilyTree))))
)
Dieser Datentyp kann nun wie gewohnt in Signaturen verwendet werden,
beispielsweise:
@#reader scribble/comment-reader
(racketblock
(: Bob FamilyTree)
(: person-has-ancestor (FamilyTree String -> Boolean))
(define (person-has-ancestor p a) ...)
)
@section{Listen}
Die @racket[FamilyTree] Datendefinition von oben steht für eine Menge von Bäumen, in denen
jeder Knoten genau zwei ausgehende Kanten hat. Selbstverständlich können wir auch auf die
gleiche Weise Bäume repräsentieren, die drei oder fünf ausgehende Kanten haben --- indem wir
eine Alternative des Summentyps haben, in der der Datentyp drei bzw. fünfmal vorkommt.
Ein besonders wichtiger Spezialfall ist der, wo jeder Knoten genau eine ausgehende Kante hat.
Diese degenerierten Bäume nennt man auch @italic{Listen}.
@subsection{Listen, hausgemacht}
Hier ist eine mögliche Definition für Listen von Zahlen:
@#reader scribble/comment-reader
(racketblock
(define-struct lst (first rest))
; A List-of-Numbers is either:
; - (make-lst Number List-Of-Numbers)
; - #false
; interp. the head and rest of a list, or the empty list
)
Hier ist ein Beispiel, wie wir eine Liste mit den Zahlen 1 bis 3 erzeugen können:
@racketblock[(make-lst 1 (make-lst 2 (make-lst 3 #false)))]
Der Entwurf von Funktionen auf Listen funktioniert genau wie der Entwurf von Funktionen auf
allen anderen Bäumen. Betrachten wir als Beispiel eine Funktion, die alle Zahlen in einer
Liste aufaddiert.
Hier ist die Spezifikation dieser Funktion:
@#reader scribble/comment-reader
(racketblock
; List-Of-Numbers -> Number
; adds up all numbers in a list
(check-expect (sum (make-lst 1 (make-lst 2 (make-lst 3 #false)))) 6)
(define (sum l) ...)
)
Für das Template schlägt das Entwurfsrezept vor, die verschiedenen Alternativen zu unterscheiden und in den rekursiven
Alternativen rekursive Funktionsaufrufe einzubauen:
@racketblock[
(define (sum l)
(cond [(lst? l) ... (lst-first l) ... (sum (lst-rest l)) ...]
[else ...]))
]
Auf Basis dieses Templates ist die Vervollständigung nun einfach:
@racketblock[
(define (sum l)
(cond [(lst? l) (+ (lst-first l) (sum (lst-rest l)))]
[else 0]))
]
@subsection{Listen aus der Konserve}
Weil Listen so ein häufig vorkommender Datentyp sind, gibt es in BSL vordefinierte Funktionen für Listen.
Wie @racket[List-of-Numbers] zeigt, benötigen wir diese vordefinierten Funktionen eigentlich nicht, weil wir
Listen einfach als degenerierte Bäume repräsentieren können. Dennoch machen die eingebauten Listenfunktionen
das Programmieren mit Listen etwas komfortabler und sicherer.
Die eingebaute Konstruktorfunktion für Listen in BSL heißt @racket[cons]; die leere Liste wird nicht durch @racket[#false]
sondern durch die spezielle Konstante @racket[empty] repräsentiert.
Es ist sinnvoll, die leere Liste durch einen neuen Wert zu repräsentieren, der für nichts anderes steht, denn dann
kann es niemals zu Verwechslungen kommen. Wenn wir etwas analoges in unserer selbstgebauten Datenstruktur für
Listen machen wollten, könnten wir dies so erreichen, indem wir eine neue Struktur ohne Felder definieren und
ihren einzigen Wert als Variable definieren:
@racketblock[
(define-struct empty-lst ())
(define EMPTYLIST (make-empty-lst))]
Dementsprechend würde die Beispielliste von oben nun so konstruiert:
@racketblock[(make-lst 1 (make-lst 2 (make-lst 3 EMPTYLIST)))]
Die @racket[cons] Operation entspricht unserem @racket[make-lst]
von oben, allerdings mit einem wichtigen Unterschied: Sie überprüft zusätzlich, dass das zweite
Argument auch eine Liste ist:
@interaction[#:eval (bsl-eval) (cons 1 2)]
Man kann sich @racket[cons] also so vorstellen wie diese selbstgebaute Variante von @racket[cons] auf Basis von @racket[List-Of-Numbers]:
@racketblock[
(define (our-cons x l)
(if (or (empty-lst? l) (lst? l))
(make-lst x l)
(error "second argument of our-cons must be a list")))]
Die wichtigsten eingebauten Listenfunktionen sind:
@racket[empty], @racket[empty?], @racket[cons], @racket[first], @racket[rest] und @racket[cons?].
Sie entsprechen in unserem selbstgebauten Listentyp:
@racket[EMPTYLIST], @racket[empty-lst?], @racket[our-cons], @racket[lst-first], @racket[lst-rest] und @racket[lst?].
Die Funktion, die @racket[make-lst] entspricht, wird von BSL versteckt, um sicherzustellen, dass alle Listen
stets mit @racket[cons] konstruiert werden und dementsprechend die Invariante forciert wird, dass das zweite
Feld der Struktur auch wirklich eine Liste ist. Mit unseren bisherigen Mitteln können wir dieses "Verstecken"
von Funktionen nicht nachbauen; dazu kommen wir später, wenn wir über Module reden.
Unser Beispiel von oben sieht bei Nutzung der eingebauten Listenfunktionen also so aus:
@#reader scribble/comment-reader
(racketblock
; A List-of-Numbers is one of:
; - (cons Number List-Of-Numbers)
; - empty
; List-Of-Numbers -> Number
; adds up all numbers in a list
(check-expect (sum (cons 1 (cons 2 (cons 3 empty)))) 6)
(define (sum l)
(cond [(cons? l) (+ (first l) (sum (rest l)))]
[else 0]))
)
@subsection{Die @racket[list] Funktion}
Es stellt sich schnell als etwas mühselig heraus, Listen mit Hilfe von @racket[cons] und @racket[empty] zu konstruieren.
Aus diesem Grund gibt es etwas syntaktischen Zucker, um Listen komfortabler zu erzeugen: Die @racket[list] Funktion.
Mit der @racket[list] Funktion können wir die Liste @racket[(cons 1 (cons 2 (cons 3 empty)))] so erzeugen:
@racketblock[(list 1 2 3)]
Die @racket[list] Funktion ist jedoch nur etwas syntaktischer Zucker, der wie folgt definiert ist:
@racketblock[(list exp-1 ... exp-n)]
steht für n verschachtelte @racket[cons] Ausdrücke:
@racketblock[(cons exp-1 (cons ... (cons exp-n empty)))]
Es ist wichtig, zu verstehen, dass dies wirklich nur eine abkürzende Schreibweise ist. Auch wenn Sie mittels
@racket[list] leichter Listen definieren können, ist es wichtig, stets im Kopf zu behalten, dass dies nur eine
Kurzschreibweise für die Benutzung von @racket[cons] und @racket[empty] ist, denn nur so ist das Entwurfsrezept
für rekursive Datentypen auch auf Listen anwendbar.
@subsection{Datentypdefinitionen für Listen}
Wir haben oben eine Datendefinition @racket[List-of-Numbers] gesehen. Sollen wir auch für @racket[List-of-Strings]
oder @racket[List-of-Booleans] eigene Datendefinitionen schreiben? Sollen diese die gleiche Struktur benutzen, oder
sollen wir separate Strukturen für jeden dieser Datentypen haben?
Es ist sinnvoll, dass alle diese Datentypen die gleiche Struktur benutzen, nämlich in Form der eingebauten
Listenfunktionen. Hierfür gibt es zwei Gründe: Erstens wären all diese Strukturen sehr ähnlich und wir würden damit
gegen das DRY Prinzip verstoßen. Zweitens gibt es eine ganze Reihe von Funktionen, die auf @italic{beliebigen}
Listen arbeiten, zum Beispiel eine Funktion @racket[second], die das zweite Listenelement einer Liste zurückgibt:
@racketblock[
(define (second l)
(if (or (empty? l) (empty? (rest l)))
(error "need at least two list elements")
(first (rest l))))]
Diese Funktion (die übrigens schon vordefiniert ist, genau wie @racket[third], @racket[fourth] und so weiter)
funktioniert für beliebige Listen, unabhängig von der Art der gespeicherten Daten. Solche Funktionen könnten wir
nicht schreiben, wenn wir je nach Typ der Listenelemente andere Strukturen verwenden würden.
Der häufigste Anwendungsfall von Listen ist der, dass die Listen @italic{homogen} sind. Das bedeutet, dass
alle Listenelemente einen gemeinsamen Typ haben. Dies ist keine sehr große Einschränkung, denn dieser
gemeinsame Typ kann beispielsweise auch ein Summentyp mit vielen Alternativen sein. Für diesen Fall
verwenden wir Datendefinitionen mit @italic{Typparametern}, und zwar so:
@#reader scribble/comment-reader
(racketblock
; A (List-of X) is one of:
; - (cons X (List-of X)
; - empty
)
Diese Datendefinitionen benutzen wir, indem wir einen Typ für den Typparameter angeben. Hierzu
verwenden wir die Syntax für Funktionsanwendung; wir schreiben also @racket[(List-of String)], @racket[(List-of Boolean)],
@racket[(List-of (List-of String))] oder @racket[(List-of FamilyTree)] und meinen damit implizit
die oben angeführte Datendefinition.
Was aber ist ein geeigneter Datentyp für die Signatur von @racket[second] oben, also im Allgemeinen
für Funktionen, die auf beliebigen Listen funktionieren?
Um deutlich zu machen, dass die Funktionen für Listen mit beliebigem Elementtyp funktionieren, sagen wir
dies in der Signatur explizit. Im Beispiel der Funktion @racket[second] sieht das so aus:
@#reader scribble/comment-reader
(racketblock
; [X] (List-of X) -> X
(define (second l) ...)
)
Das [X] am Anfang der Signatur sagt, dass diese Funktion die nachfolgende Signatur für jeden Typ X hat,
also (List-of X) -> X für jede mögliche Ersetzung von X durch einen Typen.
Man nennt Variablen wie @racket[X] @italic{Typvariablen}.
Also hat @racket[second] zum Beispiel den Typ (List-of Number) -> Number oder (List-of (List-of String)) -> (List-of String).
Allerdings werden wir im Moment nur im Ausnahmefall Funktionen wie @racket[second] selber programmieren.
Die meisten Funktionen, die wir im Moment programmieren wollen, verarbeiten Listen mit einem konkreten Elementtyp.
Auf sogenannte @italic{polymorphe} Funktionen wie @racket[second] werden wir später noch zurückkommen.
@subsection{Aber sind Listen wirklich rekursive Datenstrukturen?}
Wenn man sich Listen aus der realen Welt anschaut (auf einem Blatt Papier, in einer Tabellenkalkulation etc.),
so suggerieren diese häufig keine rekursive, verschachtelte Struktur sondern sehen "flach" aus. Ist es also
"natürlich", Listen so wie wir es getan haben, über einen rekursiven Datentyp - als degenerierten Baum - zu definieren?
In vielen (vor allem älteren) Programmiersprachen gibt es eine direktere Unterstützung für Listen. Listen sind in
diesen Programmiersprachen fest eingebaut. Listen sind in diesen Sprachen nicht rekursiv; stattdessen wird häufig über
einen Index auf die Elemente der Liste zugegriffen. Um die Programmierung mit Listen zu unterstützen, gibt es
einen ganzen Satz von Programmiersprachenkonstrukten (wie z.B. verschiedenen Schleifen- und Iterationskonstrukten),
die der Unterstützung dieser fest eingebauten Listen dienen.
@margin-note{Es gibt allerdings auch durchaus Situationen, in denen rekursiv aufgebaute Listen effizienter
sind. Mehr dazu später.}
Aus Hardware-Sicht sind solche Listen, über die man mit einem Index zugreift, sehr natürlich, denn sie entsprechen
gut dem von der Hardware unterstützen Zugriff auf den Hauptspeicher.
Insofern sind diese speziellen Listenkonstrukte zum Teil durch ihre Effizienz begründet. Es gibt auch in Racket
(allerdings nicht in BSL) solche Listen; dort heißen sie @italic{Vektoren}.
Der Reiz der rekursiven Formulierung liegt darin, dass man keinerlei zusätzliche Unterstützung für Listen in der
Programmiersprache benötigt. Wir haben ja oben gesehen, dass wir uns die Art von Listen, die von BSL direkt
unterstützt werden, auch selber programmieren können. Es ist einfach "yet another recursive datatype", und alle
Konstrukte, Entwurfsrezepte und so weiter funktionieren nahtlos auch für Listen. Wir brauchen keine speziellen
Schleifen oder andere Spezialkonstrukte um Listen zu verarbeiten, sondern machen einfach das, was wir auch
bei jedem anderen rekursiven Datentyp machen. Das ist bei fest eingebauten
Index-basierten Listen anders; es gibt viele Beispiele für Sprachkonstrukte, die für "normale Werte" funktionieren,
aber nicht für Listen, und umgekehrt.
BSL und Racket stehen in der Tradition der Programmiersprache @italic{Scheme}. Die Philosophie, die im ersten
Satz der Sprachspezifikation von Scheme (@url{http://www.r6rs.org}) zu finden ist, ist damit auch für BSL und Racket gültig:
@italic{Programming languages should be designed not by piling feature on top of feature, but by removing the weaknesses
and restrictions that make additional features appear necessary. Scheme demonstrates that a very small number of
rules for forming expressions, with no restrictions on how they are composed, suffice to form a practical and efficient
programming language that is flexible enough to support most of the major programming paradigms in use today.}
Die Behandlung von Listen als rekursive Datentypen ist ein Beispiel für diese Philosophie.
Manche Programmieranfänger finden es nicht intuitiv, Listen und Funktionen darauf rekursiv zu formulieren.
Aber ist es nicht besser, ein Universalwerkzeug zu verwenden, welches in sehr vielen Situationen verwendbar ist,
als ein Spezialwerkzeug, das in nichts besser ist als das Universalwerkzeug und nur in einer Situation anwendbar ist?
@subsection[#:tag "natrec"]{Natürliche Zahlen als rekursive Datenstruktur}
Es gibt in BSL nicht nur Funktionen, die Listen konsumieren, sondern auch solche, die Listen
produzieren. Eine davon ist @racket[make-list]. Hier ein Beispiel:
@ex[(make-list 4 "abc")]
Die @racket[make-list] Funktion konsumiert also eine natürliche Zahl n und einen Wert und produziert eine Liste mit n Wiederholungen
des Werts. Obwohl diese Funktion also nur atomare Daten konsumiert, produziert sie ein beliebig großes
Resultat. Wie ist das möglich?
@margin-note{Auch in der Mathematik werden natürliche Zahlen ähnlich rekursiv definiert. Recherchieren
sie, was die @italic{Peano-Axiome} sind.}
Eine erleuchtende Antwort darauf ist, dass man auch natürliche Zahlen als Instanzen eines rekursiven Datentyps
ansehen kann. Hier ist eine mögliche Definition:
@#reader scribble/comment-reader
(racketblock
; A Nat (Natural Number) is one of:
; – 0
; – (add1 Nat)
)
Beispielsweise können wir die Zahl @racket[3] repräsentieren als @racket[(add1 (add1 (add1 0)))]. Die @racket[add1]
Funktion hat also eine Rolle ähnlich zu den Konstruktorfunktionen bei Strukturen. Die Rolle der Selektorfunktion
wird durch die Funktion @racket[sub1] übernommen. Das Prädikat für die erste Alternative von Nat ist @racket[zero?];
das Prädikat für die zweite Alternative heißt @racket[positive?].
Mit dieser Sichtweise sind wir nun in der Lage, mit unserem Standard-Entwurfsrezept Funktionen wie @racket[make-list]
selber zu definieren. Nennen wir unsere Variante von @racket[make-list] @racket[iterate-value]. Hier ist
die Spezifikation:
@#reader scribble/comment-reader
(racketblock
; [X] Nat X -> (List-of X)
; creates a list with n occurences of x
(check-expect (iterate-value 3 "abc") (list "abc" "abc" "abc"))
(define (iterate-value n x) ...)
)
Gemäß unseres Entwurfsrezepts für rekursive Datentypen erhalten wir folgendes Template:
@racketblock[
(define (iterate-value n x)
(cond [(zero? n) ...]
[(positive? n) ... (iterate-value (sub1 n) ...)...]))]
Dieses Template zu vervollständigen ist nun nur noch ein kleiner Schritt:
@racketblock[
(define (iterate-value n x)
(cond [(zero? n) empty]
[(positive? n) (cons x (iterate-value (sub1 n) x))]))]
@section{Mehrere rekursive Datentypen gleichzeitig}
Ein schwierigerer Fall ist es, wenn mehrere Parameter einer Funktion einen rekursiven Datentyp haben. In diesem Fall ist es meist
sinnvoll, einen dieser Parameter zu bevorzugen und zu ignorieren, dass andere Parameter ebenfalls rekursiv sind. Welcher
Parameter bevorzugt werden sollte, ergibt sich aus der Fragestellung, wie Sie die Eingabe der Funktion am sinnvollsten zerlegen
können, so dass Sie aus dem Ergebnis des rekursiven Aufrufs und den anderen Parametern am einfachsten das Gesamtergebnis
berechnen können.
Betrachten wir als Beispiel eine Funktion zur Konkatenation von zwei Listen. Diese Funktion ist unter dem Namen @racket[append]
bereits eingebaut, aber wir bauen sie mal nach. Wir haben zwei Parameter, die beide rekursive Datentypen haben.
Eine Möglichkeit wäre, den ersten Parameter zu bevorzugen. Damit erhalten wir folgendes Template:
@#reader scribble/comment-reader
(racketblock
; [X] (list-of X) (list-of X) -> (list-of X)
; concatenates l1 and l2
(check-expect (lst-append (list 1 2) (list 3 4)) (list 1 2 3 4))
(define (lst-append l1 l2)
(cond [(empty? l1) ...l2...]
[(cons? l1) ... (first l1) ... (lst-append (rest l1) ...)]))
)
Tatsächlich ist es in diesem Fall leicht, das Template zu vervollständigen. Wenn wir das Ergebnis von
@racket[(lst-append (rest l1) l2)] haben, so müssen wir nur noch @racket[(first l1)] vorne anhängen:
@racketblock[
(define (lst-append l1 l2)
(cond [(empty? l1) l2]
[(cons? l1) (cons (first l1) (lst-append (rest l1) l2))]))
]
Spielen wir jetzt einmal die zweite Möglichkeit durch: Wir bevorzugen den zweiten Parameter. Damit ergibt
sich folgendes Template:
@racketblock[
(define (lst-append l1 l2)
(cond [(empty? l2) ...l1...]
[(cons? l2) ... (first l2) ... (lst-append .. (rest l2))]))
]
Der Basisfall ist zwar trivial, aber im rekursiven Fall ist es nicht offensichtlich, wie wir das Template vervollständigen können. Betrachten wir beispielsweise
den rekursiven Aufruf @racket[(lst-append l1 (rest l2))], so können wir überlegen, dass uns dieses Ergebnis
überhaupt nicht weiterhilft, denn wir müssten ja irgendwo in der Mitte (aber wo genau ist nicht klar) des Ergebnisses
noch @racket[(first l2)] einfügen.
@section{Entwurfsrezept für Funktionen mit rekursiven Datentypen}
Wir haben gesehen, wie wir mit Hilfe eines Entwurfsrezepts einfache Funktionen
(@secref{entwurfsrezept}), Funktionen mit Summentypen (@secref{entwurfsrezept-summen}),
Funktionen mit Produkttypen (@secref{entwurfsrezept-structs}) und Funktionen mit
algebraischen Datentypen (@secref{entwurfsrezept-adt}) entwerfen.
An dieser Stelle fassen wir zusammen, wie wir das Entwurfsrezept erweitern, um mit Daten beliebiger Größe umzugehen.
@itemize[#:style 'ordered
@item{Wenn es in der Problemdomäne Informationen unbegrenzter Größe gibt, benötigen
Sie eine selbstreferenzierende Datendefinition. Damit eine selbstreferenzierende
Datendefinition gültig ist, muss sie drei Bedingungen erfüllen: 1) Der Datentyp
ist ein Summentyp. 2) Es muss mindestens
zwei Alternativen geben. 3) Mindestens eine der Alternativen referenziert nicht den
gerade definierten Datentyp, ist also nicht rekursiv.
Sie sollten für rekursive Datentypen Datenbeispiele angeben, um zu validieren,
dass Ihre Definition sinnvoll ist. Wenn es nicht offensichtlich ist, wie Sie ihre
Datenbeispiele beliebig groß machen, stimmt vermutlich etwas nicht.}
@item{Beim zweiten Schritt ändert sich nichts: Sie benötigen wie immer eine Signatur, eine
Aufgabenbeschreibung und eine Dummy-Implementierung.}
@item{Bei selbstreferenzierenden Datentypen ist es nicht mehr möglich, für jede Alternative
einen Testfall anzugeben (weil es, wenn man in die Tiefe geht, unendlich viele Alternativen gibt).
Die Testfälle sollten auf jeden Fall alle Teile der Funktion abdecken. Versuchen Sie weiterhin,
kritische Grenzfälle zu identifizieren und zu testen.}
@item{Rekursive Datentypen sind algebraische Datentypen, daher kann für den
Entwurf des Templates die Methodik aus @secref{entwurfsrezept-adt} angewendet
werden. Die wichtigste Ergänzung zu dem Entwurfsrezept betrifft den Fall, dass eine
Alternative implementiert werden muss, die selbstreferenzierend ist. Statt wie sonst
eine neue Hilfsfunktion im Template zu verwenden, wird in diesem Fall ein rekursiver
Aufruf der Funktion, die Sie gerade implementieren, ins Template aufgenommen. Als Argument
des rekursiven Aufrufs wird der Aufruf des Selektors, der den zur Datenrekursion gehörigen
Wert extrahiert, ins Template mit aufgenommen. Wenn Sie beispielsweise eine Funktion
@racket[(define (f a-list-of-strings ...) ...)] auf Listen definieren, so sollte im @racket[cons?]
Fall der Funktion der Aufruf @racket[(f (rest a-list-of-strings) ...)] stehen.}
@item{Beim Entwurf des Funktionsbodies starten wir mit den Fällen der Funktion, die nicht rekursiv sind.
Diese Fälle nennt man auch die @italic{Basisfälle}, in Analogie zu Basisfällen bei Beweisen per Induktion.
Die nicht-rekursiven Fälle sind typischerweise einfach und sollten sich direkt aus den Testfällen ergeben.
In den rekursiven Fällen müssen wir uns überlegen, was der rekursive Aufruf bedeutet. Hierzu nehmen wir an,
dass der rekursive Aufruf die Funktion korrekt berechnet, so wie wir es in der Aufgabenbeschreibung in Schritt 2
festgelegt haben. Mit diesem Wissen müssen wir nun nur noch die zur Verfügung stehenden Werte zur Lösung
zusammenbauen.}
@item{Testen Sie wie üblich, ob ihre Funktion wie gewünscht funktioniert und prüfen Sie, ob die Tests alle interessanten Fälle abdecken.}
@item{Für Programme mit rekursiven Datentypen gibt es einige neue Refactorings, die möglicherweise sinnvoll sind.
Überprüfen Sie, ob ihr Programm Datentypen enthält, die nicht rekursiv sind, aber die durch einen rekursiven
Datentyp vereinfacht werden könnten. Enthält ihr Programm beispielsweise separate Datentypen für Angestellter, Gruppenleiter, Abteilungsleiter,
Bereichsleiter und so weiter, so könnte dies durch einen rekursiven Datentyp, mit dem beliebig tiefe Managementhierarchien
modelliert werden können, vereinfacht werden.
}]
@section{Refactoring von rekursiven Datentypen}
Bezüglich der Datentyp-Refactorings aus @secref{refactoring-adt} ergeben sich neue
Typisomorphien durch "inlining" bzw. Expansion von rekursiven Datendefinitionen. Beispielsweise ist in folgendem Beispiel
@racket[list-of-number] isomorph zu @racket[list-of-number2], denn letztere Definition ergibt sich aus der ersten
indem man die Rekursion einmal expandiert.
@#reader scribble/comment-reader
(racketblock
; A list-of-number is either:
; - empty
; - (cons Number list-of-number)
(define-struct Number-and-Number-List (num numlist))
; A list-of-number2 is either:
; - empty
; - (make-Number-and-Number-List Number list-of-number)
)
Versuchen Sie, Definitionen wie @racket[list-of-number2] zu vermeiden, denn Funktionen, die darauf definiert sind, sind komplexer als solche,
die @racket[list-of-number] verwenden.
Wenn wir die Notation aus @secref{refactoring-adt} verwenden, so können wir rekursive Datentypen als Funktionen modellieren, wobei der
Funktionsparameter das rekursive Vorkommen des Datentyps modelliert. Beispielsweise kann der Datentyp der Listen von Zahlen durch die
Funktion @racket[F(X) = (+ Empty (* Number X))] modelliert werden@note{Solche Funktionen nennt man auch @italic{Funktoren} und sie sind
in der universellen Algebra als @italic{F-Algebras} von wichtiger Bedeutung.}. Das schöne an dieser Notation ist, dass man sehr leicht
definieren kann, wann rekursive Datentypen isormoph sind. Die Expansion eines rekursiven Datentyps ergibt sich daraus, den Funktionsparameter
@racket[X] durch die rechte Seite der Definition zu ersetzen, also in dem Beispiel @racket[F(X) = (+ Empty (* Number (+ Empty (* Number X))))].
Inlining ist die umgekehrte Operation. Die Rechtfertigung für diese Operationen ergibt sich daraus, dass man rekursive Datentypen
als kleinsten Fixpunkt solcher Funktoren verstehen kann. Beispielsweise ist der kleinste Fixpunkt von
@racket[F(X) = (+ Empty (* Number X))] die unendliche Summe @racket[(+ Empty (* Number Empty) (* Number Number Empty) (* Number Number Number Empty) ...)].
Der kleinste Fixpunkt ändert sich durch Expansion oder Inlining nicht, daher sind solche Datentypen isomorph.
@section{Programmäquivalenz und Induktionsbeweise}
Betrachten Sie die folgenden beiden Funktionen:
@#reader scribble/comment-reader
(racketblock
; FamilyTree -> Number
; computes the number of known ancestors of p
(check-expect (numKnownAncestors Bob) 5)
(define (numKnownAncestors p)
(cond [(person? p) (+ 1
(numKnownAncestors (person-father p))
(numKnownAncestors (person-mother p)))]
[else 0]))
; FamilyTree -> Number
; computes the number of unknown ancestors of p
(check-expect (numUnknownAncestors Bob) 6)
(define (numUnknownAncestors p)
(cond [(person? p) (+ (numUnknownAncestors (person-father p))
(numUnknownAncestors (person-mother p)))]
[else 1]))
)
Die Tests suggerieren, dass für alle Personen @racket[p] folgende Äquivalenz gilt: @racket[(+ (numKnownAncestors p) 1)] @equiv @racket[(numUnknownAncestors p)].
Aber wie können wir zeigen, dass diese Eigenschaft tatsächlich stimmt?
Das Schliessen durch Gleichungen, wie wir es in @secref{equationalreasoning} kennengelernt haben, reicht hierzu alleine nicht aus.
Dadurch, dass die Funktionen rekursiv sind, können wir durch @italic{EFUN} immer größere Terme erzeugen, aber wir kommen niemals
von @racket[numKnownAncestors] zu @racket[numUnknownAncestors].
Bei strukturell rekursiven Funktionen auf rekursiven Datentypen können wir jedoch ein weiteres, sehr mächtiges Beweisprinzip verwenden, nämlich das
Prinzip der @italic{Induktion}. Betrachten Sie nochmal die Mengenkonstruktion der ft@subscript{i} aus @secref{rekursivedatentypen}. Wir wissen,
dass der Typ FamilyTree die Vereinigung aller ft@subscript{i} ist. Desweiteren wissen wir, dass, wenn @racket[p] in ft@subscript{i+1} ist, dann ist
@racket[(person-father p)] und @racket[(person-mother p)] in ft@subscript{i}. Dies rechtfertigt die Verwendung des Beweisprinzips der Induktion:
Wir zeigen die gewünschte Äquivalenz für den Basisfall i = 1, also @racket[p] = @racket[#false]. Dann zeigen wir die Äquivalenz für den Fall i = n+1,
unter der Annahme, dass die Äquivalenz bereits für i = n gilt. Anders ausgedrückt zeigen wir für die Äquivalenz für den Fall
@racket[p] = @racket[(make-person n p1 p2)] unter der Annahme, dass die Äquivalenz für @racket[p1] und @racket[p2] gilt.
Typischerweise läßt man bei dieser Art von Induktionsbeweisen die Mengenkonstruktion mit ihren Indizes weg und "übersetzt" die Indizes direkt in die
Datentyp-Notation. Dies bedeutet, dass man die gewünschte Aussage zunächst für die nicht-rekursiven Fälle des Datentyps zeigt, und dann im Induktionsschritt
die Aussage für die rekursiven Fälle zeigt unter der Annahme, dass die Aussage für die Unterkomponenten des Datentyps bereits gilt. Diese Art
des Induktionsbeweises nennt man auch @italic{strukturelle Induktion}.
Wir wollen beweisen: @racket[(+ (numKnownAncestors p) 1)] @equiv @racket[(numUnknownAncestors p)] für alle Personen @racket[p].
Betrachten wir zunächst den Basisfall @racket[p] = @racket[#false].
Dann können wir schliessen:
@racket[(+ (numKnownAncestors #false) 1)]
@equiv (gemäß @italic{EFUN} und @italic{EKONG})
@racket[(+ (cond [ (person? #false) ...] [else 0]) 1)]
@equiv (gemäß @italic{STRUCT-predfalse} und @italic{EKONG})
@racket[(+ (cond [#false ...] [else 0]) 1)]
@equiv (gemäß @italic{COND-False} und @italic{EKONG})
@racket[(+ (cond [else 0]) 1)]
@equiv (gemäß @italic{COND-True} und @italic{EKONG})
@racket[(+ 0 1)]
@equiv (gemäß @italic{PRIM})
@racket[1]
Unter Nutzung von @italic{ETRANS} können wir damit @racket[(+ (numKnownAncestors #false) 1)] @equiv @racket[1] schliessen.
Auf die gleiche Weise können wir schliessen: @racket[(numUnknownAncestors #false)] @equiv @racket[1]. Damit haben wir den Basisfall gezeigt.
Für den Induktionsschritt müssen wir die Äquivalenz für @racket[p] = @racket[(make-person n p1 p2)] zeigen und dürfen hierbei verwenden,
dass die Aussage für @racket[p1] und @racket[p2] gilt, also @racket[(+ (numKnownAncestors p1) 1)] @equiv @racket[(numUnknownAncestors p1)]
und @racket[(+ (numKnownAncestors p2) 1)] @equiv @racket[(numUnknownAncestors p2)].
Wir können nun wie folgt schliessen:
@racket[(+ (numKnownAncestors (make-person n p1 p2)) 1)]
@equiv (gemäß @italic{EFUN} und @italic{EKONG})
@racketblock[(+ (cond [(person? (make-person n p1 p2))
(+ 1
(numKnownAncestors (person-father (make-person n p1 p2)))
(numKnownAncestors (person-mother (make-person n p1 p2))))]
[else 0])
1)]
@equiv (gemäß @italic{STRUCT-predtrue} und @italic{EKONG})
@racketblock[(+ (cond [#true
(+ 1
(numKnownAncestors (person-father (make-person n p1 p2)))
(numKnownAncestors (person-mother (make-person n p1 p2))))]
[else 0])
1)]
@equiv (gemäß @italic{COND-True} und @italic{EKONG})
@racketblock[(+ (+ 1
(numKnownAncestors (person-father (make-person n p1 p2)))
(numKnownAncestors (person-mother (make-person n p1 p2))))
1)]
@equiv (gemäß @italic{STRUCT-select} und @italic{EKONG})
@racketblock[(+ (+ 1
(numKnownAncestors p1)
(numKnownAncestors p2))
1)]
@equiv (gemäß @italic{EPRIM})
@racketblock[(+
(+ (numKnownAncestors p1) 1)
(+ (numKnownAncestors p2) 1))]
@equiv (gemäß Induktionsannahme und @italic{EKONG})
@racketblock[(+
(numUnknownAncestors p1)
(numUnknownAncestors p2))]
@equiv (gemäß @italic{STRUCT-select} und @italic{EKOMM} und @italic{EKONG})
@racketblock[(+
(numUnknownAncestors (person-father (make-person n p1 p2)))
(numUnknownAncestors (person-mother (make-person n p1 p2))))]
@equiv (gemäß @italic{EFUN} und @italic{EKOMM})
@racket[(numUnknownAncestors (make-person n p1 p2))]
Damit haben wir (unter Nutzung von @italic{ETRANS}) die Äquivalenz bewiesen. Dieser Beweis ist sehr ausführlich und kleinschrittig.
Wenn Sie geübter im Nutzen von Programmäquivalenzen sind, werden ihre Beweise großschrittiger und damit
auch kompakter.
Die gleiche Beweismethodik lässt sich für alle rekursiven Datentypen anwenden. Insbesondere lässt sie sich auch für Listen anwenden.