forked from keuhdall/learn4haskell
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathChapter3.hs
1147 lines (872 loc) Β· 38.2 KB
/
Chapter3.hs
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
990
991
992
993
994
995
996
997
998
999
1000
{- π Welcome to Chapter Three of our journey, Courageous Knight!
Glad to see you back for more challenges. You fight great for the glory of the
Functional Programming in the previous Chapters. We are grateful that you are
continuing to walk this road with us.
This Chapter requires the knowledge from the previous modules, so it is highly
encouraged to at least look through the material of those chapters.
Let's refresh how the training works.
== You are on __Chapter Three__.
At this step, we are going to learn more about types in Haskell and explore
typeclasses. You will need to create a lot of types, so we rely on your
creativity, as you will be given the opportunity to create new worlds out of
Haskell code.
Specifically, in this chapter, you are going to practice:
β§ Types in Haskell
β§ ADTs: Algebraic Data Types
β§ Type aliases vs Data types vs Newtypes
β§ Parametric polymorphism
β§ Typeclasses
β§ Ad-hoc polymorphism
As usual, the explanations are in the Haskell comments of this module. We are
leaving a number of tasks on our path. Your goal is to solve them all.
After finishing the PR, you can choose to summon us, @vrom911 and @chshersh, to
look at your solution in order to give some advice on your code. This is
optional; however, you can ask us for review only if you want some feedback on
your solutions.
Okay. Ready? Set. Go!
-}
{-
=βοΈ= Language extensions*
Some Haskell features are not enabled by default. They can be enabled by turning
on corresponding extensions (the language feature) using the LANGUAGE pragma at
the top of the file before the "module" declaration. The fact that they are not
enabled by default doesn't mean they are experimental. This is just the way
Haskell works.
In this module, we enable the "InstanceSigs" feature that allows writing type
signatures in places where you can't by default. We believe it's helpful to
provide more top-level type signatures, especially when learning Haskell.
-}
{-# LANGUAGE InstanceSigs #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE RecordWildCards #-}
module Chapter3 where
{-
=π‘= Types in Haskell
Let's talk about types.
In the previous chapters, we have already worked a lot with them. And we bet,
you can name a few at this point.
But we mostly were operating on primitive types, like 'Int' or 'Bool'. We saw
the function (->) a lot! We briefly touched tuples, but we learned that lists
are far more complicated!
Haskell has several different ways to create entirely new data types. Let's talk
about them all and master our skill of data types construction.
-}
{- |
=π‘= Type aliases
The simplest way to introduce a new type in Haskell is __type aliases__. Type
aliases are nothing more than just other names to already existing types.
A few examples:
@
type IntList = [Int]
type IntPair = (Int, Int)
@
Type aliases are just syntactic sugar and would be replaced by the real types
during compilation, so they don't bring too much to the table of data types.
However, they may make the life of a developer easier in some places.
One of the most common type aliases is 'String' that we already mentioned in the
List section of the previous chapter. And it is defined in the following way:
@
type String = [Char]
@
Now it makes much more sense of why 'String' is related to lists. You can see
that any list function could be used on 'String's as well.
π©βπ¬ Due to the implementation details of lists, such representation of String
is highly inefficient. It is unfortunate that the list of characters is the
default String type. Experienced Haskellers use more efficient string types
'Text' and 'ByteString' from Haskell libraries: 'text' and 'bytestring'
correspondingly. But, for the simplicity of this training, we are using
'String'.
Another common type alias in the standard library is "FilePath", which is the
same as 'String' (which is the same as '[Char]'):
@
type FilePath = String
@
Usually, you are encouraged to introduce new data types instead of using aliases
for existing types. But it is still good to know about type aliases. They have a
few use-cases, and you can meet them in various Haskell libraries as well.
-}
{- |
=π‘= ADT
Let's not limit ourselves with just type aliases and define some real new data
structures. Are we the creators of new worlds or what?
Type in Haskell is like a box of information description that the object of that
type should contain.
Haskell uses Algebraic Data Types (ADT) system of types. That means that there
are two types of types: product and sum types.
To give you some basic understanding of the difference between these two, let's
go to the book shop. A book in there represents a product type: each book has
the name, author, cover, pages, etc. And all of these properties are mandatory
and come with the book. Bookshelf, in its turn, is the sum type. Each book in a
shelf is a different type, and you can choose one of them at once (there is no
such book where two or more physical books are sewed together).
We will show you an example now, just to illustrate all the above and then will
explain each concept separately. Note, this is not real syntax.
@
-- Product type
Book:
book name
AND book author
AND book cover
AND book pages
-- Sum type
BookShelf:
Good book 1 : {Book}
OR Good book 2 : {Book}
OR Cheap book 3 : {Book}
@
π©βπ¬ We use AND in product types to represent the notion of having all fields at
the same time. In contrast, for the sum types, we use OR to tell only about a
single possibility. AND in logic corresponds to multiplication in math, and OR
corresponds to addition. You see that there is some math theory behind the
concept of data types in Haskell, and that's why they are called Algebraic Data
Types.
-}
{- |
=π‘= Product type
Let's now see how the product data types look like in Haskell.
Product type should have a type name, one type constructor (the function that
lets you create the value of the type later) and the description of the fields
it consists of in the view of types.
When defining a custom type in Haskell, you use the __"data"__ keyword, write
the __type name__ you come up with after it, and then after the "=" sign you
specify the __constructor name__ followed by the __fields__.
When in action, a custom data type could be used in the following use case.
A definition of a data type for a knight with a name and number of victories
can look like this:
@
ββ type name
β
β ββ constructor name (or constructor tag)
β β
data Knight = MkKnight String Int
β β β
β ββββββ΄βββ types of fields
β
β "data" keyword
@
β« NOTE: The constructor can have the same name as the type itself. And in most
cases, they are indeed named identically. This is not a problem for Haskell,
because types live in the types namespace, and constructors live in the value
namespace. So there won't be any collisions and misunderstandings from the
compiler side. The names are unambiguous.
You can use the constructor name, to create a value of the "Knight" type.
@
arthur :: Knight
arthur = MkKnight "Arthur" 100
@
A constructor is just a function from fields to the type! You can verify this in GHCi:
ghci> :t MkKnight
MkNight :: String -> Int -> Knight
As in a regular function, you need to provide a 'String' and an 'Int' to the
'MkNight' constructor in order to get the full-fledged 'Knight'.
Also, you can write a function that takes a Knight and returns its name.
It is convenient to use pattern matching for that:
@
knightName :: Knight -> String
knightName (MkKnight name _) = name
@
And you can extract the name in GHCi:
ghci> knightName arthur
"Arthur"
It is comfy to have such getters for all types, so Haskell provides a syntax for
defining __records__ β named parameters for the product data type fields.
Records have similar syntax for defining in Haskell as (unnamed) ordinary
product types, but fields are specified in the {} separated by a comma. Each
field should have a name and a type in this form: 'fieldName :: FieldType'.
The same definition of 'Knight' but with records should be written in the
following way:
@
data Knight = MkKnight
{ knightName :: String
, knightVictories :: Int
}
@
The above type definition is equivalent to the one we had before. We just gave
names to our fields for clarity. In addition, we also automatically get
functions "knightName :: Knight -> String" and "knightVictories :: Knight ->
Int". This is what records bring us β free getters!
The pattern matching on constructors of such records can stay the same. Besides,
you can use field names as getters.
π©βπ¬ We are using a particular naming scheme of record field names in record
types in order to avoid name collisions. We add the data type name prefix in
front of each usual field name for that. As we saw, records create getters for
us, which are actually top-level functions. In Haskell, all functions defined at
top-level are available in the whole scope within a module. But Haskell forbids
creating multiple functions with the same name. The Haskell ecosystem has
numerous ways of solving this so-called "record" problem. Still, for simplicity
reasons, we are going to use disambiguation by a prefix which is one of the
standard resolutions for the scope problem.
In addition to getting top-level getters automatically, you get the following
features for free when using records over unnamed type fields:
1. Specify names during constructions β additional visual help for code.
2. Record-update syntax.
By default, all functions and constructors work with positional arguments
(unnamed, that are identified by its position in the type declaration). You
write a function or a constructor name first, and then you pass arguments
separated by space. But once we declare a product type as a record, we can use
field names for specifying constructor values. That means that the position
doesn't matter anymore as long as we specify the names. So it is like using
named arguments but only for constructors with records.
This is an alternative way of defining values of custom records.
Let's introduce Sir Arthur properly!
@
arthur :: Knight
arthur = MkKnight
{ knightName = "Arthur"
, knightVictories = 100
}
@
After we created our custom types and defined some values, we may want to change
some fields of our values. But we can't actually change anything! Remember that
all values in Haskell are immutable, and you can't just change a field of some
type. You need to create a new value! Fortunately, for records, we can use the
__record update syntax__. Record update syntax allows creating new objects of a
record type by assigning new values to the fields of some existing record.
The syntax for record update is the following:
@
lancelot :: Knight
lancelot = arthur { knightName = "Lancelot" }
@
Without records, we had to write a custom setter function for each field of each
data type.
@
setKnightName :: String -> Knight -> Knight
setKnightName newName (MkKnight _ victories) =
MkKnight newName victories
@
β« NOTE: By default, GHCi doesn't know how to display values of custom types. If
you want to explore custom data types in REPL, you need to add a magical
"deriving (Show)" line (will be explained later in this chapter) at the end of a
record. Like so:
@
data Knight = MkKnight
{ knightName :: String
, knightVictories :: Int
} deriving (Show)
@
Now GHCi should be able to show the values of your types! Try playing with our
knights in GHCi to get the idea behind records.
π― HINT: At this point, you may want to be able to enter multi-line strings in
GHCi. Start multi-line blocks by typing the ":{" command, and close such blocks
using the ":}" command.
ghci> :{
ghci| data Knight = MkKnight
ghci| { knightName :: String
ghci| , knightVictories :: Int
ghci| } deriving (Show)
ghci| :}
ghci>
Although, it may be easier to define data types in the module, and load it
afterwards.
-}
{- |
=βοΈ= Task 1
Define the Book product data type. You can take inspiration from our description
of a book, but you are not limited only by the book properties we described.
Create your own book type of your dreams!
-}
data Genre = Fantasy | Novel | Thriller | SciFi | Other String deriving (Eq, Show)
data Book = Book {
title :: String,
author :: String,
pages :: Int,
genre :: Genre
} deriving (Eq, Show)
{- |
=βοΈ= Task 2
Prepare to defend the honour of our kingdom! A monster attacks our brave knight.
Help him to fight this creature!
Define data types for Knights and Monsters, and write the "fight" function.
Both a knight and a monster have the following properties:
β¦ Health (the number of health points)
β¦ Attack (the number of attack units)
β¦ Gold (the number of coins)
When a monster fights a knight, the knight hits first, and the monster hits back
only if it survives (health is bigger than zero). A hit decreases the amount of
health by the number represented in the "attack" field.
Implement the "fight" function, that takes a monster and a knight, performs the
fight following the above rules and returns the amount of gold the knight has
after the fight. The battle has the following possible outcomes:
β Knight wins and takes the loot from the monster and adds it to their own
earned treasured
β Monster defeats the knight. In that case return -1
β Neither the knight nor the monster wins. On such an occasion, the knight
doesn't earn any money and keeps what they had before.
-}
data Knight = Knight {
knightName :: String,
knightHp :: Int,
knightAttack :: Int,
knightGold :: Int
} deriving (Eq, Show)
data Monster = Monster {
monsterHp :: Int,
monsterAttack :: Int,
monsterGold :: Int
} deriving (Eq, Show)
fight :: Knight -> Monster -> Int
fight Knight{..} Monster{..}
| knightAttack > monsterHp = knightGold + monsterGold
| monsterAttack > knightHp = -1
| otherwise = knightGold
{- |
=π‘= Sum types
Another powerful ambassador of ADT is __sum type__. Unlike ordinary records
(product types) that always have all the fields you wrote, sum types represent
alternatives of choices. Sum types can be seen as "one-of" data structures. They
contain many product types (described in the previous section) as alternatives.
To define a sum type, you have to specify all possible constructors separated
by "|". Each constructor on its own could have an ADT, that describes
this branch of the alternative.
There is at least one famous sum type that you have already seen β 'Bool' β the
simplest example of a sum type.
@
data Bool = False | True
@
'Bool' is a representer of so-called __enumeration__ β a special case of sum
types, a sum of nullary constructors (constructors without fields).
Sum types can have much more than two constructors (but don't abuse this)!
Look at this one. We need more than two constructors to mirror the "Magic Type".
And none of the magic streams needs any fields. Just pure magic ;)
@
data MagicType
= DarkMagic
| LightMagic
| NeutralMagic
@
However, the real power of sum types unleashes when you combine them with
fields. As we mentioned, each "|" case in the sum type could be an ADT, so,
naturally, you can have constructors with fields, which are product types from
the previous section. If you think about it, the enumeration also contains a
product type, as it is absolutely legal to create a data type with one
constructor and without any fields: `data Emptiness = TotalVoid`.
To showcase such sum type, let's represent a possible loot from successfully
completing an adventure:
@
data Loot
= Sword Int -- attack
| Shield Int -- defence
| WizardStaff Power SpellLevel
@
You can create values of the sum types by using different constructors:
@
woodenSword :: Loot
woodenSword = Sword 2
adamantiumShield :: Loot
adamantiumShield = Shield 3000
@
And you can pattern match on different constructors as well.
@
acceptLoot :: Loot -> String
acceptLoot loot = case loot of
Sword _ -> "Thanks! That's a great sword!"
Shield _ -> "I'll accept this shield as a reward!"
WizardStaff _ _ -> "What?! I'm not a wizard, take it back!"
@
To sum up all the above, a data type in Haskell can have zero or more
constructors, and each constructor can have zero or more fields. This altogether
gives us product types (records with fields) and sum types (alternatives). The
concept of product types and sum types is called __Algebraic Data Type__. They
allow you to model your domain precisely, make illegal states unrepresentable
and provide more flexibility when working with data types.
-}
{- |
=βοΈ= Task 3
Create a simple enumeration for the meal types (e.g. breakfast). The one who
comes up with the most number of names wins the challenge. Use your creativity!
-}
data MealTypes = Breakfast | Brunch | Lunch | TeaTime | Dinner | Snack | AfterPartyMeal
{- |
=βοΈ= Task 4
Define types to represent a magical city in the world! A typical city has:
β Optional castle with a __name__ (as 'String')
β Wall, but only if the city has a castle
β Church or library but not both
β Any number of houses. Each house has one, two, three or four __people__ inside.
After defining the city, implement the following functions:
β¦ buildCastle β build a castle in the city. If the city already has a castle,
the old castle is destroyed, and the new castle with the __new name__ is built
β¦ buildHouse β add a new living house
β¦ buildWalls β build walls in the city. But since building walls is a
complicated task, walls can be built only if the city has a castle
and at least 10 living __people__ inside in all houses of the city totally.
-}
data Castle = Castle(String) | CastleWithWalls(String) | None deriving (Eq, Show)
data Building = Church | Library deriving Show
data PeopleAmount = One | Two | Three | Four deriving Show
data House = House {
peopleAmount :: PeopleAmount
} deriving Show
data City = City {
castle :: Castle,
building :: Building,
houses :: [House]
} deriving Show
buildCastle :: City -> String -> City
buildCastle city castleName = case castle city of
CastleWithWalls _ -> city
_ -> city{castle = Castle castleName}
buildHouse :: City -> House -> City
buildHouse city@City{houses} house = city{houses = house:houses}
buildWalls :: City -> City
buildWalls city@City{castle,houses} = case castle of
Castle castleName -> if pplCount > 10 then city{castle = CastleWithWalls(castleName)} else city
_ -> city
where
pplCount = sum $ map (\house' -> pplInHouse . peopleAmount $ house') houses :: Int
pplInHouse pa = case pa of
One -> 1
Two -> 2
Three -> 3
Four -> 4
{-
=π‘= Newtypes
There is one more way to create a custom structure in Haskell. Let's see what
that is and how it differs from others.
__Newtype__ is a way to create a lightweight wrapper around an existing type.
Unlike type aliases, newtypes make an entirely new type for the compiler point
of view (as the name suggests). However, such data types don't have additional
runtime overhead, which means that it would work as fast as the underlying type
without the wrapper.
You can declare a data type as a newtype only if it has __exactly one
constructor__ with __exactly one field__. It is a compiler error if you try to
define a newtype with another number of constructors or fields.
The syntax is similar to defining an ordinary data type, but you use the
"newtype" keyword instead of the "data" keyword. "newtype" is a product type.
@
newtype Attack = MkAttack Int
@
The same rule about names fields as for any data types applies to newtypes
as well. Meaning you can write the above type as follows:
@
newtype Attack = MkAttack
{ unAttack :: Int
}
@
You can use the "MkAttack" constructor to create values of the "Attack" type,
and you can pattern match on "MkAttack" as on ordinary data types. When using
newtypes, you pay an extra development cost of writing extra wrappers and
unwrappers, but at the same time, you get additional compile-time guarantees of
not mixing types.
Newtypes serve the purpose of creating safer and more maintainable interfaces.
Let's prove that.
Say, we have a function to get a BMI.
@
myBMI :: Double -> Double -> Double
myBMI height weight = ...
@
And I can use it to calculate my BMI:
ghci> myBMI 200 70
Imagine how terrifying it could be if one accidentally messes with the order of
height and weight. π±
ghci> myBMI 70 200
However, this could be avoided if our function would look like this:
@
myBMI :: Height -> Weight -> Double
-- | β°β΄ newtype
-- β°β΄newtype
myBMI height weight = ...
@
And to run it you won't be able to mess arguments:
ghci> myBMI (Height 200) (Weight 70)
-}
{-
=βοΈ= Task 5
Improve the following code (types definition and function implementations) by
introducing extra newtypes.
π― HINT: if you complete this task properly, you don't need to change the
implementation of the "hitPlayer" function at all!
-}
newtype Health = Health Int
newtype Armor = Armor Int
newtype Attack = Attack Int
newtype Dexterity = Dexterity Int
newtype Strength = Strength Int
newtype Damage = Damage Int
newtype Defense = Defense Int
data Player = Player
{ playerHealth :: Health
, playerArmor :: Armor
, playerAttack :: Attack
, playerDexterity :: Dexterity
, playerStrength :: Strength
}
calculatePlayerDamage :: Attack -> Strength -> Damage
calculatePlayerDamage (Attack a) (Strength s) = Damage $ a + s
calculatePlayerDefense :: Armor -> Dexterity -> Defense
calculatePlayerDefense (Armor a) (Dexterity d) = Defense $ a * d
calculatePlayerHit :: Damage -> Defense -> Health -> Health
calculatePlayerHit (Damage da) (Defense de) (Health hp) = Health $ hp + de - da
-- The second player hits first player and the new first player is returned
hitPlayer :: Player -> Player -> Player
hitPlayer player1 player2 =
let damage = calculatePlayerDamage
(playerAttack player2)
(playerStrength player2)
defense = calculatePlayerDefense
(playerArmor player1)
(playerDexterity player1)
newHealth = calculatePlayerHit
damage
defense
(playerHealth player1)
in player1 { playerHealth = newHealth }
{- |
=π‘= Polymorphic data types
Similar to functions, data types in Haskell can be __polymorphic__. This means
that they can use some type variables as placeholders, representing general
types. You can either reason about data types in terms of such variables (and
don't worry about the specific types), or substitute variables with some
particular types.
Such polymorphism in Haskell is an example of the __parametric polymorphism__.
The process of defining a polymorphic type is akin to the ordinary data type
definition. The only difference is that all the type variables should go after
the type name so that you can reuse them in the constructor fields later.
For example,
@
data Foo a = MkFoo a
@
Note that both product and sum types can be parameterised.
> Actually, we've already seen a polymorphic data type! Remember Lists from Chapter Two?
To give an example of a custom polymorphic type, let's implement a
"TreasureChest" data type. Our treasure chest is flexible, and it can store some
amount of gold. Additionally there is some space for one more arbitrary
treasure. But that could be any treasure, and we don't know what it is
beforehand.
In Haskell words, the data type can be defined like this:
@
data TreasureChest x = TreasureChest
{ treasureChestGold :: Int
, treasureChestLoot :: x
}
@
You can see that a treasure chest can store any treasure, indeed! We call it
treasure 'x'.
And when writing functions involving the "TreasureChest" type, we don't always
need to know what kind of treasure is inside besides gold.
We can either use a type variable in our type signature:
@
howMuchGoldIsInMyChest :: TreasureChest x -> Int
@
or we can specify a concrete type:
@
isEnoughDiamonds :: TreasureChest Diamond -> Bool
@
In the same spirit, we can implement a function that creates treasure with some
predefined amount of gold and a given treasure:
@
mkMehChest :: x -> TreasureChest x
mkMehChest treasure = TreasureChest
{ treasureChestGold = 50
, treasureChestLoot = treasure
}
@
Polymorphic Algebraic Data Types are a great deal! One of the most common and
useful standard polymorphic types is __"Maybe"__. It represents the notion of
optional value (maybe the value is there, or maybe it is not).
"Maybe" is defined in the standard library in the following way:
@
data Maybe a
= Nothing
| Just a
@
Haskell doesn't have a concept of "null" values. If you want to work with
potentially absent values, use the "Maybe" type explicitly.
> Is there a good way to avoid null-pointer bugs? Maybe. Β© Jasper Van der Jeught
Another standard polymorphic data type is "Either". It stores either the value
of one type or a value of another.
@
data Either a b
= Left a
| Right b
@
β« NOTE: It can help to explore types of constructors "Nothing", "Just", "Left"
and "Right". Let's stretch our fingers and blow off the dust from our GHCi and
check that!
You can pattern match on values of the "Either" type as well as on any other
custom data type.
@
showEither :: Either String Int -> String
showEither (Left msg) = "Left with string: " ++ msg
showEither (Right n) = "Right with number: " ++ show n
@
Now, after we covered polymorphic types, you are finally ready to learn how
lists are actually defined in the Haskell world. Behold the might list type!
@
data [] a
= []
| a : [a]
@
Immediately we know what all of that means!
The ":" is simply the constructor name for the list. Constructors in Haskell can
be defined as infix operators as well (i.e. be written after the first argument,
the same way we write `1 + 2` and not `+ 1 2`), but only if they start with a
colon ":". The ":" is taken by lists. Now you see why we were able to pattern
match on it?
The type name uses built-in syntax to reserve the square brackets [] exclusively
for lists but, otherwise, is a simple polymorphic recursive sum type.
If you rename some constructor and type names, the list type could look quite
simple, as any of us could have written it:
@
data List a
= Empty
| Cons a (List a)
@
β« NOTE: We use () to group "List" with "a" type variable in the second field of
the "Cons" constructor. This is done to tell the compiler that "List" and "a"
should go together as one type.
-}
{- |
=βοΈ= Task 6
Before entering the real world of adventures and glorious victories, we should
prepare for different things in this world. It is always a good idea to
understand the whole context before going for a quest. And, before fighting a
dragon, it makes sense to prepare for different unexpected things. So let's
define data types describing a Dragon Lair!
β A lair has a dragon and possibly a treasure chest (as described in the
previous section). A lair also may not contain any treasures, but we'll never
know until we explore the cave!
β A dragon can have a unique magical power. But it can be literally anything!
And we don't know in advance what power it has.
Create data types that describe such Dragon Lair! Use polymorphism to
parametrise data types in places where values can be of any general type.
π― HINT: 'Maybe' that some standard types we mentioned above are useful for
maybe-treasure ;)
-}
data TreasureChest x = TreasureChest {
treasureChestGold :: Int,
treasureChestLoot :: x
}
data DragonLair a b = DragonLair {
treasure :: Maybe (TreasureChest a),
power :: b
}
{-
=π‘= Typeclasses
__Typeclass__ is a regularly used way to express common characteristics of the
different data types. In some sense, a typeclass describes the interface of some
value without telling you the implementation details.
__Instance__ is a representation of the typeclass βοΈοΈ data type relationships. In
order to show that the data type obeys the typeclasses rules and to use the
methods of the typeclass on the data values, you need to provide the work
instructions under this particular typeclass. And that is the instance of the
data type for the specific typeclass.
Letβs consolidate the typeclasses and instances concepts on the analogues from
our fantasy world.
Many lovely princesses need to be rescued. And those processes are usually
alike: find a path to the particular castle (differs from princess to princess),
defeat the monster (also unique for each princess) and rescue the beloved
beauty. So we can see that the "Rescue Mission Plan" is the typeclass, and each
princess has its own instance for that. If you are a prince on a white horse,
you'd better check the particular instance for your princess to get into the
salvation journey.
Next, letβs look at one code example for the better illustration of the
instance-typeclass relationship. We can define a typeclass that would tell us
one's arch enemy.
The syntax is as follows: you need to use the "class" keyword, then you need to
specify the typeclass name. Typeclasses should start with the upper letter.
After that, the type parameter should be identified, which represents the data
types that would have instances of this typeclass. And, finally, the "where"
keyword. After, you can specify methods of the typeclass β functions that should
work with the type parameter.
@
ββ typeclass name
β
β ββ type parameter
β β ββββββ "where" keyword
β β β
class ArchEnemy a where
β getArchEnemy :: a -> String
β β β
β β ββββ the same type parameter
β ββββββ method name
β
β "class" keyword
@
And that 'getArchEnemy' method could be used with a lot of data types: Bool,
Double,β¦ name them all! Letβs have our first instances to show how it works:
The syntax is simple and consistent with the typeclass declaration, but instead
of the "class: keyword you need to have the "instance" keyword. Of course,
instead of the type parameter, you have to specify the concrete type, for which
you are implementing the instance. And all the necessary methods should have its
implementation for the particular data type.
@
instance ArchEnemy Bool where
getArchEnemy :: Bool -> String
getArchEnemy True = "False"
getArchEnemy False = "True"
instance ArchEnemy Int where
getArchEnemy :: Int -> String
getArchEnemy i = case i of
0 -> "Division"
_ -> "Derivative"
instance ArchEnemy Double where
getArchEnemy :: Double -> String
getArchEnemy n
| isNaN n = "Infinity"
| isInfinite n = "Limit"
| otherwise = "NaN"
@
And then you can write polymorphic functions and not worry about which specific
type is underhood until it has the instance of the desired typeclass. For that
we are using __constrains__ in Haskell. It is the identification of affiliation
to the typeclass. The constraints should go after the "::" sign in the function
type declaration. You can specify one or many constraints. If more than one they
should be in parenthesis and comma-separated. The end of constraints is
determined with the "=>" arrow, and the function type could be written as usual.
@
revealArchEnemy :: (ArchEnemy a, Show a) => a -> String
revealArchEnemy x =
"The arch-enemy of " ++ show x ++ " is " ++ getArchEnemy x
@
The behaviour of this polymorphic function depends on the data type used with
this function. Such dependency is called __ad-hoc polymorphism__.
This is how it works in action:
ghci> revealArchEnemy (42 :: Int)
"The arch-enemy of 42 is Derivative"
However, if we try to use this function with something that doesnβt implement an
instance of our typeclass, we will get the corresponding compiler error, that
would warn us precisely about that:
ghci> revealArchEnemy "An adorable string that has no enemies (βΏβ Οβ )"
<interactive>:21:1: error:
β’ No instance for (ArchEnemy String)
arising from a use of revealArchEnemy
β’ In the expression: revealArchEnemy "An adorable string that has no enemies (βΏβ Οβ )"
In an equation for 'it': it = revealArchEnemy "An adorable string that has no enemies (βΏβ Οβ )"
Interestingly, it is possible to reuse existing instances of data types in the
same typeclass instances as well. And we also can reuse the __constraints__ in
the instance declaration for that!
This gives us the ability to specify the instances for polymorphic data types
with some conditions (constraints).
Let's see it in the example of the 'ArchEnemy' typeclass instance for the
"Maybe" something data type.
@
instance (ArchEnemy a) => ArchEnemy (Maybe a) where
getArchEnemy :: Maybe a -> String
getArchEnemy (Just x) = getArchEnemy x
getArchEnemy Nothing = "NullPointerException"
@
This instance is suitable for any Maybe as long as the instance of the inside
type exists. You can see how we reuse the fact that the underlying type has this
instance and apply this typeclass method to it.
-}
{- |
=βοΈ= Task 7
Often we want to combine several values of a type and get a single value of the
exact same type. We can combine different things: treasures, adventures, groups
of heroes, etc.. So it makes sense to implement a typeclass for such a concept
and define helpful instances.
We will call such a typeclass "Append". You can find its definition below.
Implement instances of "Append" for the following types: