Haskell - Funzioni

Le funzioni giocano un ruolo importante in Haskell, poiché è un linguaggio di programmazione funzionale. Come altre lingue, Haskell ha una propria definizione e dichiarazione funzionale.

  • La dichiarazione di funzione consiste nel nome della funzione e nel suo elenco di argomenti insieme al suo output.

  • La definizione della funzione è dove si definisce effettivamente una funzione.

Facciamo un piccolo esempio di add funzione per comprendere questo concetto in dettaglio.

add :: Integer -> Integer -> Integer   --function declaration 
add x y =  x + y                       --function definition 

main = do 
   putStrLn "The addition of the two numbers is:"  
   print(add 2 5)    --calling a function

Qui, abbiamo dichiarato la nostra funzione nella prima riga e nella seconda riga, abbiamo scritto la nostra funzione effettiva che prenderà due argomenti e produrrà un output di tipo intero.

Come la maggior parte degli altri linguaggi, Haskell inizia a compilare il codice dal mainmetodo. Il nostro codice genererà il seguente output:

The addition of the two numbers is:
7

Pattern Matching

Pattern Matching è il processo di corrispondenza di un tipo specifico di espressioni. Non è altro che una tecnica per semplificare il tuo codice. Questa tecnica può essere implementata in qualsiasi tipo di classe Type. If-Else può essere utilizzato come opzione alternativa di corrispondenza del modello.

Il Pattern Matching può essere considerato come una variante del polimorfismo dinamico in cui in fase di esecuzione possono essere eseguiti diversi metodi a seconda del loro elenco di argomenti.

Dai un'occhiata al seguente blocco di codice. Qui abbiamo utilizzato la tecnica del Pattern Matching per calcolare il fattoriale di un numero.

fact :: Int -> Int 
fact 0 = 1 
fact n = n * fact ( n - 1 ) 

main = do 
   putStrLn "The factorial of 5 is:" 
   print (fact 5)

Sappiamo tutti come calcolare il fattoriale di un numero. Il compilatore inizierà a cercare una funzione chiamata "fact" con un argomento. Se l'argomento non è uguale a 0, il numero continuerà a chiamare la stessa funzione con 1 inferiore a quello dell'argomento effettivo.

Quando il pattern dell'argomento corrisponde esattamente a 0, chiamerà il nostro pattern che è "fact 0 = 1". Il nostro codice produrrà il seguente output:

The factorial of 5 is:
120

Guardie ✔

Guardsè un concetto molto simile al pattern matching. Nella corrispondenza dei modelli, di solito abbiniamo una o più espressioni, ma usiamoguards per testare alcune proprietà di un'espressione.

Sebbene sia consigliabile utilizzare il pattern matching su guards, ma dal punto di vista di uno sviluppatore, guardsè più leggibile e semplice. Per gli utenti alle prime armi,guards possono sembrare molto simili alle istruzioni If-Else, ma sono funzionalmente differenti.

Nel codice seguente abbiamo modificato il nostro factorial programma utilizzando il concetto di guards.

fact :: Integer -> Integer 
fact n | n == 0 = 1 
       | n /= 0 = n * fact (n-1) 
main = do 
   putStrLn "The factorial of 5 is:"  
   print (fact 5)

Qui ne abbiamo dichiarati due guards, separati da "|" e chiamando ilfact funzione da main. Internamente, il compilatore funzionerà nello stesso modo del caso del pattern matching per produrre il seguente output:

The factorial of 5 is:
120

Dove la clausola

Whereè una parola chiave o una funzione incorporata che può essere utilizzata in fase di esecuzione per generare l'output desiderato. Può essere molto utile quando il calcolo della funzione diventa complesso.

Considera uno scenario in cui il tuo input è un'espressione complessa con più parametri. In questi casi, puoi suddividere l'intera espressione in piccole parti utilizzando la clausola "dove".

Nell'esempio seguente, stiamo prendendo un'espressione matematica complessa. Mostreremo come trovare le radici di un'equazione polinomiale [x ^ 2 - 8x + 6] usando Haskell.

roots :: (Float, Float, Float) -> (Float, Float)  
roots (a,b,c) = (x1, x2) where 
   x1 = e + sqrt d / (2 * a) 
   x2 = e - sqrt d / (2 * a) 
   d = b * b - 4 * a * c  
   e = - b / (2 * a)  
