İksir - Enumerables
Bir numaralandırılabilir, numaralandırılabilen bir nesnedir. "Numaralandırılmış", bir kümenin / koleksiyonun / kategorinin üyelerini tek tek saymak anlamına gelir (genellikle sırayla, genellikle adıyla).
Elixir, numaralandırılabilir kavramları ve bunlarla çalışmak için Enum modülünü sağlar. Enum modülündeki işlevler, adından da anlaşılacağı gibi, veri yapılarındaki değerleri numaralandırmakla sınırlıdır. Numaralandırılabilir bir veri yapısı örneği bir liste, kayıt, harita vb. Olabilir. Enum modülü, numaralandırmalarla başa çıkmak için bize 100'den biraz fazla işlev sağlar. Bu bölümde birkaç önemli işlevi tartışacağız.
Tüm bu işlevler, ilk öğe olarak bir numaralandırılabilir ve ikinci öğe olarak bir işlevi alır ve bunlar üzerinde çalışır. Fonksiyonlar aşağıda açıklanmıştır.
herşey?
Kullandığımız zaman all? işlev, koleksiyonun tamamı doğru olarak değerlendirilmelidir, aksi takdirde yanlış döndürülür. Örneğin, listedeki tüm elemanların tek sayı olup olmadığını kontrol etmek için, o zaman.
res = Enum.all?([1, 2, 3, 4], fn(s) -> rem(s,2) == 1 end)
IO.puts(res)
Yukarıdaki program çalıştırıldığında, aşağıdaki sonucu verir -
false
Bunun nedeni, bu listenin tüm öğelerinin tuhaf olmamasıdır.
hiç?
Adından da anlaşılacağı gibi, koleksiyonun herhangi bir öğesi doğru olarak değerlendirilirse bu işlev true değerini döndürür. Örneğin -
res = Enum.any?([1, 2, 3, 4], fn(s) -> rem(s,2) == 1 end)
IO.puts(res)
Yukarıdaki program çalıştırıldığında, aşağıdaki sonucu verir -
true
yığın
Bu işlev, koleksiyonumuzu ikinci argüman olarak sağlanan boyutta küçük parçalara böler. Örneğin -
res = Enum.chunk([1, 2, 3, 4, 5, 6], 2)
IO.puts(res)
Yukarıdaki program çalıştırıldığında, aşağıdaki sonucu verir -
[[1, 2], [3, 4], [5, 6]]
her biri
Yeni bir değer üretmeden bir koleksiyon üzerinde yineleme yapmak gerekebilir, bu durumda each işlev -
Enum.each(["Hello", "Every", "one"], fn(s) -> IO.puts(s) end)
Yukarıdaki program çalıştırıldığında, aşağıdaki sonucu verir -
Hello
Every
one
harita
İşlevimizi her öğeye uygulamak ve yeni bir koleksiyon oluşturmak için harita işlevini kullanıyoruz. Oldukça etkileyici ve kısa olduğu için fonksiyonel programlamada en kullanışlı yapılardan biridir. Bunu anlamak için bir örnek ele alalım. Bir listede saklanan değerleri ikiye katlayacağız ve yeni bir listede saklayacağızres -
res = Enum.map([2, 5, 3, 6], fn(a) -> a*2 end)
IO.puts(res)
Yukarıdaki program çalıştırıldığında, aşağıdaki sonucu verir -
[4, 10, 6, 12]
azaltmak
reduceişlevi, numaralandırılmamızı tek bir değere indirmemize yardımcı olur. Bunu yapmak için, fonksiyonumuza geçirilecek isteğe bağlı bir akümülatör (bu örnekte 5) sağlıyoruz; akümülatör sağlanmadıysa, ilk değer kullanılır -
res = Enum.reduce([1, 2, 3, 4], 5, fn(x, accum) -> x + accum end)
IO.puts(res)
Yukarıdaki program çalıştırıldığında, aşağıdaki sonucu verir -
15
Akümülatör, aktarılan ilk değerdir. fn. İkinci aramadan itibaren, önceki aramadan dönen değer birikmiş olarak aktarılır. Ayrıca akümülatör olmadan azaltmayı da kullanabiliriz -
res = Enum.reduce([1, 2, 3, 4], fn(x, accum) -> x + accum end)
IO.puts(res)
Yukarıdaki program çalıştırıldığında, aşağıdaki sonucu verir -
10
uniq
Uniq işlevi, koleksiyonumuzdan kopyaları kaldırır ve yalnızca koleksiyondaki öğe kümesini döndürür. Örneğin -
res = Enum.uniq([1, 2, 2, 3, 3, 3, 4, 4, 4, 4])
IO.puts(res)
Yukarıdaki program çalıştırıldığında, aşağıdaki sonucu verir -
[1, 2, 3, 4]
Hevesli Değerlendirme
Enum modülündeki tüm işlevler isteklidir. Birçok işlev bir numaralandırılabilir ve bir listeyi geri döndürür. Bu, Enum ile birden fazla işlem gerçekleştirirken, sonuca ulaşana kadar her işlemin bir ara liste oluşturacağı anlamına gelir. Bunu anlamak için aşağıdaki örneği ele alalım -
odd? = &(odd? = &(rem(&1, 2) != 0)
res = 1..100_000 |> Enum.map(&(&1 * 3)) |> Enum.filter(odd?) |> Enum.sum
IO.puts(res)
Yukarıdaki program çalıştırıldığında, aşağıdaki sonucu verir -
7500000000
Yukarıdaki örnek bir işlem hattına sahiptir. Bir aralıkla başlıyoruz ve ardından aralıktaki her bir öğeyi 3 ile çarpıyoruz. Bu ilk işlem şimdi 100_000 öğeli bir liste oluşturacak ve döndürecektir. Sonra listedeki tüm garip öğeleri saklar, şimdi 50_000 öğe içeren yeni bir liste oluştururuz ve sonra tüm girişleri toplarız.
|> Yukarıdaki kod parçacığında kullanılan sembol, pipe operator: Çıktıyı sol tarafındaki ifadeden alır ve ilk argüman olarak sağ tarafındaki işlev çağrısına iletir. Unix'e benzer | Şebeke. Amacı, bir dizi işlev tarafından dönüştürülen veri akışını vurgulamaktır.
Olmadan pipe operatör, kod karmaşık görünüyor -
Enum.sum(Enum.filter(Enum.map(1..100_000, &(&1 * 3)), odd?))
Başka birçok fonksiyonumuz var, ancak burada sadece birkaç önemli fonksiyon açıklanmıştır.