-
Notifications
You must be signed in to change notification settings - Fork 0
Home
In deze cursus gaan we de programmeertaal OCaml gebruiken. (F# is een uitbreiding van OCaml.)
We gaan de online programmeeromgeving Ocsigen gebruiken.
OCaml ondersteunt verschillende soorten waarden.
Opdracht:
- Tik
2 + 2
in. - OCaml antwoordt met
- : int = 4
. OCaml geeft aan dat de opgegeven uitdrukking van het typeint
is (het type van de gehele getallen, integers in het Engels) en evalueert tot de waarde 4.
Opdracht:
- Tik
"Hello, " ^ "world!"
in. - OCaml antwoordt met
- : string = "Hello, world!"
. De opgegeven uitdrukking is van het typestring
(het type van de stukken tekst) en evalueert tot de waarde"Hello, world!"
. Het bewerkingsteken^
dient om stukken tekst aan elkaar te plakken (string concatenation in het Engels).
Opdracht:
- Tik
1 / 2
in. - OCaml antwoordt met
- : int = 0
. Het bewerkingsteken/
staat voor gehele deling.
Opdracht:
- Tik
1.0 /. 2.0
in. - OCaml antwoordt met
- : float = 0.5
. De opgegeven uitdrukking is van het typefloat
(het type van de vlottende-kommagetallen, floating-point numbers in het Engels). Het bewerkingsteken/.
staat voor reële deling.
Opdracht:
- Tik
1 /. 2
in. - OCaml onderlijnt de uitdrukking
1
in het rood en zegt erover Error: This expression has typeint
but an expression was expected of typefloat
. OCaml is heel streng qua het correct gebruik van types: de reële deling mag enkel gebruikt worden met kommagetallen, en de gehele deling enkel met gehele getallen.
Opdracht:
- Tik
1 < 2
in. - OCaml antwoordt met
- : bool = true
. De ongelijkheid is waar (true
in Engels).true
enfalse
zijn de waarheidswaarden, ook Booleaanse waarden genoemd. Ze zijn van het typebool
.
Opdracht:
- Tik
true && false
in. - OCaml antwoordt met
- : bool = false
. Het bewerkingsteken&&
staat voor de logische EN-operatie.
Andere opdrachten zijn: aftrekking -
en vermenigvuldiging *
van gehele getallen, optelling +.
, aftrekking -.
, en vermenigvuldiging *.
van kommagetallen, en de logische OF-operatie ||
op Booleaanse waarden.
Opdracht:
- Tik
let x = 3
in. - Tik dan
x + 5
in. - OCaml antwoordt met
- : int = 8
. De opdrachtlet x = 3
bindt de waarde 3 aan de variabelex
.
Opdracht:
- Tik
let f x = x + 1
in. - Tik dan
f 3
in. - OCaml antwoordt met
- : int = 4
. De opdrachtlet f x = x + 1
definieert de functief
met parameterx
en lichaamx + 1
. De uitdrukkingf 3
staat voor de toepassing van de functief
op het argument3
. De waarde van zo'n functietoepassing is de waarde van het lichaam van de functie, waarbij de parameter gebonden wordt aan het opgegeven argument. Dus, de waarde vanf 3
is de waarde vanx + 1
waarbijx
gebonden wordt aan3
.
Oefening:
- Definieer een functie
begroeting
zodanig datbegroeting "Jan"
evalueert tot"Hallo, Jan!"
enbegroeting "Piet"
evalueert tot"Hallo, Piet!"
.
Oefening:
- Definieer een functie
positief
zodanig datpositief 1
enpositief 10
evalueren tottrue
enpositief 0
enpositief (-7)
totfalse
.
Opdracht:
- Tik
let dubbel x = x + x
in. - Tik dan
dubbel 5
in. - OCaml antwoordt met
- : int = 10
.
Opdracht:
- Tik
let viervoud y = dubbel y + dubbel y
in. - Tik dan
viervoud 7
in. - OCaml antwoordt met
- : int = 28
. Je kan een functie gebruiken in het lichaam van een andere functie.
Oefening:
- Definieer de functie
citeer
, zodanig datciteer "Hallo"
evalueert tot"'Hallo'"
.
Opdracht:
- Tik
let spreuk x = citeer x ^ ", dat is de vraag."
in. - Tik dan
spreuk "Zijn of niet zijn?"
in. - OCaml antwoordt met
- : string = "'Zijn of niet zijn?', dat is de vraag."
.
Opdracht:
- Tik
let f x y = x + y
in. - Tik dan
f 10 30
in. - OCaml antwoordt met
- : int = 40
. Functies kunnen meerdere parameters hebben. Je scheidt ze gewoon met een spatie. - Tik dan
f 100 (f 10 30)
in. - OCaml antwoordt met
- : int = 140
. Je moet haakjes gebruiken als je een complexe uitdrukking wilt gebruiken als argument voor een functie-oproep.
Opdracht:
- Tik
let volledige_naam voornaam achternaam = voornaam ^ " " ^ achternaam
in. - Tik dan
volledige_naam "Jan" "Janssens"
in. - OCaml antwoordt met
"Jan Janssens"
. - Tik dan
volledige_naam "Piet" "Pieters"
in. - OCaml antwoordt met
"Piet Pieters"
.
Opdracht:
- Tik
let adres naam straat nummer postcode gemeente = naam ^ ", " ^ straat ^ " " ^ nummer ^ ", " ^ postcode ^ " " ^ gemeente
in. - Tik dan
let adres2 voornaam achternaam straat nummer postcode gemeente = adres (volledige_naam voornaam achternaam) straat nummer postcode gemeente
in. - Tik dan
adres2 "Jan" "Janssens" "Dorpsstraat" "10" "1000" "Brussel"
in. - OCaml antwoordt met
"Jan Janssens, Dorpsstraat 10, 1000 Brussel"
.
Oefening:
- Definieer een functie
gemiddelde
zodanig datgemiddelde 10 20
evalueert tot15
engemiddelde 3 7
evalueert tot5
.
Oefening:
- Definieer een functie
tussen
zodanig dattussen x y z
evalueert tottrue
als en slechts alsy
tussenx
enz
ligt.
Opdracht:
- Tik
let rec macht x n = if n = 0 then 1 else x * macht x (n - 1)
in. - Tik dan
macht 2 3
in. - OCaml antwoordt met
- : int = 8
. Je kan een functie toepassen in haar eigen lichaam, op voorwaarde dat je delet rec
-opdracht gebruikt in plaats van delet
-opdracht. Zo'n functie noemen we een recursieve functie.
Je kan de evaluatie van deze functie begrijpen als volgt:
macht 2 3
= if 3 = 0 then 1 else 2 * macht 2 (3 - 1) (definitie van macht)
= 2 * macht 2 2 (vereenvoudiging)
= 2 * (if 2 = 0 then 1 else 2 * macht 2 (2 - 1)) (definitie van macht)
= 2 * (2 * macht 2 1) (vereenvoudiging)
= 2 * (2 * (if 1 = 0 then 1 else 2 * macht 2 (1 - 1))) (definitie van macht)
= 2 * (2 * (2 * macht 2 0)) (vereenvoudiging)
= 2 * (2 * (2 * (if 0 = 0 then 1 else 2 * macht 2 (0 - 1)))) (definitie van macht)
= 2 * (2 * (2 * 1)) (vereenvoudiging)
= 8 (vereenvoudiging)
Opdracht:
- Tik
let sterren_0 = ""
in. - OCaml antwoordt met
sterren_0 : string = ""
. - Tik
let sterren_1 = "*" ^ sterren_0
in. - OCaml antwoordt met
sterren_1 : string = "*"
. - Tik
let sterren_2 = "*" ^ sterren_1
in. - OCaml antwoordt met
sterren_2 : string = "**"
. - Tik
let sterren_3 = "*" ^ sterren_2
in. - OCaml antwoordt met
sterren_3 : string = "***"
.
Oefening:
- Definieer de functie
sterren
zodanig datsterren 3
evalueert tot"***"
ensterren 5
evalueert tot"*****"
.
Opdracht:
- Tik
[10; 20; 30; 40]
in. - OCaml antwoordt met
- : int list = [10; 20; 30; 40]
. Het typeint list
is het type van lijsten van gehele getallen.
Opdracht:
- Tik
10::20::30::40::[]
in. - OCaml antwoordt met
- : int list = [10; 20; 30; 40]
. De uitdrukkinge::l
betekent: de lijst bestaande uit het elemente
gevolgd door de lijstl
. Het eerste element van een (niet-lege) lijst noemen we het hoofd en de elementen ná het eerste element de staart. De uitdrukkinge::l
betekent dus: de lijst met hoofde
en staartl
.
Opdracht:
- Tik
let aftelling_0 = [0]
in. - OCaml antwoordt met
aftelling_0 : int list = [0]
. - Tik
let aftelling_1 = 1::aftelling_0
in. - OCaml antwoordt met
aftelling_1 : int list = [1; 0]
. - Tik
let aftelling_2 = 2::aftelling_1
in. - OCaml antwoordt met
aftelling_2 : int list = [2; 1; 0]
. - Tik
let aftelling_3 = 3::aftelling_2
in. - OCaml antwoordt met
aftelling_3 : int list = [3; 2; 1; 0]
.
Oefening:
- Definieer een functie
aftelling
zodanig dataftelling 2
evalueert tot[2; 1; 0]
enaftelling 4
evalueert tot[4; 3; 2; 1; 0]
.
Opdracht:
- Tik
let leeg xs = match xs with [] -> true | hoofd::staart -> false
in. - Tik dan
leeg []
in. - OCaml antwoordt met
- : bool = true
. - Tik dan
leeg [10]
in. - OCaml antwoordt met
- : bool = false
. Je kan eenmatch
-uitdrukking gebruiken om verschillende berekeningen te doen afhankelijk van de vorm van een lijst. Meer bepaald: de uitdrukkingmatch xs with [] -> U1 | hoofd::staart -> U2
evalueert tot de waarde vanU1
alsxs
een lege lijst is, en tot de waarde vanU2
, waarbijhoofd
gebonden wordt aan het hoofd (eerste element) vanxs
enstaart
aan de staart vanxs
(de lijst die je krijgt door het eerste element vanxs
weg te laten), alsxs
niet leeg is.
Opdracht:
- Tik
let som_eerste_twee xs = match xs with eerste::tweede::_ -> eerste + tweede
in. - Tik dan
som_eerste_twee [10; 20; 30]
in. - OCaml antwoordt met
- : int = 30
. - Tik dan
som_eerste_twee [10]
in. - OCaml antwoordt met
Exception: Match_failure
. Als je in eenmatch
-uitdrukking niet alle gevallen behandelt, krijg je een waarschuwing. Als je dan de uitdrukking evalueert voor een geval dat niet behandeld wordt, krijg je een uitzondering (exception in het Engels).
Opdracht:
- Tik
let hoofd_van xs = match xs with hoofd::staart -> hoofd
in. - Tik dan
hoofd_van [1;2;3]
in. - OCaml antwoordt met
- : int = 1
. - Tik dan
hoofd_van [10;20]
in. - OCaml antwoordt met
- : int = 10
.
Opdracht:
- Tik
let staart_van xs = match xs with hoofd::staart -> staart
in. - Tik dan
staart_van [1;2;3]
in. - OCaml antwoordt met
- : int list = [2;3]
. - Tik dan
staart_van [10;20]
in. - OCaml antwoordt met
- : int list = [20]
.
Oefening:
- Definieer een functie
lengte
zodanig datlengte [true; false]
evalueert tot2
enlengte ["Hallo"]
evalueert tot1
enlengte [1.0; 2.0; 3.0]
evalueert tot3
.
Merk op:
lengte ["a"; "b"; "c"]
= lengte ("a"::"b"::"c"::[])
= lengte ("a"::("b"::("c"::[])))
= 1 + lengte ("b"::("c"::[]))
= 1 + (1 + lengte ("c"::[]))
= 1 + (1 + (1 + lengte []))
= 1 + (1 + (1 + 0))
= 3
Oefening:
- Definieer een functie
som
zodanig datsom [1; 2; 3]
evalueert tot6
ensom [7; 3; 11]
evalueert tot21
.
Oefening:
- Definieer een functie
filter_pos
zodanig datfilter_pos [10; -3; 5]
evalueert tot[10; 5]
enfilter_pos [-7; 0; 8]
evalueert tot[8]
.
Oefening:
- Definieer een functie
negatie
zodanig datnegatie [10; -20]
evalueert tot[-10; 20]
.
Oefening:
- Definieer een functie
aan_elkaar
zodanig dataan_elkaar [1; 2] [3; 4]
evalueert tot[1; 2; 3; 4]
.
Oefening:
- Definieer een functie
alles_aan_elkaar
zodanig datalles_aan_elkaar [[10; 20]; [30; 40]; [50]]
evalueert tot[10; 20; 30; 40; 50]
.
Oefening:
- Definieer een functie
voeg_in
die, gegeven een geheel getal en een gesorteerde lijst van gehele getallen, de gesorteerde lijst berekent die je bekomt als je het element op de juiste plaats invoegt in de gegeven gesorteerde lijst.
Oefening:
- Definieer een functie
sorteer
zodanig datsorteer [3; 1; 2]
evalueert tot[1; 2; 3]
. Hint: gebruik de functievoeg_in
.
Merk op dat de functie sorteer
een kwadratische uitvoeringstijd heeft: het aantal evaluaties van voeg_in
gegenereerd door een evaluatie van sorteer xs
, waarbij xs
een lijst van lengte N is, is in het slechtste geval N + (N - 1) + ... + 1 = N*(N+1)/2. Dit betekent dat de uitvoeringstijd sterk toeneemt naarmate de lengte van de te sorteren lijst toeneemt. Later zullen we efficiëntere sorteeralgoritmes zien.
Opdracht:
- Tik
("Jan", 14)
in. - OCaml antwoordt met
- : string * int = ("Jan", 14)
. Dit is een tuple (een geordend n-tal) bestaande uit een stuk tekst en een geheel getal. - Tik
let (naam, leeftijd) = ("Jan", 14)
in. - OCaml antwoordt met
naam : string = "Jan"
enleeftijd : int = 14
.
Opdracht:
- Tik
let eerste paar = let (x, y) = paar in x
in. - Tik
eerste (1, 2)
in. - OCaml antwoordt met
1
.
Oefening:
- Definieer een functie
splits
zodanig datsplits [1;2;3;4;5]
evalueert tot([1;3;5], [2;4])
, ensplits [2;3;4;5;6]
evalueert tot([2;4;6], [3;5])
.
Oefening:
- Definieer een functie
merge
(Engels voor samenvoegen) die, gegeven twee gesorteerde lijsten, de gecombineerde gesorteerde lijst berekent. Bijvoorbeeld:merge [2; 4; 6] [1; 3; 5] = [1; 2; 3; 4; 5; 6]
. Zorg ervoor dat deze functie een lineaire uitvoeringstijd heeft; m.a.w., zorg ervoor dat het aantal evaluatiestappen een lineaire functie is van de lengte van de invoerlijsten. In het bijzonder mag je niet de eerder gedefinieerde functiesorteer
gebruiken!
Oefening:
- Definieer een functie
merge_sort
die een gegeven lijst sorteert door ze eerst te splitsen in twee ongeveer even grote deel-lijsten, door middel van de functiesplits
, dan beide deel-lijsten te sorteren door middel van een recursieve oproep vanmerge_sort
, en dan de resulterende gesorteerde lijsten samen te voegen metmerge
.
De uitvoeringstijd van de functie merge_sort
is O(N*log(N)). Inderdaad: als M(N) het aantal evaluaties van merge
is, gegenereerd door een evaluatie van merge_sort xs
, waarbij xs
een lijst van lengte N is, dan bestaat er een constante C zodanig dat voor alle voldoende grote N geldt dat M(N) <= C * N * log(N). Dit is een betere asymptotische uitvoeringstijd dan het sorteeralgoritme met invoeging dat we hierboven zagen.
Oefening:
- Definieer een functie
zoek_op
, zodanig datzoek_op "Jan" [("Piet", 14); ("Jan", 16)]
evalueert tot16
.
Oefening:
- Definieer een functie
verwijder_associatie
, zodanig datverwijder_associatie "Jan" [("Piet", 14); ("Jan", 16)]
evalueert tot[("Piet", 14)]
.
Opdracht:
- Tik
[| 1; 2; 3; 4 |]
in. - OCaml antwoordt met
- : int array = [| 1; 2; 3; 4 |]
. - Tik
let xs = [| 1; 2; 3; 4 |]
in. - Tik dan
xs.(0)
in. - OCaml antwoordt met
- : int = 1
. Arrays zijn zoals lijsten, behalve dat sommige operaties een andere performantie hebben. Het ophalen van het N-de element gebeurt bij arrays in constante tijd, terwijl de tijd die hiervoor nodig is bij lijsten lineair is in het aantal elementen van de lijst. - Tik dan
Array.length xs
in. - OCaml antwoordt met
- : int = 4
.
Oefening:
- Definieer de functie
positie_van
, die, gegeven een getal en een gesorteerde array, de positie van dit getal in de array teruggeeft, ofwel de positie waar dit getal ingevoegd mag worden zodat de array gesorteerd blijft. Bijvoorbeeld:positie_van 4 [| 3; 4; 5 |]
evalueert tot1
, enpositie_van 5 [| 2; 4; 6 |]
evalueert tot2
. Definieer de functie zó dat ze slechts een aantal opzoekingen doet van elementen in de array dat logaritmisch is in de lengte van de array. Dus: in een array van lengte 32 mag je slechts 5 opzoekingen doen.
Opdracht:
- Tik
let xs = [| 0 |]
in. - Tik dan
xs.(0) <- 3
in. - Tik dan
xs.(0)
in. - OCaml antwoordt met
- : int = 3
.
Oefening:
- Definieer een functie
sorteer
die, gegeven een array, de elementen van deze array verplaatst zodat de array gesorteerd wordt.