Haskell
See artikkel ootab keeletoimetamist. (Veebruar 2018) |
Haskell on standardiseeritud üldotstarbeline puhtalt funktsionaalne programmeerimiskeel. Keel on nimetatud loogiku Haskell Curry järgi.[1]
Haskell | |
---|---|
Faililaiend | .hs, .lhs |
Paradigma | funktsionaalne, mitterange, modulaarne |
Väljalaskeaeg | 1990 |
Looja | Simon Peyton Jones, Paul Hudak, Philip Wadler ja teised |
Viimane väljalase | 5.10.1 (22. august 2009) |
Tüüpimine | staatiline, tugev, tuletatud |
Implementatsioonid | GHC, Hugs, NHC, JHC, Yhc |
Dialektid | Helium, Gofer |
Mõjutatud keeltest | Lisp ja Scheme, ISWIM, FP, APL, Hope ja Hope+, SISAL, Miranda, ML ja Standard ML, Lazy ML, Orwell, Alfl, Id, Ponder |
Mõjutanud keeli | Agda, Bluespec, Clojure, C#, CAL, Cat, Cayenne, Clean, Curry, Epigram, Escher, F#, Factor, Isabelle, Java Generics, LINQ, Mercury, Omega, Perl 6, Python, Qi, Scala, Timber, Visual Basic 9.0 |
OS | mitmeplatvormiline |
Veebileht | haskell.org |
Haskell toetab laiska väärtustamist, mustrisobitust, loendite hõlmamist, tüübiklasse ja tüüpide polümorfismi. Kuna tegemist on puhta funktsionaalse keelega, siis puuduvad Haskelli funktsioonidel kõrvalefektid. Viimaste esitamiseks on eraldi andmetüüp: monaad.[2]
Süntaksist üldiselt
muudaKommentaari algust tähistavad Haskellis kaks sidekriipsu (--). Näiteks
-– Siin on kommentaar
Kommentaar võib olla eraldi real või rea lõpus.
Sarnaselt Pythonile eraldatakse Haskellis koodiplokke läbi taandamise. Näiteks funktsiooni keha ei tohi olla funktsiooni nimega samal kaugusel ekraani vasakust servast. Kui taanded pole korralikult vormistatud, siis kuvatakse vastavasisuline veateade ning programmi ei saa käivitada.
Laisk ja agar väärtustamine
muudaHaskellis väärtustatakse kõik avaldised laisalt, mis tähendab, et avaldised väärtustatakse ainult siis, kui nende väärtus on tingimata käesoleval arvutuse sammul oluline. Kui aga avaldise väärtus pole hetkel oluline, siis jäetakse väärtustamata avaldis mällu ja vajadusel väärtustatakse hiljem. Haskellis on olemas ka operaator $! (dollarimärk ja hüüumärk), mis sunnib agaralt väärtustama. Näiteks deklaratsioonis
eelnevateSumma n = ($!) eelnevateSumma (n – 1) * n
operaator $! sunnib väärtustama avaldist n – 1.
Hello world
muudaJärgneb Haskellis kirjutatud programm "Hello world" (kõik read peale viimase võib ära jätta):
module Main where main :: IO () main = putStrLn "Hello world!"
module Main where
on programmi päis. See sisaldab võtmesõna module
, protseduuri nime Main
ja võtmesõna where
. Rida main :: IO ()
määrab funktsiooni main
tüübi, mis antud juhul on tüübile void vastav protseduuritüüp (IO
määrab selle, et tegemist on protseduuritüübiga). Käsk putStrLn
on lühend väljendist "put String Line", mis sisuliselt tähendab, et väljastatakse sõne, millele järgneb reavahetus. Sarnane käsk on putStr
, kuid sel juhul jäetakse lõpust ära reavahetus.
Loendid
muudaHaskellis on olemas loendid ehk "listid". Tühja loendit tähistab []
(nurksulud). Loendi võib moodustada näiteks sisestades [1, 2, 3, 4]
, mille tulemusel luuakse loend elementidega 1, 2, 3 ja 4. Sama loendi moodustamiseks võib kasutada ka konstruktsiooni 1 : 2 : 3 : 4 : []
, kus elemendid on ühendatud koolonitega ja viimane on tühja loendi tähis, mis märgib listi lõppu. Seda loendit saab moodustada ka aritmeetilise jada abil: [1 .. 4], kus 1 tähistab esimest elementi ja 4 viimast elementi ning jada samm on vaikimisi 1.
Sõnesid käsitletakse Haskellis ka kui tähemärkide loendeid. Näiteks "Tere, maailm!" on loend, mille elementideks on selles sõnes sisalduvad sümbolid.
Loendite hõlmamine
muudaHaskell toetab ka loendite hõlmamist (ingl. list comprehension). Näiteks
[x ^ 2 | x <- [0, 0.5 .. 3]]
Selle tulemusel väljastatakse loend, mille elementideks on arvude 0, 0.5, 1, 1.5, 2, 2.5 ja 3 ruudud. x ^ 2
on avaldis ja x <- [0, 0.5 .. 3]
on generaator. Generaatori parem pool peab alati olema loenditüüpi – näites sisuliselt loend [0, 0.5, 1, 1.5, 2, 2.5, 3].
Mustrisobitus
muudaMuster esitab skeemi, millega ette antud avaldist võrreldakse.[3] Muster esitab vajalikku ehitust. Näiteks, kui oodatav väärtus, mis ette antakse, on mittetühi loend, siis saab näidisega x : xs kontrollida, et etteantavas loendis oleks vähemalt üks element. Kusjuures x tähistab loendi pead ehk esimest elementi ning xs tähistab loendi saba ehk ülejäänud elemente alates teisest elemendist. Näiteks loendi [5] korral, mis on üheelemendiline, on x väärtuseks 5 ja xs väärtuseks [].
Kohane näidis ilmestamaks loendi ehituse kontrolli oleks järgnev:
ymberPoord xs = let loppu x : xs -- 1) = loppu xs ++ [x] loppu _ -- 2) = [] in loppu xs
Protseduur ymberPoord
saab argumendina ette loendi. Kutsutakse välja loppu
argumendiks saadud loendil. Kuna on loppu
on defineeritud kaks korda (1) ja 2)), siis valitakse näidiste sobitamise teel õige. Esimene näidis 1) sobitub mittetühjade loenditega. Teine näidis 2) on ainult alakriips ehk jokker, millega sobituvad kõik ette antavad loendid. Näiteks tühja loendi korral sobitamine näidisega 1) ebaõnnestub, järelikult täidetakse kood, mis järgneb näidisele 2).
Näidise 1) sobitumisel kutsutakse rekursiivselt välja loppu
argumendiks olnud loendi sabale, kusjuures argumendiks olnud loendi pea eraldatakse ja lisatakse lõppu. Rekursiivsed pöördumised toimuvad, kuni argumendiks antakse mittetühi loend. Tühja loendi korral väljastatakse algne (protseduurile ymberPoord argumendina ette antud) loend.
Rekursioon
muudaKuna Haskellis puuduvad tsüklid, siis nende asemel tuleb kasutada rekursiooni, sageli koos mustrisobitusega.
Ebaõnnestunud arvutuse märgiks on nn bottom, mida tähistatakse tagurpidi T-tähega (⊥). Bottom tekib programmi täitmisaegse vea või lõpmatu arvutuse korral. Kuna see võib juhtuda mistahes funktsiooni puhul, on bottom osa igast tüübist.[4]
Avaldiste liigid
muudaLet-avaldis
muudaLet avaldis võimaldab defineerida hulga lisamuutujaid protseduuri sees. Näiteks
tehe x = let ruut = x * x veerand = x / 4 in ruut + veerand
See funktsioon tagastab etteantud argumendi veerandi ja ruudu summa. Let-avaldise puhul on olulised võtmesõnad let ja in ning pärast võtmesõna in peab järgnema avaldis.
Lambdaavaldis
muudaLambdaavaldis on näiteks järgnev:
viieKordne = \ x -> 5 * x
See võtab argumendiks arvu ning tagastab selle arvu viiekordsena. Näiteks, kui argumendiks on 6, tagastab see lambdaavaldis arvu 30. Interaktiivses keskkonnas jõuab sama tulemuseni sisestades
(\ x -> 5 * x) 6
Lambdaavaldist alustav kaldkriips sarnaneb kreeka tähestiku tähe lambdaga – sellest tõenäoliselt ka nimi "lambdaavaldis".
Valikuavaldis
muudaValikuavaldis võrdleb argumenti suurema hulga etteantud näidistega ning neist sobiva leidmisel täidab selle näidise kohta käivad käsud ning järgmiseid näidiseid enam ei võrdle.
faktoriaal n = case compare n 0 of GT -> ($!) faktoriaal (n - 1) * n EQ -> 1 _ -> error "Argument peab olema mittenegatiivne!"
Olulised võtmesõnad on case ja of. Argumendiks antud arvu võrreldakse nulliga (compare 0 n
). Kui n on nullist suurem, siis kutsutakse rekursiivselt sama funktsiooni uuesti välja ühe võrra väiksemal argumendil. Kui n on nulliga võrdne, siis tagastatakse arv 1. Kui argument on negatiivne, siis kuvatakse veateade ("Argument peab olema mittenegatiivne!").
Sisend ja väljund
muudaHaskellis on ka vahendid standardväljundist lugemiseks ning sinna kirjutamiseks. Illustreerimiseks toome järgneva näite:
dialoog = do putStr "Sisesta palun oma nimi: " nimi <- getLine putStrLn ("Tere, " ++ nimi ++ ". Meeldiv tutvuda!")
See funktsioon küsib kasutajalt nime ning tervitab teda seejärel viisakalt nimepidi. Kasutatud on selle funktsiooni koostamisel do-süntaksit, mis võimaldab kirjeldada mitmeid tegevusi järjest.
Viited
muuda- ↑ Haskell Wiki: Introduction
- ↑ Haskell Wiki: Monad
- ↑ "Funktsionaalprogrammeerimine - Kursused - Arvutiteaduse instituut". courses.cs.ut.ee. Vaadatud 4. jaanuaril 2024.
- ↑ "Bottom - HaskellWiki". wiki.haskell.org. 16. juuni 2014. Originaali arhiivikoopia seisuga 29. august 2023. Vaadatud 4. jaanuaril 2024.