Haskell - İşlevler

Fonksiyonlar, işlevsel bir programlama dili olduğu için Haskell'de önemli bir rol oynar. Diğer diller gibi Haskell'in de kendi işlevsel tanımı ve bildirimi vardır.

  • İşlev bildirimi, işlev adı ve çıktısı ile birlikte argüman listesinden oluşur.

  • İşlev tanımı, aslında bir işlevi tanımladığınız yerdir.

Küçük bir örnek alalım add bu kavramı ayrıntılı olarak anlama işlevi.

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

Burada fonksiyonumuzu ilk satırda ve ikinci satırda ilan ettik, iki argüman alacak ve bir tamsayı türü çıktı üretecek gerçek fonksiyonumuzu yazdık.

Diğer dillerin çoğunda olduğu gibi, Haskell kodu mainyöntem. Kodumuz aşağıdaki çıktıyı üretecektir -

The addition of the two numbers is:
7

Desen Eşleştirme

Desen Eşleştirme, belirli türdeki ifadeleri eşleştirme işlemidir. Kodunuzu basitleştirmek için bir teknikten başka bir şey değildir. Bu teknik, herhangi bir Type sınıfına uygulanabilir. If-Else, alternatif bir model eşleştirme seçeneği olarak kullanılabilir.

Örüntü Eşleştirme, çalışma zamanında, bağımsız değişken listesine bağlı olarak farklı yöntemlerin yürütülebildiği dinamik çok biçimliliğin bir çeşidi olarak düşünülebilir.

Aşağıdaki kod bloğuna bir göz atın. Burada, bir sayının faktöriyelini hesaplamak için Örüntü Eşleştirme tekniğini kullandık.

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

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

Hepimiz bir sayının faktöriyelini nasıl hesaplayacağımızı biliyoruz. Derleyici, bir bağımsız değişkenle "olgu" adlı bir işlevi aramaya başlayacaktır. Eğer argüman 0'a eşit değilse, o zaman sayı aynı fonksiyonu gerçek argümandan 1 daha küçük olarak çağırmaya devam edecektir.

Argümanın modeli 0 ile tam olarak eşleştiğinde, bizim kalıbımızı "olgu 0 = 1" olarak çağıracaktır. Kodumuz aşağıdaki çıktıyı üretecektir -

The factorial of 5 is:
120

Muhafızlar

Guardsdesen eşleştirmeye çok benzeyen bir kavramdır. Desen eşleştirmede genellikle bir veya daha fazla ifadeyle eşleşiriz, ancakguards bir ifadenin bazı özelliklerini test etmek için.

Örüntü eşleştirmenin kullanılması tavsiye edilse de guardsancak bir geliştiricinin bakış açısından, guardsdaha okunaklı ve basittir. İlk kez kullananlar için,guards If-Else ifadelerine çok benzer görünebilir, ancak işlevsel olarak farklıdırlar.

Aşağıdaki kodda, bizim factorial kavramını kullanarak program 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)

Burada iki ilan ettik guards"|" ile ayrılmış ve çağırmakfact işlevi main. Dahili olarak, derleyici desen eşleştirme durumunda olduğu gibi aynı şekilde çalışacak ve aşağıdaki çıktıyı verecektir -

The factorial of 5 is:
120

Madde nerede

Whereistenen bir çıktıyı oluşturmak için çalışma zamanında kullanılabilen bir anahtar sözcük veya dahili işlevdir. Fonksiyon hesaplaması karmaşık hale geldiğinde çok yardımcı olabilir.

Girişinizin birden çok parametresi olan karmaşık bir ifade olduğu bir senaryo düşünün. Bu gibi durumlarda, "where" cümlesini kullanarak tüm ifadeyi küçük parçalara ayırabilirsiniz.

Aşağıdaki örnekte, karmaşık bir matematiksel ifade alıyoruz. Haskell'i kullanarak bir polinom denkleminin [x ^ 2 - 8x + 6] köklerini nasıl bulabileceğinizi göstereceğiz.

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))

Verilen polinom fonksiyonunun köklerini hesaplamak için ifademizin karmaşıklığına dikkat edin. Oldukça karmaşık. Bu nedenle, ifadeyi kullanarakwherefıkra. Yukarıdaki kod parçası aşağıdaki çıktıyı üretecektir -

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

Özyineleme İşlevi

Özyineleme, bir fonksiyonun kendisini tekrar tekrar çağırdığı bir durumdur. Haskell, herhangi bir ifadeyi birden fazla döngü yapma olanağı sağlamaz. Bunun yerine Haskell, tüm işlevselliğinizi farklı işlevler koleksiyonuna ayırmanızı ve işlevselliğinizi uygulamak için özyineleme tekniğini kullanmanızı ister.

Bir sayının faktöriyelini hesapladığımız örüntü eşleştirme örneğimizi tekrar ele alalım. Bir sayının faktöriyelini bulmak, Özyinelemeyi kullanmanın klasik bir örneğidir. Burada, "Desen eşlemesinin özyinelemeden farkı nedir?" Bu ikisi arasındaki fark, kullanılma şekillerinde yatmaktadır.Örüntü eşleme, terminal sınırlamasını ayarlamakta çalışır, oysa özyineleme bir işlev çağrısıdır.

Aşağıdaki örnekte, 5'in faktöriyelini hesaplamak için hem örüntü eşleştirmeyi hem de özyinelemeyi kullandık.

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

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

Aşağıdaki çıktıyı üretecektir -

The factorial of 5 is:
120

Yüksek Sipariş İşlevi

Şimdiye kadar Haskell fonksiyonlarının bir tane aldığını gördük. type girdi olarak ve başka bir şey üret typediğer zorunlu dillerde oldukça benzer olan çıktı olarak. Yüksek Sıralı İşlevler, bir işlevi girdi veya çıktı bağımsız değişkeni olarak kullanabileceğiniz Haskell'in benzersiz bir özelliğidir.

Sanal bir kavram olmasına rağmen, gerçek dünya programlarında Haskell'de tanımladığımız her işlev, çıktı sağlamak için üst düzey mekanizma kullanır. Haskell'in kütüphane fonksiyonuna bakma şansınız olursa, kütüphane fonksiyonlarının çoğunun üst sıralarda yazıldığını göreceksiniz.

Yerleşik bir yüksek dereceli fonksiyon haritasını içe aktaracağımız ve seçimimize göre başka bir yüksek dereceli fonksiyonu uygulamak için aynısını kullanacağımız bir örnek alalım.

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"

Yukarıdaki örnekte, toUpper Tip Sınıfının işlevi Chargirdimizi büyük harfe dönüştürmek için. Burada, "harita" yöntemi, bir işlevi argüman olarak alıyor ve gerekli çıktıyı döndürüyor. İşte çıktısı -

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

Lambda İfadesi

Bazen bir uygulamanın tüm ömrü boyunca yalnızca bir kez kullanılacak bir işlev yazmamız gerekir. Bu tür durumlarla başa çıkmak için Haskell geliştiricileri,lambda expression veya lambda function.

Tanımı olmayan bir işleve lambda işlevi denir. Bir lambda işlevi "\" karakteriyle gösterilir. Herhangi bir fonksiyon oluşturmadan girdi değerini 1 artıracağımız aşağıdaki örneği alalım.

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

Burada adı olmayan anonim bir işlev oluşturduk. 4 tamsayısını bağımsız değişken olarak alır ve çıktı değerini yazdırır. Temelde, düzgün bir şekilde bildirmeden bir işlevi çalıştırıyoruz. Lambda ifadelerinin güzelliği budur.

Lambda ifademiz aşağıdaki çıktıyı üretecektir -

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