main = do 
   putStrLn "The roots of our Polynomial equation are:" 
   print (roots(1,-8,6))

Notare la complessità della nostra espressione per calcolare le radici di una data funzione polinomiale. È abbastanza complesso. Quindi, stiamo rompendo l'espressione usando ilwhereclausola. Il pezzo di codice sopra genererà il seguente output:

The roots of our Polynomial equation are:
(7.1622777,0.8377223)

Funzione di ricorsione

La ricorsione è una situazione in cui una funzione chiama se stessa ripetutamente. Haskell non fornisce alcuna funzionalità per eseguire il loop di qualsiasi espressione per più di una volta. Invece, Haskell vuole che tu suddivida l'intera funzionalità in una raccolta di funzioni diverse e utilizzi la tecnica di ricorsione per implementare la tua funzionalità.

Consideriamo nuovamente il nostro esempio di pattern matching, in cui abbiamo calcolato il fattoriale di un numero. Trovare il fattoriale di un numero è un classico caso di utilizzo della ricorsione. Qui potresti: "In che modo la corrispondenza del pattern è diversa dalla ricorsione?" La differenza tra questi due sta nel modo in cui vengono utilizzati: il pattern matching lavora sulla configurazione del vincolo terminale, mentre la ricorsione è una chiamata di funzione.

Nell'esempio seguente, abbiamo utilizzato sia il pattern matching che la ricorsione per calcolare il fattoriale di 5.

fact :: Int -> Int 
fact 0 = 1 
fact n = n * fact ( n - 1 ) 

main = do 
   putStrLn "The factorial of 5 is:" 
   print (fact 5)

Produrrà il seguente output:

The factorial of 5 is:
120

Funzione di ordine superiore

Fino ad ora, quello che abbiamo visto è che le funzioni Haskell ne prendono una type come input e produrne un altro typecome output, che è abbastanza simile in altri linguaggi imperativi. Le funzioni di ordine superiore sono una caratteristica unica di Haskell in cui è possibile utilizzare una funzione come argomento di input o di output.

Sebbene sia un concetto virtuale, ma nei programmi del mondo reale, ogni funzione che definiamo in Haskell utilizza un meccanismo di ordine superiore per fornire l'output. Se hai la possibilità di esaminare la funzione di libreria di Haskell, scoprirai che la maggior parte delle funzioni di libreria sono state scritte in un modo di ordine superiore.

Facciamo un esempio in cui importeremo una mappa di funzioni di ordine superiore incorporata e utilizzeremo la stessa per implementare un'altra funzione di ordine superiore in base alla nostra scelta.

import Data.Char  
import Prelude hiding (map) 

map :: (a -> b) -> [a] -> [b] 
map _ [] = [] 
map func (x : abc) = func x : map func abc  
main = print $ map toUpper "tutorialspoint.com"

Nell'esempio sopra, abbiamo usato il toUpper funzione della classe di tipo Charper convertire il nostro input in maiuscolo. Qui, il metodo "map" accetta una funzione come argomento e restituisce l'output richiesto. Ecco il suo output:

sh-4.3$ ghc -O2 --make *.hs -o main -threaded -rtsopts
sh-4.3$ main
"TUTORIALSPOINT.COM"

Espressione lambda

A volte dobbiamo scrivere una funzione che verrà utilizzata una sola volta, per l'intera durata di un'applicazione. Per affrontare questo tipo di situazioni, gli sviluppatori Haskell utilizzano un altro blocco anonimo noto comelambda expression o lambda function.

Una funzione senza avere una definizione è chiamata funzione lambda. Una funzione lambda è indicata dal carattere "\". Prendiamo il seguente esempio in cui aumenteremo il valore di input di 1 senza creare alcuna funzione.

main = do 
   putStrLn "The successor of 4 is:"  
   print ((\x -> x + 1) 4)

Qui abbiamo creato una funzione anonima che non ha un nome. Prende l'intero 4 come argomento e stampa il valore di output. Fondamentalmente stiamo operando una funzione senza nemmeno dichiararla correttamente. Questa è la bellezza delle espressioni lambda.

La nostra espressione lambda produrrà il seguente output:

sh-4.3$ main
The successor of 4 is:
5