Haskell - Fungsi

Fungsi memainkan peran utama di Haskell, karena ini adalah bahasa pemrograman fungsional. Seperti bahasa lain, Haskell memang memiliki definisi dan deklarasi fungsionalnya sendiri.

  • Deklarasi fungsi terdiri dari nama fungsi dan daftar argumennya beserta keluarannya.

  • Definisi fungsi adalah tempat Anda sebenarnya mendefinisikan suatu fungsi.

Mari kita ambil contoh kecil add berfungsi untuk memahami konsep ini secara detail.

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

Di sini, kami telah mendeklarasikan fungsi kami di baris pertama dan di baris kedua, kami telah menulis fungsi aktual kami yang akan mengambil dua argumen dan menghasilkan satu keluaran tipe integer.

Seperti kebanyakan bahasa lain, Haskell mulai mengompilasi kode dari mainmetode. Kode kami akan menghasilkan keluaran berikut -

The addition of the two numbers is:
7

Pencocokan Pola

Pencocokan Pola adalah proses pencocokan jenis ekspresi tertentu. Ini tidak lain adalah teknik untuk menyederhanakan kode Anda. Teknik ini dapat diimplementasikan ke dalam semua tipe kelas Type. If-Else dapat digunakan sebagai opsi alternatif pencocokan pola.

Pencocokan Pola dapat dianggap sebagai varian dari polimorfisme dinamis di mana pada waktu proses, metode yang berbeda dapat dieksekusi bergantung pada daftar argumennya.

Lihatlah blok kode berikut. Di sini kita telah menggunakan teknik Pencocokan Pola untuk menghitung faktorial sebuah bilangan.

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

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

Kita semua tahu cara menghitung faktorial sebuah bilangan. Kompilator akan mulai mencari fungsi yang disebut "fakta" dengan sebuah argumen. Jika argumen tidak sama dengan 0, maka angka tersebut akan terus memanggil fungsi yang sama dengan 1 lebih kecil dari argumen sebenarnya.

Ketika pola argumen sama persis dengan 0, itu akan memanggil pola kita yang "fakta 0 = 1". Kode kami akan menghasilkan keluaran berikut -

The factorial of 5 is:
120

Penjaga

Guardsadalah konsep yang sangat mirip dengan pencocokan pola. Dalam pencocokan pola, kami biasanya mencocokkan satu atau lebih ekspresi, tetapi kami menggunakanguards untuk menguji beberapa properti ekspresi.

Meskipun disarankan untuk menggunakan pencocokan pola guards, tetapi dari perspektif pengembang, guardslebih mudah dibaca dan sederhana. Untuk pengguna pertama kali,guards dapat terlihat sangat mirip dengan pernyataan If-Else, tetapi secara fungsional berbeda.

Dalam kode berikut, kami telah memodifikasi factorial program dengan menggunakan konsep 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)

Di sini, kami telah mendeklarasikan dua guards, dipisahkan oleh "|" dan memanggilfact fungsi dari main. Secara internal, kompilator akan bekerja dengan cara yang sama seperti dalam kasus pencocokan pola untuk menghasilkan keluaran berikut -

The factorial of 5 is:
120

Dimana Klausul

Whereadalah kata kunci atau fungsi bawaan yang dapat digunakan saat runtime untuk menghasilkan keluaran yang diinginkan. Ini bisa sangat membantu ketika kalkulasi fungsi menjadi kompleks.

Pertimbangkan skenario di mana masukan Anda adalah ekspresi kompleks dengan banyak parameter. Dalam kasus seperti itu, Anda dapat memecah seluruh ekspresi menjadi bagian-bagian kecil menggunakan klausa "di mana".

Dalam contoh berikut, kami mengambil ekspresi matematika yang kompleks. Kami akan menunjukkan bagaimana Anda bisa mencari akar dari persamaan polinomial [x ^ 2 - 8x + 6] menggunakan 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))

Perhatikan kompleksitas ekspresi kita untuk menghitung akar dari fungsi polinomial yang diberikan. Ini cukup rumit. Karenanya, kami memecahkan ekspresi menggunakanwhereayat. Potongan kode di atas akan menghasilkan keluaran berikut -

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

Fungsi Rekursi

Rekursi adalah situasi di mana suatu fungsi memanggil dirinya sendiri berulang kali. Haskell tidak menyediakan fasilitas untuk mengulang ekspresi apapun lebih dari sekali. Sebaliknya, Haskell ingin Anda memecah seluruh fungsionalitas Anda menjadi sekumpulan fungsi yang berbeda dan menggunakan teknik rekursi untuk mengimplementasikan fungsionalitas Anda.

Mari kita perhatikan contoh pencocokan pola kita lagi, di mana kita telah menghitung faktorial sebuah bilangan. Menemukan faktorial sebuah bilangan adalah kasus klasik penggunaan Rekursi. Di sini, Anda mungkin, "Apa perbedaan pencocokan pola dari rekursi?” Perbedaan antara keduanya terletak pada cara penggunaannya. Pencocokan pola berfungsi untuk menyiapkan batasan terminal, sedangkan rekursi adalah pemanggilan fungsi.

Dalam contoh berikut, kami telah menggunakan pencocokan pola dan rekursi untuk menghitung faktorial dari 5.

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

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

Ini akan menghasilkan keluaran sebagai berikut -

The factorial of 5 is:
120

Fungsi Orde Tinggi

Sampai sekarang, apa yang kita lihat adalah fungsi Haskell mengambil satu type sebagai masukan dan menghasilkan yang lain typesebagai keluaran, yang sangat mirip dalam bahasa imperatif lainnya. Fungsi Urutan Tinggi adalah fitur unik Haskell di mana Anda dapat menggunakan fungsi sebagai argumen input atau output.

Meskipun merupakan konsep virtual, namun dalam program dunia nyata, setiap fungsi yang kami definisikan di Haskell menggunakan mekanisme tingkat tinggi untuk memberikan keluaran. Jika Anda mendapat kesempatan untuk melihat ke dalam fungsi perpustakaan Haskell, maka Anda akan menemukan bahwa sebagian besar fungsi perpustakaan telah ditulis dengan urutan yang lebih tinggi.

Mari kita ambil contoh di mana kita akan mengimpor peta fungsi orde tinggi inbuilt dan menggunakan yang sama untuk mengimplementasikan fungsi orde tinggi lainnya sesuai pilihan kita.

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"

Dalam contoh di atas, kami telah menggunakan toUpper fungsi dari Kelas Jenis Charuntuk mengubah masukan kita menjadi huruf besar. Di sini, metode "map" mengambil fungsi sebagai argumen dan mengembalikan keluaran yang diperlukan. Ini hasilnya -

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

Ekspresi Lambda

Terkadang kita harus menulis fungsi yang akan digunakan hanya sekali, sepanjang umur aplikasi. Untuk menghadapi situasi seperti ini, pengembang Haskell menggunakan blok anonim lain yang dikenal sebagailambda expression atau lambda function.

Fungsi tanpa definisi disebut fungsi lambda. Fungsi lambda dilambangkan dengan karakter "\". Mari kita ambil contoh berikut di mana kita akan meningkatkan nilai input sebesar 1 tanpa membuat fungsi apa pun.

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

Di sini, kami telah membuat fungsi anonim yang tidak memiliki nama. Dibutuhkan integer 4 sebagai argumen dan mencetak nilai output. Kami pada dasarnya mengoperasikan satu fungsi bahkan tanpa mendeklarasikannya dengan benar. Itulah keindahan ekspresi lambda.

Ekspresi lambda kami akan menghasilkan keluaran berikut -

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