SlideShare a Scribd company logo
1 of 31
Download to read offline
fold
Ξ»
The Func%onal Programming Triad of Folding, Scanning and Itera%on
a first example in Scala and Haskell
Polyglot FP for Fun and Profit
@philip_schwarz
slides by
h"ps://www.slideshare.net/pjschwarz
Richard Bird
Sergei Winitzki
sergei-winitzki-11a6431
http://www.cs.ox.ac.uk/people/richard.bird/
This slide deck can work both as an aide mémoire (memory jogger), or as a first (not
completely trivial) example of using le/ folds, le/ scans and itera5on to implement
mathema5cal induc5on.
We first look at the implementaAon, using a le/ fold, of a digits-to-int funcAon
that converts a sequence of digits into a whole number.
Then we look at the implementaAon, using the iterate funcAon, of an int-to-
digits funcAon that converts an integer into a sequence of digits. In Scala this
funcAon is very readable because the signatures of funcAons provided by collec5ons
permit the piping of such funcAons with zero syntac5c overhead. In Haskell, there is
some syntac5c sugar that can be used to achieve the same readability, so we look at
how that works.
We then set ourselves a simple task involving digits-to-int and int-to-digits,
and write a funcAon whose logic can be simplified with the introducAon of a le/ scan.
Let’s begin, on the next slide, by looking at how Richard Bird describes the digits-
to-int funcAon, which he calls π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™.
@philip_schwarz
Suppose we want a funcAon decimal that takes a list of digits and returns the corresponding decimal number;
thus
π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™ [π‘₯0, π‘₯1, … , π‘₯n] = βˆ‘!"#
$
π‘₯ π‘˜10($&!)
It is assumed that the most significant digit comes first in the list. One way to compute decimal efficiently is by a
process of mulAplying each digit by ten and adding in the following digit. For example
π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™ π‘₯0, π‘₯1, π‘₯2 = 10 Γ— 10 Γ— 10 Γ— 0 + π‘₯0 + π‘₯1 + π‘₯2
This decomposiAon of a sum of powers is known as Horner’s rule.
Suppose we define βŠ• by 𝑛 βŠ• π‘₯ = 10 Γ— 𝑛 + π‘₯. Then we can rephrase the above equaAon as
π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™ π‘₯0, π‘₯1, π‘₯2 = (0 βŠ• π‘₯0) βŠ• π‘₯1 βŠ• π‘₯2
This example moAvates the introducAon of a second fold operator called π‘“π‘œπ‘™π‘‘π‘™ (pronounced β€˜fold leM’).
Informally:
π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, … , π‘₯𝑛 βˆ’ 1 = … ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) … βŠ• π‘₯ 𝑛 βˆ’ 1
The parentheses group from the leM, which is the reason for the name. The full definiAon of π‘“π‘œπ‘™π‘‘π‘™ is
π‘“π‘œπ‘™π‘‘π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ 𝛽
π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 = 𝑒
π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠
Richard Bird
On the next slide we look at how Sergei
Winitzki describes the digits-to-int
funcAon, and how he implements it in Scala.
Example 2.2.5.3 Implement the funcAon digits-to-int using foldLeft.
…
The required computaAon can be wriTen as the formula
π‘Ÿ = @
!"#
$&(
𝑑 π‘˜ βˆ— 10$&(&!.
…
Solu5on The induc5ve defini5on of digitsToInt
β€’ For an empty sequence of digits, Seq(), the result is 0. This is a convenient base case, even if we never call
digitsToInt on an empty sequence.
β€’ If digitsToInt(xs) is already known for a sequence xs of digits, and we have a sequence xs :+ x with one
more digit x, then
is directly translated into code:
def digitsToInt(d: Seq[Int]): Int = d.foldLeft(0){ (n, x) => n * 10 + x }
Sergei Winitzki
sergei-winitzki-11a6431
(βŠ•) :: Int -> Int -> Int
(βŠ•) n d = 10 * n + d
digits_to_int :: [Int] -> Int
digits_to_int = foldl (βŠ•) 0
def `(βŠ•)`(n: Int, d: Int): Int =
10 * n + d
def digitsToInt(xs: Seq[Int]): Int =
xs.foldLeft(0)(`(βŠ•)`)
assert( digitsToInt(List(5,3,7,4)) == 5374)
implicit class Sugar(m: Int){
def βŠ•(n: Int) = `(βŠ•)`(m,n)
}
assert( (150 βŠ• 7) == 1507 )
π‘“π‘œπ‘™π‘‘π‘™ + 0 1,2,3 = 6π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, … , π‘₯𝑛 = … ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) … βŠ• π‘₯ 𝑛
π‘“π‘œπ‘™π‘‘π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ 𝛽
π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 = 𝑒
π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠
π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2
= π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 βŠ• π‘₯0 π‘₯1, π‘₯2
= π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 βŠ• π‘₯0 βŠ• π‘₯1 π‘₯2
= π‘“π‘œπ‘™π‘‘π‘™ βŠ• (( 𝑒 βŠ• π‘₯0 βŠ• π‘₯1) βŠ• π‘₯2) [ ]
= ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2
π‘“π‘œπ‘™π‘‘π‘™ βŠ• 0 1,2,3
= π‘“π‘œπ‘™π‘‘π‘™ βŠ• 0 βŠ• 1 2,3
= π‘“π‘œπ‘™π‘‘π‘™ βŠ• 0 βŠ• 1 βŠ• 2 3
= π‘“π‘œπ‘™π‘‘π‘™ βŠ• (( 0 βŠ• 1 βŠ• 2) βŠ• 3) [ ]
= ((0 βŠ• 1) βŠ• 2) βŠ• 3
Here is a quick refresher on fold le/
So let’s implement the
digits-to-int funcAon
in Haskell.
And now in Scala.
π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™ [π‘₯0, π‘₯1, … , π‘₯n] = @
!"#
$
π‘₯ π‘˜10($&!)
Now we turn to int-to-digits, a funcAon that is the opposite of digits-to-int,
and one that can be implemented using the iterate funcAon.
In the next two slides, we look at how Sergei Winitzki describes digits-to-int
(which he calls digitsOf) and how he implements it in Scala.
2.3 Converting a single value into a sequence
An aggregation converts (β€œfolds”) a sequence into a single value; the opposite operation (β€œunfolding”) converts a single value into
a sequence. An example of this task is to compute the sequence of decimal digits for a given integer:
def digitsOf(x: Int): Seq[Int] = ???
scala> digitsOf(2405)
res0: Seq[Int] = List(2, 4, 0, 5)
We cannot implement digitsOf using map, zip, or foldLeft, because these methods work only if we already have a sequence;
but the function digitsOf needs to create a new sequence.
…
To figure out the code for digitsOf, we first write this function as a mathematical formula. To compute the digits for, say, n =
2405, we need to divide n repeatedly by 10, getting a sequence nk of intermediate numbers (n0 = 2405, n1 = 240, ...) and the
corresponding sequence of last digits, nk mod 10 (in this example: 5, 0, ...). The sequence nk is defined using mathematical
induction:
β€’ Base case: n0 = n, where n is the given initial integer.
β€’ Inductive step: nk+1 =
nk
10 for k = 1, 2, ...
Here
nk
10 is the mathematical notation for the integer division by 10.
Let us tabulate the evaluation of the sequence nk for n = 2405:
The numbers nk will remain all zeros after k = 4. It is clear that the useful part of the sequence is before it becomes all zeros. In this
example, the sequence nk needs to be stopped at k = 4. The sequence of digits then becomes [5, 0, 4, 2], and we need to reverse it
to obtain [2, 4, 0, 5]. For reversing a sequence, the Scala library has the standard method reverse.
Sergei Winitzki
sergei-winitzki-11a6431
So, a complete implementa.on for digitsOf is:
def digitsOf(n: Int): Seq[Int] =
if (n == 0) Seq(0) else { // n == 0 is a special case.
Stream.iterate(n) { nk => nk / 10 }
.takeWhile { nk => nk != 0 }
.map { nk => nk % 10 }
.toList.reverse
}
We can shorten the code by using the syntax such as (_ % 10) instead of { nk => nk % 10 },
def digitsOf(n: Int): Seq[Int] =
if (n == 0) Seq(0) else { // n == 0 is a special case.
Stream.iterate(n) (_ / 10 )
.takeWhile ( _ != 0 )
.map ( _ % 10 )
.toList.reverse
}
Sergei Winitzki
sergei-winitzki-11a6431
The Scala library has a general stream-producing func8on Stream.iterate. This func8on has two arguments, the ini8al
value and a func8on that computes the next value from the previous one:
…
The type signature of the method Stream.iterate can be wri?en as
def iterate[A](init: A)(next: A => A): Stream[A]
and shows a close correspondence to a defini8on by mathema8cal induc8on. The base case is the first value, init, and
the induc8ve step is a func8on, next, that computes the next element from the previous one. It is a general way of
crea8ng sequences whose length is not determined in advance.
the prelude funcAon iterate returns an infinite list:
π‘–π‘‘π‘’π‘Ÿπ‘Žπ‘‘π‘’ ∷ π‘Ž β†’ π‘Ž β†’ π‘Ž β†’ π‘Ž
π‘–π‘‘π‘’π‘Ÿπ‘Žπ‘‘π‘’ 𝑓 π‘₯ = π‘₯ ∢ π‘–π‘‘π‘’π‘Ÿπ‘Žπ‘‘π‘’ 𝑓 (𝑓 π‘₯)
In parAcular, π‘–π‘‘π‘’π‘Ÿπ‘Žπ‘‘π‘’ +1 1 is an infinite list of the posiAve integers, a value we can also
write as [1..].
Richard Bird
Here is how Richard Bird
describes the iterate funcAon
int_to_digits :: Int -> [Int]
int_to_digits n =
reverse (
map (n -> mod n 10) (
takeWhile (n -> n /= 0) (
iterate (n -> div n 10)
n
)
)
)
import LazyList.iterate
def intToDigits(n: Int): Seq[Int] =
iterate(n) (_ / 10 )
.takeWhile ( _ != 0 )
.map ( _ % 10 )
.toList
.reverse
Here on the leM I had a go at a Haskell implementaAon of the int-to-digits funcAon (we
are not handling cases like n=0 or negaAve n), and on the right, the same logic in Scala.
I find the Scala version slightly easier
to understand, because the funcAons
that we are calling appear in the order
in which they are executed. Contrast
that with the Haskell version, in which
the funcAon invocaAons occur in the
opposite order.
While the β€˜ο¬‚uent’ chaining of funcAon calls on Scala collecAons is very convenient, when using other funcAons, we
face the same problem that we saw in Haskell on the previous slide. e.g. in the following code, square appears first,
even though it is executed last, and vice versa for inc.
assert(square(twice(inc(3))) == 64)
assert ((3 pipe inc pipe twice pipe square) == 64)
To help a bit with that problem, in Scala
there is a pipe funcAon which is available
on all values, and which allows us to order
funcAon invocaAons in the β€˜right’ order.
Armed with pipe, we can rewrite the code so that funcAon
names occur in the same order in which the funcAons are
invoked, which makes the code more understandable.
def inc(n: Int): Int = n + 1
def twice(n: Int): Int = n * 2
def square(n: Int): Int = n * n
3 pipe inc
pipe twice
pipe square
square(twice(inc(3))
@philip_schwarz
What about in Haskell? First of all, in Haskell there is a func8on applica8on
operator called $, which we can some.mes use to omit parentheses
Application operator. This operator is redundant, since ordinary application (f x)
means the same as (f $ x).
However, $ has low, right-associative binding precedence, so it sometimes allows
parentheses to be omitted; for example:
f $ g $ h x = f (g (h x))
int_to_digits :: Int -> [Int]
int_to_digits n =
reverse (
map (n -> mod n 10) (
takeWhile (n -> n /= 0) (
iterate (n -> div n 10)
n
)
)
)
int_to_digits :: Int -> [Int]
int_to_digits n =
reverse $ map (x -> mod x 10)
$ takeWhile (x -> x > 0)
$ iterate (x -> div x 10)
$ n
Armed with $, we can simplify our func.on as follows
simplify
For beginners, the $ oVen makes Haskell code more difficult to
parse. In pracXce, the $ operator is used frequently, and you’ll
likely find you prefer using it over many parentheses. There’s
nothing magical about $; if you look at its type signature, you
can see how it works:
($) :: (a -> b) -> a -> b
The arguments are just a funcXon and a value. The trick is
that $ is a binary operator, so it has lower precedence than the
other funcXons you’re using. Therefore, the argument for the
funcXon will be evaluated as though it were in parentheses.
But there is more we can do. In addiAon to the
func5on applica5on operator $, in Haskell there
is also a reverse func5on applica5on operator &.
hTps://www.fpcomplete.com/haskell/tutorial/operators/
int_to_digits :: Int -> [Int]
int_to_digits n =
reverse $ map (x -> mod x 10)
$ takeWhile (x -> x > 0)
$ iterate (x -> div x 10)
$ n
int_to_digits :: Int -> [Int]
int_to_digits s n =
n & iterate (x -> div x 10)
& takeWhile (x -> x > 0)
& map (x -> mod x 10)
& reverse
def intToDigits(n: Int): Seq[Int] =
iterate(n) (_ / 10 )
.takeWhile ( _ != 0 )
.map ( _ % 10 )
.toList
.reverse
Thanks to the & operator, we can rearrange our int_to_digits
funcAon so that it is as readable as the Scala version.
There is one school of thought according to which the choice of names for $ and & make
Haskell hard to read for newcomers, that it is beTer if $ and & are instead named |> and <|.
Flow provides operators for writing more understandable Haskell. It is an alternative to some
common idioms like ($) for function application and (.) for function composition.
…
Rationale
I think that Haskell can be hard to read. It has two operators for applying functions. Both are not
really necessary and only serve to reduce parentheses. But they make code hard to read. People
who do not already know Haskell have no chance of guessing what foo $ bar or baz &
qux mean.
…
I think we can do better. By using directional operators, we can allow readers to move their eye
in only one direction, be that left-to-right or right-to-left. And by using idioms common in other
programming languages, we can allow people who aren't familiar with Haskell to guess at the
meaning.
So instead of ($), I propose (<|). It is a pipe, which anyone who has touched a Unix system
should be familiar with. And it points in the direction it sends arguments along. Similarly,
replace (&) with (|>). …
square $ twice $ inc $ 3
square <| twice <| inc <| 3
3 & inc & twice & square
3 |> inc |> twice |> square
Here is an example of how |> and
<| improve readability
Since & is just the reverse of $, we can define |> ourselves simply by flipping $
Ξ» "left" ++ "right"
"leftright”
Ξ»
Ξ» (##) = flip (++)
Ξ»
Ξ» "left" ## "right"
"rightleft"
Ξ»
Ξ» inc n = n + 1
Ξ» twice n = n * 2
Ξ» square n = n * n
Ξ»
Ξ» square $ twice $ inc $ 3
64
Ξ»
Ξ» (|>) = flip ($)
Ξ»
Ξ» 3 |> inc |> twice |> square
64
Ξ»
int_to_digits :: Int -> [Int]
int_to_digits s n =
n |> iterate (x -> div x 10)
|> takeWhile (x -> x > 0)
|> map (x -> mod x 10)
|> reverse
And here is how our
funcAon looks using |>.
@philip_schwarz
Now let’s set ourselves the following task. Given a posiAve integer N with n digits, e.g. the five-digit number
12345, we want to compute the following:
[(0,0),(1,1),(12,3),(123,6),(1234,10),(12345,15)]
i.e. we want to compute a list of pairs p0 , p1 , … , pn with pk being ( Nk , Nk 𝛴 ), where Nk is the integer number
formed by the first k digits of N, and Nk𝛴 is the sum of those digits. We can use our int_to_digits funcAon to
convert N into its digits d1 , d2 , … , dn :
Ξ» int_to_digits 12345
[1,2,3,4,5]
Ξ»
And we can use digits_to_int to turn digits d1 , d2 , … , dk into Nk , e.g. for k = 3 :
Ξ» digits_to_int [1,2,3]
123
Ξ»
How can we generate the following sequences of digits ?
[ [ ] , [d1 ] , [d1 , d2 ] , [d1 , d2 , d3 ] , … , [d1 , d2 , d3 , … , dn ] ]
As we’ll see on the next slide, that is exactly what the inits funcAon produces when passed [d1 , d2 , d3 , … , dn ] !
𝑖𝑛𝑖𝑑𝑠 π‘₯0, π‘₯1, π‘₯2 = [[ ], π‘₯0 , π‘₯0, π‘₯1 , π‘₯0, π‘₯1, π‘₯2 ]
𝑖𝑛𝑖𝑑𝑠 ∷ Ξ± β†’ Ξ±
𝑖𝑛𝑖𝑑𝑠 = [ ]
𝑖𝑛𝑖𝑑𝑠 π‘₯: π‘₯𝑠 = ∢ π‘šπ‘Žπ‘ π‘₯: (𝑖𝑛𝑖𝑑𝑠 π‘₯𝑠)
Here is a definiAon for inits.
So we can apply inits to [d1 , d2 , d3 , … , dn ] to generate the following:
[ [ ] , [d1 ] , [d1 , d2 ] , [d1 , d2 , d3 ] , … , [d1 , d2 , d3 , … , dn ] ]
e.g.
Ξ» inits [1,2,3,4]
[[],[1],[1,2],[1,2,3],[1,2,3,4]]]
Ξ»
And here is what inits produces, i.e.
the list of all ini5al segments of a list.
So if we map digits_to_int over the iniXal segments of [d1 , d2 , d3 , … , dn ], i.e
[ [ ] , [d1 ] , [d1 , d2 ] , [d1 , d2 , d3 ] , … , [d1 , d2 , d3 , … , dn ] ]
we obtain a list containing N0 , N1 , … , Nn , e.g.
Ξ» map digits_to_int (inits [1,2,3,4,5]))
[0,1,12,123,1234,12345]]
Ξ»
What we need now is a funcXon digits_to_sum which is similar to digits_to_int but which instead of converXng a list of digits
[d1 , d2 , d3 , … , dk ] into Nk , i.e. the number formed by those digits, it turns the list into Nk𝛴, i.e. the sum of the digits. Like
digits_to_int, the digits_to_sum funcXon can be defined using a leB fold:
digits_to_sum :: [Int] -> Int
digits_to_sum = foldl (+) 0
Let’s try it out:
Ξ» map digits_to_sum [1,2,3,4])
1234]
Ξ»
Now if we map digits_to_sum over the iniXal segments of [d1 , d2 , d3 , … , dn ], we obtain a list containing N0 𝛴 , N1 𝛴 , … , Nn 𝛴 ,
e.g.
Ξ» map digits_to_sum (inits [1,2,3,4,5]))
[0,1,12,123,1234,12345]]
Ξ»
(βŠ•) :: Int -> Int -> Int
(βŠ•) n d = 10 * n + d
digits_to_int :: [Int] -> Int
digits_to_int = foldl (βŠ•) 0
digits_to_sum :: [Int] -> Int
digits_to_sum = foldl (+) 0
(βŠ•) :: Int -> Int -> Int
(βŠ•) n d = 10 * n + d
digits_to_int :: [Int] -> Int
digits_to_int = foldl (βŠ•) 0
convert :: Int -> [(Int,Int)]
convert n = zip nums sums
where nums = map digits_to_int segments
sums = map digits_to_sum segments
segments = inits (int_to_digits n)
int_to_digits :: Int -> [Int]
int_to_digits s n =
n |> iterate (x -> div x 10)
|> takeWhile (x -> x > 0)
|> map (x -> mod x 10)
|> reverse
Ξ» convert 12345
[(0,0),(1,1),(12,3),(123,6),(1234,10),(12345,15)]
Ξ»
So here is how our complete
program looks at the moment.
@philip_schwarz
What we are going to do next is see how, by using the scan le/ funcAon, we are
able to simplify the definiAon of convert which we just saw on the previous slide.
As a quick refresher of (or introducAon to) the scan le/ funcAon, in the next two
slides we look at how Richard Bird describes the funcAon.
4.5.2 Scan le/
SomeXmes it is convenient to apply a π‘“π‘œπ‘™π‘‘π‘™ operaXon to every iniXal segment of a list. This is done by a funcXon π‘ π‘π‘Žπ‘›π‘™
pronounced β€˜scan leB’. For example,
π‘ π‘π‘Žπ‘›π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2 = [𝑒, 𝑒 βŠ• π‘₯0, (𝑒 βŠ• π‘₯0) βŠ• π‘₯1, ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2]
In parXcular, π‘ π‘π‘Žπ‘›π‘™ + 0 computes the list of accumulated sums of a list of numbers, and π‘ π‘π‘Žπ‘›π‘™ Γ— 1 [1. . 𝑛] computes a list of
the first 𝑛 factorial numbers. … We will give two programs for π‘ π‘π‘Žπ‘›π‘™; the first is the clearest, while the second is more efficient.
For the first program we will need the funcXon inits that returns the list of all iniDal segments of a list. For Example,
𝑖𝑛𝑖𝑑𝑠 π‘₯0, π‘₯1, π‘₯2 = [[ ], π‘₯0 , π‘₯0, π‘₯1 ,
π‘₯0, π‘₯1, π‘₯2 ]
The empty list has only one segment, namely the empty list itself; A list π‘₯: π‘₯𝑠 has the empty list as its shortest iniDal segment,
and all the other iniDal segments begin with π‘₯ and are followed by an iniDal segment of π‘₯𝑠. Hence
𝑖𝑛𝑖𝑑𝑠 ∷ Ξ± β†’ Ξ±
𝑖𝑛𝑖𝑑𝑠 = [ ]
𝑖𝑛𝑖𝑑𝑠 π‘₯: π‘₯𝑠 = ∢ π‘šπ‘Žπ‘ (π‘₯: )(𝑖𝑛𝑖𝑑𝑠 π‘₯𝑠)
The funcXon 𝑖𝑛𝑖𝑑𝑠 can be defined more succinctly as an instance of π‘“π‘œπ‘™π‘‘π‘Ÿ :
𝑖𝑛𝑖𝑑𝑠 = π‘“π‘œπ‘™π‘‘π‘Ÿ 𝑓 [ ] π‘€β„Žπ‘’π‘Ÿπ‘’ 𝑓 π‘₯ π‘₯𝑠𝑠 = ∢ π‘šπ‘Žπ‘ π‘₯: π‘₯𝑠𝑠
Now we define
π‘ π‘π‘Žπ‘›π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ [𝛽]
π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = π‘šπ‘Žπ‘ π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 . 𝑖𝑛𝑖𝑑𝑠
This is the clearest definiXon of π‘ π‘π‘Žπ‘›π‘™ but it leads to an inefficient program. The funcXon 𝑓 is applied k	 Xmes in the evaluaXon of Richard Bird
π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 on a list of length k and, since the initial segments of a list of length n are lists with lengths 0,1,…,n, the function 𝑓 is
applied about n2/2 times in total.
Let us now synthesise a more efficient program. The synthesis is by an induction argument on π‘₯𝑠 so we lay out the calculation in
the same way.
<…not shown…>
In summary, we have derived
π‘ π‘π‘Žπ‘›π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ [𝛽]
π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = [𝑒]
π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = 𝑒 ∢ π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠
This program is more efficient in that function 𝑓 is applied exactly n times on a list of length n.
Richard Bird
π‘ π‘π‘Žπ‘›π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2
⬇
[𝑒, 𝑒 βŠ• π‘₯0, (𝑒 βŠ• π‘₯0) βŠ• π‘₯1, ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2]
π‘“π‘œπ‘™π‘‘π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ 𝛽
π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 = 𝑒
π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠
Note the similariXes and differences between π‘ π‘π‘Žπ‘›π‘™ and π‘“π‘œπ‘™π‘‘π‘™, e.g. the leV hand sides of their equaXons are the
same, and their signatures are very similar, but π‘ π‘π‘Žπ‘›π‘™ returns [𝛽] rather than 𝛽 and while π‘“π‘œπ‘™π‘‘π‘™ is tail recursive,
π‘ π‘π‘Žπ‘›π‘™ isn’t.
π‘ π‘π‘Žπ‘›π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ [𝛽]
π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = [𝑒]
π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = 𝑒 ∢ π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠
π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2
⬇
((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2
On the next slide, a very simple example of
using scanl, and a reminder of how the result
of scanl relates to the result of inits and foldl.
𝑖𝑛𝑖𝑑𝑠 2,3,4 = [[ ], 2 , 2,3 , 2,3,4 ]
π‘ π‘π‘Žπ‘›π‘™ + 0 2,3,4 = [0, 2, 5, 9]
π‘“π‘œπ‘™π‘‘π‘™ + 0 [ ] =0
π‘“π‘œπ‘™π‘‘π‘™ + 0 2 = 2
π‘“π‘œπ‘™π‘‘π‘™ + 0 2,3 = 5
π‘“π‘œπ‘™π‘‘π‘™ + 0 2,3,4 = 9
π‘ π‘π‘Žπ‘›π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2
⬇
[𝑒, 𝑒 βŠ• π‘₯0, (𝑒 βŠ• π‘₯0) βŠ• π‘₯1, ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2]
𝑖𝑛𝑖𝑑𝑠 π‘₯0, π‘₯1, π‘₯2 = [[ ], π‘₯0 , π‘₯0, π‘₯1 , π‘₯0, π‘₯1, π‘₯2 ]
π‘ π‘π‘Žπ‘›π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ [𝛽]
π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = π‘šπ‘Žπ‘ π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 . 𝑖𝑛𝑖𝑑𝑠
π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, … , π‘₯𝑛 βˆ’ 1 =
… ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) … βŠ• π‘₯ 𝑛 βˆ’ 1
π‘“π‘œπ‘™π‘‘π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ 𝛽
π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 = 𝑒
π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠
π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2
= π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 βŠ• π‘₯0 π‘₯1, π‘₯2
= π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 βŠ• π‘₯0 βŠ• π‘₯1 π‘₯2
= π‘“π‘œπ‘™π‘‘π‘™ βŠ• (( 𝑒 βŠ• π‘₯0 βŠ• π‘₯1) βŠ• π‘₯2) [ ]
= ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2
π‘ π‘π‘Žπ‘›π‘™ + 0 2,3,4
⬇
[0, 0 + 2, 0 + 2 + 3, 0 + 2 + 3 + 4]
π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = π‘šπ‘Žπ‘ π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 . 𝑖𝑛𝑖𝑑𝑠
convert :: Int -> [(Int,Int)]
convert n = zip nums sums
where nums = map digits_to_int segments
sums = map digits_to_sum segments
segments = inits (int_to_digits n)
convert :: Int -> [(Int,Int)]
convert n = zip nums sums
where nums = map foldl (βŠ•) 0 segments
sums = map foldl (+) 0 segments
segments = inits (int_to_digits n)
AVer that refresher of (introducXon to) the scanl funcXon, let’s see how it can help us simplify our definiXon of the convert funcXon.
The first thing to do is to take the definiXon of convert and inline its invocaXons of digits_to_int and digits_to_sum:
refactor
Now let’s extract (int_to_digits n) into digits and inline segments.
refactor
convert :: Int -> [(Int,Int)]
convert n = zip nums sums
where nums = map foldl (βŠ•) 0 (inits digits)
sums = map foldl (+) 0 (inits digits)
digits = int_to_digits n
convert :: Int -> [(Int,Int)]
convert n = zip nums sums
where nums = map foldl (βŠ•) 0 segments
sums = map foldl (+) 0 segments
segments = inits (int_to_digits n)
convert :: Int -> [(Int,Int)]
convert n = zip nums sums
where nums = scanl (βŠ•) 0 digits
sums = scanl (+) 0 digits
digits = int_to_digits n
convert :: Int -> [(Int,Int)]
convert n = zip nums sums
where nums = map foldl (βŠ•) 0 (inits digits)
sums = map foldl (+) 0 (inits digits)
digits = int_to_digits n
refactor
As we saw earlier, mapping foldl over the result of applying inits to a list, is just applying
scanl to that list, so let’s simplify convert by calling scanl rather than mapping foldl.
convert :: Int -> [(Int,Int)]
convert n = zip nums sums
where nums = scanl (βŠ•) 0 digits
sums = scanl (+) 0 digits
digits = int_to_digits n
The suboptimal thing about our current definition of convert
is that it does two left scans over the same list of digits.
Can we refactor it to do a single scan? Yes, by using tupling, i.e. by changing the func.on
that we pass to scanl from a func.on like (βŠ•) or (+), which computes a single result, to
a func.on, let’s call it next, which uses those two func.ons to compute two results
convert :: Int -> [(Int,Int)]
convert n = scanl next (0, 0) digits
where next (number, sum) digit = (number βŠ• digit, sum + digit)
digits = int_to_digits n
On the next slide, we inline digits and compare the resul.ng
convert func.on with our ini.al version, which invoked scanl twice.
@philip_schwarz
convert :: Int -> [(Int,Int)]
convert n = scanl next (0, 0) (int_to_digits n)
where next (number, sum) digit = (number βŠ• digit, sum + digit)
convert :: Int -> [(Int,Int)]
convert n = zip nums sums
where nums = map digits_to_int segments
sums = map digits_to_sum segments
segments = inits (int_to_digits n)
Here is our first definiAon of convert
And here is our refactored version, which uses a left scan.
The next slide shows the complete Haskell program,
and next to it, the equivalent Scala program.
digits_to_sum :: [Int] -> Int
digits_to_sum = foldl (+) 0
(βŠ•) :: Int -> Int -> Int
(βŠ•) n d = 10 * n + d
int_to_digits :: Int -> [Int]
int_to_digits s n =
n |> iterate (x -> div x 10)
|> takeWhile (x -> x > 0)
|> map (x -> mod x 10)
|> reverse
Ξ» convert 12345
[(0,0),(1,1),(12,3),(123,6),(1234,10),(12345,15)]
Ξ»
convert :: Int -> [(Int,Int)]
convert n = scanl next (0, 0) (int_to_digits n)
where next (number, sum) digit =
(number βŠ• digit, sum + digit)
def digitsToInt(ds: Seq[Int]): Int =
ds.foldLeft(0)(`(βŠ•)`)
implicit class Sugar(m: Int)
{ def βŠ•(n: Int) = `(βŠ•)`(m,n) }
def digitsToSum(ds: Seq[Int]): Int =
ds.foldLeft(0)(_+_)
def intToDigits(n: Int): Seq[Int] =
iterate(n) (_ / 10 )
.takeWhile ( _ != 0 )
.map ( _ % 10 )
.toList
.reverse
def convert(n: Int): Seq[(Int,Int)] = {
val next: ((Int,Int),Int) => (Int,Int) = {
case ((number, sum), digit) =>
(number βŠ• digit, sum + digit) }
intToDigits(n).scanLeft((0,0))(next)}
digits_to_int :: [Int] -> Int
digits_to_int = foldl (βŠ•) 0
def `(βŠ•)`(n: Int, d: Int): Int =
10 * n + d
scala> convert(12345)
val res0…= List((0,0),(1,1),(12,3),(123,6),(1234,10),(12345,15))
scala>
fold
Ξ»
In the next slide we conclude this deck with Sergei Winitzkiβ€˜s
recap of how in func5onal programming we implement
mathema5cal induc5on using folding, scanning and itera5on.
Use arbitrary inducDve (i.e., recursive) formulas to:
β€’ convert sequences to single values (aggregaDon or β€œfolding”);
β€’ create new sequences from single values (β€œunfolding”);
β€’ transform exisXng sequences into new sequences.
Table 2.1: ImplemenXng mathemaDcal inducDon
Table 2.1 shows Scala code implemenDng those tasks. IteraDve calculaDons are implemented by translaDng mathemaDcal
inducDon directly into code. In the funcDonal programming paradigm, the programmer does not need to write any loops or use
array indices. Instead, the programmer reasons about sequences as mathemaDcal values: β€œStarDng from this value, we get that
sequence, then transform it into this other sequence,” etc. This is a powerful way of working with sequences, dicDonaries, and
sets. Many kinds of programming errors (such as an incorrect array index) are avoided from the outset, and the code is shorter
and easier to read than convenDonal code wriPen using loops.
Sergei Winitzki
sergei-winitzki-11a6431

More Related Content

What's hot

Functional Core and Imperative Shell - Game of Life Example - Haskell and Scala
Functional Core and Imperative Shell - Game of Life Example - Haskell and ScalaFunctional Core and Imperative Shell - Game of Life Example - Haskell and Scala
Functional Core and Imperative Shell - Game of Life Example - Haskell and ScalaPhilip Schwarz
Β 
Sequence and Traverse - Part 2
Sequence and Traverse - Part 2Sequence and Traverse - Part 2
Sequence and Traverse - Part 2Philip Schwarz
Β 
The Functional Programming Triad of Map, Filter and Fold
The Functional Programming Triad of Map, Filter and FoldThe Functional Programming Triad of Map, Filter and Fold
The Functional Programming Triad of Map, Filter and FoldPhilip Schwarz
Β 
Purely Functional Data Structures in Scala
Purely Functional Data Structures in ScalaPurely Functional Data Structures in Scala
Purely Functional Data Structures in ScalaVladimir Kostyukov
Β 
One Monad to Rule Them All
One Monad to Rule Them AllOne Monad to Rule Them All
One Monad to Rule Them AllJohn De Goes
Β 
Exploring ZIO Prelude: The game changer for typeclasses in Scala
Exploring ZIO Prelude: The game changer for typeclasses in ScalaExploring ZIO Prelude: The game changer for typeclasses in Scala
Exploring ZIO Prelude: The game changer for typeclasses in ScalaJorge VΓ‘squez
Β 
Sum and Product Types - The Fruit Salad & Fruit Snack Example - From F# to Ha...
Sum and Product Types -The Fruit Salad & Fruit Snack Example - From F# to Ha...Sum and Product Types -The Fruit Salad & Fruit Snack Example - From F# to Ha...
Sum and Product Types - The Fruit Salad & Fruit Snack Example - From F# to Ha...Philip Schwarz
Β 
N-Queens Combinatorial Problem - Polyglot FP for Fun and Profit – Haskell and...
N-Queens Combinatorial Problem - Polyglot FP for Fun and Profit – Haskell and...N-Queens Combinatorial Problem - Polyglot FP for Fun and Profit – Haskell and...
N-Queens Combinatorial Problem - Polyglot FP for Fun and Profit – Haskell and...Philip Schwarz
Β 
The Uniform Access Principle
The Uniform Access PrincipleThe Uniform Access Principle
The Uniform Access PrinciplePhilip Schwarz
Β 
Sequence and Traverse - Part 1
Sequence and Traverse - Part 1Sequence and Traverse - Part 1
Sequence and Traverse - Part 1Philip Schwarz
Β 
Ad hoc Polymorphism using Type Classes and Cats
Ad hoc Polymorphism using Type Classes and CatsAd hoc Polymorphism using Type Classes and Cats
Ad hoc Polymorphism using Type Classes and CatsPhilip Schwarz
Β 
Implicit parameters, when to use them (or not)!
Implicit parameters, when to use them (or not)!Implicit parameters, when to use them (or not)!
Implicit parameters, when to use them (or not)!Julien Truffaut
Β 
non-strict functions, bottom and scala by-name parameters
non-strict functions, bottom and scala by-name parametersnon-strict functions, bottom and scala by-name parameters
non-strict functions, bottom and scala by-name parametersPhilip Schwarz
Β 
Functor, Apply, Applicative And Monad
Functor, Apply, Applicative And MonadFunctor, Apply, Applicative And Monad
Functor, Apply, Applicative And MonadOliver Daff
Β 
Functors, Applicatives and Monads In Scala
Functors, Applicatives and Monads In ScalaFunctors, Applicatives and Monads In Scala
Functors, Applicatives and Monads In ScalaKnoldus Inc.
Β 
Monadic Java
Monadic JavaMonadic Java
Monadic JavaMario Fusco
Β 
Peeking inside the engine of ZIO SQL.pdf
Peeking inside the engine of ZIO SQL.pdfPeeking inside the engine of ZIO SQL.pdf
Peeking inside the engine of ZIO SQL.pdfJaroslavRegec1
Β 
Point free or die - tacit programming in Haskell and beyond
Point free or die - tacit programming in Haskell and beyondPoint free or die - tacit programming in Haskell and beyond
Point free or die - tacit programming in Haskell and beyondPhilip Schwarz
Β 
Sierpinski Triangle - Polyglot FP for Fun and Profit - Haskell and Scala
Sierpinski Triangle - Polyglot FP for Fun and Profit - Haskell and ScalaSierpinski Triangle - Polyglot FP for Fun and Profit - Haskell and Scala
Sierpinski Triangle - Polyglot FP for Fun and Profit - Haskell and ScalaPhilip Schwarz
Β 

What's hot (20)

Functional Core and Imperative Shell - Game of Life Example - Haskell and Scala
Functional Core and Imperative Shell - Game of Life Example - Haskell and ScalaFunctional Core and Imperative Shell - Game of Life Example - Haskell and Scala
Functional Core and Imperative Shell - Game of Life Example - Haskell and Scala
Β 
Sequence and Traverse - Part 2
Sequence and Traverse - Part 2Sequence and Traverse - Part 2
Sequence and Traverse - Part 2
Β 
The Functional Programming Triad of Map, Filter and Fold
The Functional Programming Triad of Map, Filter and FoldThe Functional Programming Triad of Map, Filter and Fold
The Functional Programming Triad of Map, Filter and Fold
Β 
Purely Functional Data Structures in Scala
Purely Functional Data Structures in ScalaPurely Functional Data Structures in Scala
Purely Functional Data Structures in Scala
Β 
One Monad to Rule Them All
One Monad to Rule Them AllOne Monad to Rule Them All
One Monad to Rule Them All
Β 
Exploring ZIO Prelude: The game changer for typeclasses in Scala
Exploring ZIO Prelude: The game changer for typeclasses in ScalaExploring ZIO Prelude: The game changer for typeclasses in Scala
Exploring ZIO Prelude: The game changer for typeclasses in Scala
Β 
Sum and Product Types - The Fruit Salad & Fruit Snack Example - From F# to Ha...
Sum and Product Types -The Fruit Salad & Fruit Snack Example - From F# to Ha...Sum and Product Types -The Fruit Salad & Fruit Snack Example - From F# to Ha...
Sum and Product Types - The Fruit Salad & Fruit Snack Example - From F# to Ha...
Β 
N-Queens Combinatorial Problem - Polyglot FP for Fun and Profit – Haskell and...
N-Queens Combinatorial Problem - Polyglot FP for Fun and Profit – Haskell and...N-Queens Combinatorial Problem - Polyglot FP for Fun and Profit – Haskell and...
N-Queens Combinatorial Problem - Polyglot FP for Fun and Profit – Haskell and...
Β 
The Uniform Access Principle
The Uniform Access PrincipleThe Uniform Access Principle
The Uniform Access Principle
Β 
Sequence and Traverse - Part 1
Sequence and Traverse - Part 1Sequence and Traverse - Part 1
Sequence and Traverse - Part 1
Β 
Ad hoc Polymorphism using Type Classes and Cats
Ad hoc Polymorphism using Type Classes and CatsAd hoc Polymorphism using Type Classes and Cats
Ad hoc Polymorphism using Type Classes and Cats
Β 
Zio in real world
Zio in real worldZio in real world
Zio in real world
Β 
Implicit parameters, when to use them (or not)!
Implicit parameters, when to use them (or not)!Implicit parameters, when to use them (or not)!
Implicit parameters, when to use them (or not)!
Β 
non-strict functions, bottom and scala by-name parameters
non-strict functions, bottom and scala by-name parametersnon-strict functions, bottom and scala by-name parameters
non-strict functions, bottom and scala by-name parameters
Β 
Functor, Apply, Applicative And Monad
Functor, Apply, Applicative And MonadFunctor, Apply, Applicative And Monad
Functor, Apply, Applicative And Monad
Β 
Functors, Applicatives and Monads In Scala
Functors, Applicatives and Monads In ScalaFunctors, Applicatives and Monads In Scala
Functors, Applicatives and Monads In Scala
Β 
Monadic Java
Monadic JavaMonadic Java
Monadic Java
Β 
Peeking inside the engine of ZIO SQL.pdf
Peeking inside the engine of ZIO SQL.pdfPeeking inside the engine of ZIO SQL.pdf
Peeking inside the engine of ZIO SQL.pdf
Β 
Point free or die - tacit programming in Haskell and beyond
Point free or die - tacit programming in Haskell and beyondPoint free or die - tacit programming in Haskell and beyond
Point free or die - tacit programming in Haskell and beyond
Β 
Sierpinski Triangle - Polyglot FP for Fun and Profit - Haskell and Scala
Sierpinski Triangle - Polyglot FP for Fun and Profit - Haskell and ScalaSierpinski Triangle - Polyglot FP for Fun and Profit - Haskell and Scala
Sierpinski Triangle - Polyglot FP for Fun and Profit - Haskell and Scala
Β 

Similar to FP Triad of Folding, Scanning and Iteration

The Functional Programming Triad of Folding, Scanning and Iteration - a first...
The Functional Programming Triad of Folding, Scanning and Iteration - a first...The Functional Programming Triad of Folding, Scanning and Iteration - a first...
The Functional Programming Triad of Folding, Scanning and Iteration - a first...Philip Schwarz
Β 
Folding Cheat Sheet #4 - fourth in a series
Folding Cheat Sheet #4 - fourth in a seriesFolding Cheat Sheet #4 - fourth in a series
Folding Cheat Sheet #4 - fourth in a seriesPhilip Schwarz
Β 
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - Part 2
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - Part 2Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - Part 2
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - Part 2Philip Schwarz
Β 
Functional programming ii
Functional programming iiFunctional programming ii
Functional programming iiPrashant Kalkar
Β 
Scala as a Declarative Language
Scala as a Declarative LanguageScala as a Declarative Language
Scala as a Declarative Languagevsssuresh
Β 
Scala Back to Basics: Type Classes
Scala Back to Basics: Type ClassesScala Back to Basics: Type Classes
Scala Back to Basics: Type ClassesTomer Gabel
Β 
Scala - where objects and functions meet
Scala - where objects and functions meetScala - where objects and functions meet
Scala - where objects and functions meetMario Fusco
Β 
Functional Programming by Examples using Haskell
Functional Programming by Examples using HaskellFunctional Programming by Examples using Haskell
Functional Programming by Examples using Haskellgoncharenko
Β 
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - with ...
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - with ...Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - with ...
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - with ...Philip Schwarz
Β 
Real World Haskell: Lecture 6
Real World Haskell: Lecture 6Real World Haskell: Lecture 6
Real World Haskell: Lecture 6Bryan O'Sullivan
Β 
How to start functional programming (in Scala): Day1
How to start functional programming (in Scala): Day1How to start functional programming (in Scala): Day1
How to start functional programming (in Scala): Day1Taisuke Oe
Β 
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and ScalaFolding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and ScalaPhilip Schwarz
Β 
Scala by Luc Duponcheel
Scala by Luc DuponcheelScala by Luc Duponcheel
Scala by Luc DuponcheelStephan Janssen
Β 
High-Performance Haskell
High-Performance HaskellHigh-Performance Haskell
High-Performance HaskellJohan Tibell
Β 
Introduction to Functional Programming with Scala
Introduction to Functional Programming with ScalaIntroduction to Functional Programming with Scala
Introduction to Functional Programming with Scalapramode_ce
Β 
Monadologie
MonadologieMonadologie
Monadologieleague
Β 
Fp in scala part 2
Fp in scala part 2Fp in scala part 2
Fp in scala part 2Hang Zhao
Β 
Real World Haskell: Lecture 2
Real World Haskell: Lecture 2Real World Haskell: Lecture 2
Real World Haskell: Lecture 2Bryan O'Sullivan
Β 
Monoids - Part 1 - with examples using Scalaz and Cats
Monoids - Part 1 - with examples using Scalaz and CatsMonoids - Part 1 - with examples using Scalaz and Cats
Monoids - Part 1 - with examples using Scalaz and CatsPhilip Schwarz
Β 
Composition birds-and-recursion
Composition birds-and-recursionComposition birds-and-recursion
Composition birds-and-recursionDavid Atchley
Β 

Similar to FP Triad of Folding, Scanning and Iteration (20)

The Functional Programming Triad of Folding, Scanning and Iteration - a first...
The Functional Programming Triad of Folding, Scanning and Iteration - a first...The Functional Programming Triad of Folding, Scanning and Iteration - a first...
The Functional Programming Triad of Folding, Scanning and Iteration - a first...
Β 
Folding Cheat Sheet #4 - fourth in a series
Folding Cheat Sheet #4 - fourth in a seriesFolding Cheat Sheet #4 - fourth in a series
Folding Cheat Sheet #4 - fourth in a series
Β 
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - Part 2
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - Part 2Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - Part 2
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - Part 2
Β 
Functional programming ii
Functional programming iiFunctional programming ii
Functional programming ii
Β 
Scala as a Declarative Language
Scala as a Declarative LanguageScala as a Declarative Language
Scala as a Declarative Language
Β 
Scala Back to Basics: Type Classes
Scala Back to Basics: Type ClassesScala Back to Basics: Type Classes
Scala Back to Basics: Type Classes
Β 
Scala - where objects and functions meet
Scala - where objects and functions meetScala - where objects and functions meet
Scala - where objects and functions meet
Β 
Functional Programming by Examples using Haskell
Functional Programming by Examples using HaskellFunctional Programming by Examples using Haskell
Functional Programming by Examples using Haskell
Β 
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - with ...
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - with ...Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - with ...
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala - with ...
Β 
Real World Haskell: Lecture 6
Real World Haskell: Lecture 6Real World Haskell: Lecture 6
Real World Haskell: Lecture 6
Β 
How to start functional programming (in Scala): Day1
How to start functional programming (in Scala): Day1How to start functional programming (in Scala): Day1
How to start functional programming (in Scala): Day1
Β 
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and ScalaFolding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala
Folding Unfolded - Polyglot FP for Fun and Profit - Haskell and Scala
Β 
Scala by Luc Duponcheel
Scala by Luc DuponcheelScala by Luc Duponcheel
Scala by Luc Duponcheel
Β 
High-Performance Haskell
High-Performance HaskellHigh-Performance Haskell
High-Performance Haskell
Β 
Introduction to Functional Programming with Scala
Introduction to Functional Programming with ScalaIntroduction to Functional Programming with Scala
Introduction to Functional Programming with Scala
Β 
Monadologie
MonadologieMonadologie
Monadologie
Β 
Fp in scala part 2
Fp in scala part 2Fp in scala part 2
Fp in scala part 2
Β 
Real World Haskell: Lecture 2
Real World Haskell: Lecture 2Real World Haskell: Lecture 2
Real World Haskell: Lecture 2
Β 
Monoids - Part 1 - with examples using Scalaz and Cats
Monoids - Part 1 - with examples using Scalaz and CatsMonoids - Part 1 - with examples using Scalaz and Cats
Monoids - Part 1 - with examples using Scalaz and Cats
Β 
Composition birds-and-recursion
Composition birds-and-recursionComposition birds-and-recursion
Composition birds-and-recursion
Β 

More from Philip Schwarz

Folding Cheat Sheet #3 - third in a series
Folding Cheat Sheet #3 - third in a seriesFolding Cheat Sheet #3 - third in a series
Folding Cheat Sheet #3 - third in a seriesPhilip Schwarz
Β 
Folding Cheat Sheet #2 - second in a series
Folding Cheat Sheet #2 - second in a seriesFolding Cheat Sheet #2 - second in a series
Folding Cheat Sheet #2 - second in a seriesPhilip Schwarz
Β 
Folding Cheat Sheet #1 - first in a series
Folding Cheat Sheet #1 - first in a seriesFolding Cheat Sheet #1 - first in a series
Folding Cheat Sheet #1 - first in a seriesPhilip Schwarz
Β 
Scala Left Fold Parallelisation - Three Approaches
Scala Left Fold Parallelisation- Three ApproachesScala Left Fold Parallelisation- Three Approaches
Scala Left Fold Parallelisation - Three ApproachesPhilip Schwarz
Β 
Tagless Final Encoding - Algebras and Interpreters and also Programs
Tagless Final Encoding - Algebras and Interpreters and also ProgramsTagless Final Encoding - Algebras and Interpreters and also Programs
Tagless Final Encoding - Algebras and Interpreters and also ProgramsPhilip Schwarz
Β 
Fusing Transformations of Strict Scala Collections with Views
Fusing Transformations of Strict Scala Collections with ViewsFusing Transformations of Strict Scala Collections with Views
Fusing Transformations of Strict Scala Collections with ViewsPhilip Schwarz
Β 
A sighting of traverse_ function in Practical FP in Scala
A sighting of traverse_ function in Practical FP in ScalaA sighting of traverse_ function in Practical FP in Scala
A sighting of traverse_ function in Practical FP in ScalaPhilip Schwarz
Β 
A sighting of traverseFilter and foldMap in Practical FP in Scala
A sighting of traverseFilter and foldMap in Practical FP in ScalaA sighting of traverseFilter and foldMap in Practical FP in Scala
A sighting of traverseFilter and foldMap in Practical FP in ScalaPhilip Schwarz
Β 
A sighting of sequence function in Practical FP in Scala
A sighting of sequence function in Practical FP in ScalaA sighting of sequence function in Practical FP in Scala
A sighting of sequence function in Practical FP in ScalaPhilip Schwarz
Β 
N-Queens Combinatorial Puzzle meets Cats
N-Queens Combinatorial Puzzle meets CatsN-Queens Combinatorial Puzzle meets Cats
N-Queens Combinatorial Puzzle meets CatsPhilip Schwarz
Β 
Kleisli composition, flatMap, join, map, unit - implementation and interrelat...
Kleisli composition, flatMap, join, map, unit - implementation and interrelat...Kleisli composition, flatMap, join, map, unit - implementation and interrelat...
Kleisli composition, flatMap, join, map, unit - implementation and interrelat...Philip Schwarz
Β 
The aggregate function - from sequential and parallel folds to parallel aggre...
The aggregate function - from sequential and parallel folds to parallel aggre...The aggregate function - from sequential and parallel folds to parallel aggre...
The aggregate function - from sequential and parallel folds to parallel aggre...Philip Schwarz
Β 
Nat, List and Option Monoids - from scratch - Combining and Folding - an example
Nat, List and Option Monoids -from scratch -Combining and Folding -an exampleNat, List and Option Monoids -from scratch -Combining and Folding -an example
Nat, List and Option Monoids - from scratch - Combining and Folding - an examplePhilip Schwarz
Β 
The Sieve of Eratosthenes - Part II - Genuine versus Unfaithful Sieve - Haske...
The Sieve of Eratosthenes - Part II - Genuine versus Unfaithful Sieve - Haske...The Sieve of Eratosthenes - Part II - Genuine versus Unfaithful Sieve - Haske...
The Sieve of Eratosthenes - Part II - Genuine versus Unfaithful Sieve - Haske...Philip Schwarz
Β 
Algebraic Data Types for Data Oriented Programming - From Haskell and Scala t...
Algebraic Data Types forData Oriented Programming - From Haskell and Scala t...Algebraic Data Types forData Oriented Programming - From Haskell and Scala t...
Algebraic Data Types for Data Oriented Programming - From Haskell and Scala t...Philip Schwarz
Β 
Jordan Peterson - The pursuit of meaning and related ethical axioms
Jordan Peterson - The pursuit of meaning and related ethical axiomsJordan Peterson - The pursuit of meaning and related ethical axioms
Jordan Peterson - The pursuit of meaning and related ethical axiomsPhilip Schwarz
Β 
Defining filter using (a) recursion (b) folding (c) folding with S, B and I c...
Defining filter using (a) recursion (b) folding (c) folding with S, B and I c...Defining filter using (a) recursion (b) folding (c) folding with S, B and I c...
Defining filter using (a) recursion (b) folding (c) folding with S, B and I c...Philip Schwarz
Β 
Defining filter using (a) recursion (b) folding with S, B and I combinators (...
Defining filter using (a) recursion (b) folding with S, B and I combinators (...Defining filter using (a) recursion (b) folding with S, B and I combinators (...
Defining filter using (a) recursion (b) folding with S, B and I combinators (...Philip Schwarz
Β 
The Sieve of Eratosthenes - Part 1 - with minor corrections
The Sieve of Eratosthenes - Part 1 - with minor correctionsThe Sieve of Eratosthenes - Part 1 - with minor corrections
The Sieve of Eratosthenes - Part 1 - with minor correctionsPhilip Schwarz
Β 
The Sieve of Eratosthenes - Part 1
The Sieve of Eratosthenes - Part 1The Sieve of Eratosthenes - Part 1
The Sieve of Eratosthenes - Part 1Philip Schwarz
Β 

More from Philip Schwarz (20)

Folding Cheat Sheet #3 - third in a series
Folding Cheat Sheet #3 - third in a seriesFolding Cheat Sheet #3 - third in a series
Folding Cheat Sheet #3 - third in a series
Β 
Folding Cheat Sheet #2 - second in a series
Folding Cheat Sheet #2 - second in a seriesFolding Cheat Sheet #2 - second in a series
Folding Cheat Sheet #2 - second in a series
Β 
Folding Cheat Sheet #1 - first in a series
Folding Cheat Sheet #1 - first in a seriesFolding Cheat Sheet #1 - first in a series
Folding Cheat Sheet #1 - first in a series
Β 
Scala Left Fold Parallelisation - Three Approaches
Scala Left Fold Parallelisation- Three ApproachesScala Left Fold Parallelisation- Three Approaches
Scala Left Fold Parallelisation - Three Approaches
Β 
Tagless Final Encoding - Algebras and Interpreters and also Programs
Tagless Final Encoding - Algebras and Interpreters and also ProgramsTagless Final Encoding - Algebras and Interpreters and also Programs
Tagless Final Encoding - Algebras and Interpreters and also Programs
Β 
Fusing Transformations of Strict Scala Collections with Views
Fusing Transformations of Strict Scala Collections with ViewsFusing Transformations of Strict Scala Collections with Views
Fusing Transformations of Strict Scala Collections with Views
Β 
A sighting of traverse_ function in Practical FP in Scala
A sighting of traverse_ function in Practical FP in ScalaA sighting of traverse_ function in Practical FP in Scala
A sighting of traverse_ function in Practical FP in Scala
Β 
A sighting of traverseFilter and foldMap in Practical FP in Scala
A sighting of traverseFilter and foldMap in Practical FP in ScalaA sighting of traverseFilter and foldMap in Practical FP in Scala
A sighting of traverseFilter and foldMap in Practical FP in Scala
Β 
A sighting of sequence function in Practical FP in Scala
A sighting of sequence function in Practical FP in ScalaA sighting of sequence function in Practical FP in Scala
A sighting of sequence function in Practical FP in Scala
Β 
N-Queens Combinatorial Puzzle meets Cats
N-Queens Combinatorial Puzzle meets CatsN-Queens Combinatorial Puzzle meets Cats
N-Queens Combinatorial Puzzle meets Cats
Β 
Kleisli composition, flatMap, join, map, unit - implementation and interrelat...
Kleisli composition, flatMap, join, map, unit - implementation and interrelat...Kleisli composition, flatMap, join, map, unit - implementation and interrelat...
Kleisli composition, flatMap, join, map, unit - implementation and interrelat...
Β 
The aggregate function - from sequential and parallel folds to parallel aggre...
The aggregate function - from sequential and parallel folds to parallel aggre...The aggregate function - from sequential and parallel folds to parallel aggre...
The aggregate function - from sequential and parallel folds to parallel aggre...
Β 
Nat, List and Option Monoids - from scratch - Combining and Folding - an example
Nat, List and Option Monoids -from scratch -Combining and Folding -an exampleNat, List and Option Monoids -from scratch -Combining and Folding -an example
Nat, List and Option Monoids - from scratch - Combining and Folding - an example
Β 
The Sieve of Eratosthenes - Part II - Genuine versus Unfaithful Sieve - Haske...
The Sieve of Eratosthenes - Part II - Genuine versus Unfaithful Sieve - Haske...The Sieve of Eratosthenes - Part II - Genuine versus Unfaithful Sieve - Haske...
The Sieve of Eratosthenes - Part II - Genuine versus Unfaithful Sieve - Haske...
Β 
Algebraic Data Types for Data Oriented Programming - From Haskell and Scala t...
Algebraic Data Types forData Oriented Programming - From Haskell and Scala t...Algebraic Data Types forData Oriented Programming - From Haskell and Scala t...
Algebraic Data Types for Data Oriented Programming - From Haskell and Scala t...
Β 
Jordan Peterson - The pursuit of meaning and related ethical axioms
Jordan Peterson - The pursuit of meaning and related ethical axiomsJordan Peterson - The pursuit of meaning and related ethical axioms
Jordan Peterson - The pursuit of meaning and related ethical axioms
Β 
Defining filter using (a) recursion (b) folding (c) folding with S, B and I c...
Defining filter using (a) recursion (b) folding (c) folding with S, B and I c...Defining filter using (a) recursion (b) folding (c) folding with S, B and I c...
Defining filter using (a) recursion (b) folding (c) folding with S, B and I c...
Β 
Defining filter using (a) recursion (b) folding with S, B and I combinators (...
Defining filter using (a) recursion (b) folding with S, B and I combinators (...Defining filter using (a) recursion (b) folding with S, B and I combinators (...
Defining filter using (a) recursion (b) folding with S, B and I combinators (...
Β 
The Sieve of Eratosthenes - Part 1 - with minor corrections
The Sieve of Eratosthenes - Part 1 - with minor correctionsThe Sieve of Eratosthenes - Part 1 - with minor corrections
The Sieve of Eratosthenes - Part 1 - with minor corrections
Β 
The Sieve of Eratosthenes - Part 1
The Sieve of Eratosthenes - Part 1The Sieve of Eratosthenes - Part 1
The Sieve of Eratosthenes - Part 1
Β 

Recently uploaded

Der Spagat zwischen BIAS und FAIRNESS (2024)
Der Spagat zwischen BIAS und FAIRNESS (2024)Der Spagat zwischen BIAS und FAIRNESS (2024)
Der Spagat zwischen BIAS und FAIRNESS (2024)OPEN KNOWLEDGE GmbH
Β 
(Genuine) Escort Service Lucknow | Starting β‚Ή,5K To @25k with A/C πŸ§‘πŸ½β€β€οΈβ€πŸ§‘πŸ» 89...
(Genuine) Escort Service Lucknow | Starting β‚Ή,5K To @25k with A/C πŸ§‘πŸ½β€β€οΈβ€πŸ§‘πŸ» 89...(Genuine) Escort Service Lucknow | Starting β‚Ή,5K To @25k with A/C πŸ§‘πŸ½β€β€οΈβ€πŸ§‘πŸ» 89...
(Genuine) Escort Service Lucknow | Starting β‚Ή,5K To @25k with A/C πŸ§‘πŸ½β€β€οΈβ€πŸ§‘πŸ» 89...gurkirankumar98700
Β 
Russian Call Girls in Karol Bagh Aasnvi ➑️ 8264348440 πŸ’‹πŸ“ž Independent Escort S...
Russian Call Girls in Karol Bagh Aasnvi ➑️ 8264348440 πŸ’‹πŸ“ž Independent Escort S...Russian Call Girls in Karol Bagh Aasnvi ➑️ 8264348440 πŸ’‹πŸ“ž Independent Escort S...
Russian Call Girls in Karol Bagh Aasnvi ➑️ 8264348440 πŸ’‹πŸ“ž Independent Escort S...soniya singh
Β 
Steps To Getting Up And Running Quickly With MyTimeClock Employee Scheduling ...
Steps To Getting Up And Running Quickly With MyTimeClock Employee Scheduling ...Steps To Getting Up And Running Quickly With MyTimeClock Employee Scheduling ...
Steps To Getting Up And Running Quickly With MyTimeClock Employee Scheduling ...MyIntelliSource, Inc.
Β 
chapter--4-software-project-planning.ppt
chapter--4-software-project-planning.pptchapter--4-software-project-planning.ppt
chapter--4-software-project-planning.pptkotipi9215
Β 
The Evolution of Karaoke From Analog to App.pdf
The Evolution of Karaoke From Analog to App.pdfThe Evolution of Karaoke From Analog to App.pdf
The Evolution of Karaoke From Analog to App.pdfPower Karaoke
Β 
Cloud Management Software Platforms: OpenStack
Cloud Management Software Platforms: OpenStackCloud Management Software Platforms: OpenStack
Cloud Management Software Platforms: OpenStackVICTOR MAESTRE RAMIREZ
Β 
cybersecurity notes for mca students for learning
cybersecurity notes for mca students for learningcybersecurity notes for mca students for learning
cybersecurity notes for mca students for learningVitsRangannavar
Β 
What is Fashion PLM and Why Do You Need It
What is Fashion PLM and Why Do You Need ItWhat is Fashion PLM and Why Do You Need It
What is Fashion PLM and Why Do You Need ItWave PLM
Β 
EY_Graph Database Powered Sustainability
EY_Graph Database Powered SustainabilityEY_Graph Database Powered Sustainability
EY_Graph Database Powered SustainabilityNeo4j
Β 
Unit 1.1 Excite Part 1, class 9, cbse...
Unit 1.1 Excite Part 1, class 9, cbse...Unit 1.1 Excite Part 1, class 9, cbse...
Unit 1.1 Excite Part 1, class 9, cbse...aditisharan08
Β 
The Essentials of Digital Experience Monitoring_ A Comprehensive Guide.pdf
The Essentials of Digital Experience Monitoring_ A Comprehensive Guide.pdfThe Essentials of Digital Experience Monitoring_ A Comprehensive Guide.pdf
The Essentials of Digital Experience Monitoring_ A Comprehensive Guide.pdfkalichargn70th171
Β 
Building a General PDE Solving Framework with Symbolic-Numeric Scientific Mac...
Building a General PDE Solving Framework with Symbolic-Numeric Scientific Mac...Building a General PDE Solving Framework with Symbolic-Numeric Scientific Mac...
Building a General PDE Solving Framework with Symbolic-Numeric Scientific Mac...stazi3110
Β 
What is Binary Language? Computer Number Systems
What is Binary Language?  Computer Number SystemsWhat is Binary Language?  Computer Number Systems
What is Binary Language? Computer Number SystemsJheuzeDellosa
Β 
Unveiling the Tech Salsa of LAMs with Janus in Real-Time Applications
Unveiling the Tech Salsa of LAMs with Janus in Real-Time ApplicationsUnveiling the Tech Salsa of LAMs with Janus in Real-Time Applications
Unveiling the Tech Salsa of LAMs with Janus in Real-Time ApplicationsAlberto GonzΓ‘lez Trastoy
Β 
Short Story: Unveiling the Reasoning Abilities of Large Language Models by Ke...
Short Story: Unveiling the Reasoning Abilities of Large Language Models by Ke...Short Story: Unveiling the Reasoning Abilities of Large Language Models by Ke...
Short Story: Unveiling the Reasoning Abilities of Large Language Models by Ke...kellynguyen01
Β 
Professional Resume Template for Software Developers
Professional Resume Template for Software DevelopersProfessional Resume Template for Software Developers
Professional Resume Template for Software DevelopersVinodh Ram
Β 
BATTLEFIELD ORM: TIPS, TACTICS AND STRATEGIES FOR CONQUERING YOUR DATABASE
BATTLEFIELD ORM: TIPS, TACTICS AND STRATEGIES FOR CONQUERING YOUR DATABASEBATTLEFIELD ORM: TIPS, TACTICS AND STRATEGIES FOR CONQUERING YOUR DATABASE
BATTLEFIELD ORM: TIPS, TACTICS AND STRATEGIES FOR CONQUERING YOUR DATABASEOrtus Solutions, Corp
Β 
DNT_Corporate presentation know about us
DNT_Corporate presentation know about usDNT_Corporate presentation know about us
DNT_Corporate presentation know about usDynamic Netsoft
Β 

Recently uploaded (20)

Der Spagat zwischen BIAS und FAIRNESS (2024)
Der Spagat zwischen BIAS und FAIRNESS (2024)Der Spagat zwischen BIAS und FAIRNESS (2024)
Der Spagat zwischen BIAS und FAIRNESS (2024)
Β 
(Genuine) Escort Service Lucknow | Starting β‚Ή,5K To @25k with A/C πŸ§‘πŸ½β€β€οΈβ€πŸ§‘πŸ» 89...
(Genuine) Escort Service Lucknow | Starting β‚Ή,5K To @25k with A/C πŸ§‘πŸ½β€β€οΈβ€πŸ§‘πŸ» 89...(Genuine) Escort Service Lucknow | Starting β‚Ή,5K To @25k with A/C πŸ§‘πŸ½β€β€οΈβ€πŸ§‘πŸ» 89...
(Genuine) Escort Service Lucknow | Starting β‚Ή,5K To @25k with A/C πŸ§‘πŸ½β€β€οΈβ€πŸ§‘πŸ» 89...
Β 
Call Girls In Mukherjee Nagar πŸ“± 9999965857 🀩 Delhi 🫦 HOT AND SEXY VVIP 🍎 SE...
Call Girls In Mukherjee Nagar πŸ“±  9999965857  🀩 Delhi 🫦 HOT AND SEXY VVIP 🍎 SE...Call Girls In Mukherjee Nagar πŸ“±  9999965857  🀩 Delhi 🫦 HOT AND SEXY VVIP 🍎 SE...
Call Girls In Mukherjee Nagar πŸ“± 9999965857 🀩 Delhi 🫦 HOT AND SEXY VVIP 🍎 SE...
Β 
Russian Call Girls in Karol Bagh Aasnvi ➑️ 8264348440 πŸ’‹πŸ“ž Independent Escort S...
Russian Call Girls in Karol Bagh Aasnvi ➑️ 8264348440 πŸ’‹πŸ“ž Independent Escort S...Russian Call Girls in Karol Bagh Aasnvi ➑️ 8264348440 πŸ’‹πŸ“ž Independent Escort S...
Russian Call Girls in Karol Bagh Aasnvi ➑️ 8264348440 πŸ’‹πŸ“ž Independent Escort S...
Β 
Steps To Getting Up And Running Quickly With MyTimeClock Employee Scheduling ...
Steps To Getting Up And Running Quickly With MyTimeClock Employee Scheduling ...Steps To Getting Up And Running Quickly With MyTimeClock Employee Scheduling ...
Steps To Getting Up And Running Quickly With MyTimeClock Employee Scheduling ...
Β 
chapter--4-software-project-planning.ppt
chapter--4-software-project-planning.pptchapter--4-software-project-planning.ppt
chapter--4-software-project-planning.ppt
Β 
The Evolution of Karaoke From Analog to App.pdf
The Evolution of Karaoke From Analog to App.pdfThe Evolution of Karaoke From Analog to App.pdf
The Evolution of Karaoke From Analog to App.pdf
Β 
Cloud Management Software Platforms: OpenStack
Cloud Management Software Platforms: OpenStackCloud Management Software Platforms: OpenStack
Cloud Management Software Platforms: OpenStack
Β 
cybersecurity notes for mca students for learning
cybersecurity notes for mca students for learningcybersecurity notes for mca students for learning
cybersecurity notes for mca students for learning
Β 
What is Fashion PLM and Why Do You Need It
What is Fashion PLM and Why Do You Need ItWhat is Fashion PLM and Why Do You Need It
What is Fashion PLM and Why Do You Need It
Β 
EY_Graph Database Powered Sustainability
EY_Graph Database Powered SustainabilityEY_Graph Database Powered Sustainability
EY_Graph Database Powered Sustainability
Β 
Unit 1.1 Excite Part 1, class 9, cbse...
Unit 1.1 Excite Part 1, class 9, cbse...Unit 1.1 Excite Part 1, class 9, cbse...
Unit 1.1 Excite Part 1, class 9, cbse...
Β 
The Essentials of Digital Experience Monitoring_ A Comprehensive Guide.pdf
The Essentials of Digital Experience Monitoring_ A Comprehensive Guide.pdfThe Essentials of Digital Experience Monitoring_ A Comprehensive Guide.pdf
The Essentials of Digital Experience Monitoring_ A Comprehensive Guide.pdf
Β 
Building a General PDE Solving Framework with Symbolic-Numeric Scientific Mac...
Building a General PDE Solving Framework with Symbolic-Numeric Scientific Mac...Building a General PDE Solving Framework with Symbolic-Numeric Scientific Mac...
Building a General PDE Solving Framework with Symbolic-Numeric Scientific Mac...
Β 
What is Binary Language? Computer Number Systems
What is Binary Language?  Computer Number SystemsWhat is Binary Language?  Computer Number Systems
What is Binary Language? Computer Number Systems
Β 
Unveiling the Tech Salsa of LAMs with Janus in Real-Time Applications
Unveiling the Tech Salsa of LAMs with Janus in Real-Time ApplicationsUnveiling the Tech Salsa of LAMs with Janus in Real-Time Applications
Unveiling the Tech Salsa of LAMs with Janus in Real-Time Applications
Β 
Short Story: Unveiling the Reasoning Abilities of Large Language Models by Ke...
Short Story: Unveiling the Reasoning Abilities of Large Language Models by Ke...Short Story: Unveiling the Reasoning Abilities of Large Language Models by Ke...
Short Story: Unveiling the Reasoning Abilities of Large Language Models by Ke...
Β 
Professional Resume Template for Software Developers
Professional Resume Template for Software DevelopersProfessional Resume Template for Software Developers
Professional Resume Template for Software Developers
Β 
BATTLEFIELD ORM: TIPS, TACTICS AND STRATEGIES FOR CONQUERING YOUR DATABASE
BATTLEFIELD ORM: TIPS, TACTICS AND STRATEGIES FOR CONQUERING YOUR DATABASEBATTLEFIELD ORM: TIPS, TACTICS AND STRATEGIES FOR CONQUERING YOUR DATABASE
BATTLEFIELD ORM: TIPS, TACTICS AND STRATEGIES FOR CONQUERING YOUR DATABASE
Β 
DNT_Corporate presentation know about us
DNT_Corporate presentation know about usDNT_Corporate presentation know about us
DNT_Corporate presentation know about us
Β 

FP Triad of Folding, Scanning and Iteration

  • 1. fold Ξ» The Func%onal Programming Triad of Folding, Scanning and Itera%on a first example in Scala and Haskell Polyglot FP for Fun and Profit @philip_schwarz slides by h"ps://www.slideshare.net/pjschwarz Richard Bird Sergei Winitzki sergei-winitzki-11a6431 http://www.cs.ox.ac.uk/people/richard.bird/
  • 2. This slide deck can work both as an aide mΓ©moire (memory jogger), or as a first (not completely trivial) example of using le/ folds, le/ scans and itera5on to implement mathema5cal induc5on. We first look at the implementaAon, using a le/ fold, of a digits-to-int funcAon that converts a sequence of digits into a whole number. Then we look at the implementaAon, using the iterate funcAon, of an int-to- digits funcAon that converts an integer into a sequence of digits. In Scala this funcAon is very readable because the signatures of funcAons provided by collec5ons permit the piping of such funcAons with zero syntac5c overhead. In Haskell, there is some syntac5c sugar that can be used to achieve the same readability, so we look at how that works. We then set ourselves a simple task involving digits-to-int and int-to-digits, and write a funcAon whose logic can be simplified with the introducAon of a le/ scan. Let’s begin, on the next slide, by looking at how Richard Bird describes the digits- to-int funcAon, which he calls π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™. @philip_schwarz
  • 3. Suppose we want a funcAon decimal that takes a list of digits and returns the corresponding decimal number; thus π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™ [π‘₯0, π‘₯1, … , π‘₯n] = βˆ‘!"# $ π‘₯ π‘˜10($&!) It is assumed that the most significant digit comes first in the list. One way to compute decimal efficiently is by a process of mulAplying each digit by ten and adding in the following digit. For example π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™ π‘₯0, π‘₯1, π‘₯2 = 10 Γ— 10 Γ— 10 Γ— 0 + π‘₯0 + π‘₯1 + π‘₯2 This decomposiAon of a sum of powers is known as Horner’s rule. Suppose we define βŠ• by 𝑛 βŠ• π‘₯ = 10 Γ— 𝑛 + π‘₯. Then we can rephrase the above equaAon as π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™ π‘₯0, π‘₯1, π‘₯2 = (0 βŠ• π‘₯0) βŠ• π‘₯1 βŠ• π‘₯2 This example moAvates the introducAon of a second fold operator called π‘“π‘œπ‘™π‘‘π‘™ (pronounced β€˜fold leM’). Informally: π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, … , π‘₯𝑛 βˆ’ 1 = … ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) … βŠ• π‘₯ 𝑛 βˆ’ 1 The parentheses group from the leM, which is the reason for the name. The full definiAon of π‘“π‘œπ‘™π‘‘π‘™ is π‘“π‘œπ‘™π‘‘π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ 𝛽 π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 = 𝑒 π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠 Richard Bird
  • 4. On the next slide we look at how Sergei Winitzki describes the digits-to-int funcAon, and how he implements it in Scala.
  • 5. Example 2.2.5.3 Implement the funcAon digits-to-int using foldLeft. … The required computaAon can be wriTen as the formula π‘Ÿ = @ !"# $&( 𝑑 π‘˜ βˆ— 10$&(&!. … Solu5on The induc5ve defini5on of digitsToInt β€’ For an empty sequence of digits, Seq(), the result is 0. This is a convenient base case, even if we never call digitsToInt on an empty sequence. β€’ If digitsToInt(xs) is already known for a sequence xs of digits, and we have a sequence xs :+ x with one more digit x, then is directly translated into code: def digitsToInt(d: Seq[Int]): Int = d.foldLeft(0){ (n, x) => n * 10 + x } Sergei Winitzki sergei-winitzki-11a6431
  • 6. (βŠ•) :: Int -> Int -> Int (βŠ•) n d = 10 * n + d digits_to_int :: [Int] -> Int digits_to_int = foldl (βŠ•) 0 def `(βŠ•)`(n: Int, d: Int): Int = 10 * n + d def digitsToInt(xs: Seq[Int]): Int = xs.foldLeft(0)(`(βŠ•)`) assert( digitsToInt(List(5,3,7,4)) == 5374) implicit class Sugar(m: Int){ def βŠ•(n: Int) = `(βŠ•)`(m,n) } assert( (150 βŠ• 7) == 1507 ) π‘“π‘œπ‘™π‘‘π‘™ + 0 1,2,3 = 6π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, … , π‘₯𝑛 = … ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) … βŠ• π‘₯ 𝑛 π‘“π‘œπ‘™π‘‘π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ 𝛽 π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 = 𝑒 π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠 π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2 = π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 βŠ• π‘₯0 π‘₯1, π‘₯2 = π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 βŠ• π‘₯0 βŠ• π‘₯1 π‘₯2 = π‘“π‘œπ‘™π‘‘π‘™ βŠ• (( 𝑒 βŠ• π‘₯0 βŠ• π‘₯1) βŠ• π‘₯2) [ ] = ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2 π‘“π‘œπ‘™π‘‘π‘™ βŠ• 0 1,2,3 = π‘“π‘œπ‘™π‘‘π‘™ βŠ• 0 βŠ• 1 2,3 = π‘“π‘œπ‘™π‘‘π‘™ βŠ• 0 βŠ• 1 βŠ• 2 3 = π‘“π‘œπ‘™π‘‘π‘™ βŠ• (( 0 βŠ• 1 βŠ• 2) βŠ• 3) [ ] = ((0 βŠ• 1) βŠ• 2) βŠ• 3 Here is a quick refresher on fold le/ So let’s implement the digits-to-int funcAon in Haskell. And now in Scala. π‘‘π‘’π‘π‘–π‘šπ‘Žπ‘™ [π‘₯0, π‘₯1, … , π‘₯n] = @ !"# $ π‘₯ π‘˜10($&!)
  • 7. Now we turn to int-to-digits, a funcAon that is the opposite of digits-to-int, and one that can be implemented using the iterate funcAon. In the next two slides, we look at how Sergei Winitzki describes digits-to-int (which he calls digitsOf) and how he implements it in Scala.
  • 8. 2.3 Converting a single value into a sequence An aggregation converts (β€œfolds”) a sequence into a single value; the opposite operation (β€œunfolding”) converts a single value into a sequence. An example of this task is to compute the sequence of decimal digits for a given integer: def digitsOf(x: Int): Seq[Int] = ??? scala> digitsOf(2405) res0: Seq[Int] = List(2, 4, 0, 5) We cannot implement digitsOf using map, zip, or foldLeft, because these methods work only if we already have a sequence; but the function digitsOf needs to create a new sequence. … To figure out the code for digitsOf, we first write this function as a mathematical formula. To compute the digits for, say, n = 2405, we need to divide n repeatedly by 10, getting a sequence nk of intermediate numbers (n0 = 2405, n1 = 240, ...) and the corresponding sequence of last digits, nk mod 10 (in this example: 5, 0, ...). The sequence nk is defined using mathematical induction: β€’ Base case: n0 = n, where n is the given initial integer. β€’ Inductive step: nk+1 = nk 10 for k = 1, 2, ... Here nk 10 is the mathematical notation for the integer division by 10. Let us tabulate the evaluation of the sequence nk for n = 2405: The numbers nk will remain all zeros after k = 4. It is clear that the useful part of the sequence is before it becomes all zeros. In this example, the sequence nk needs to be stopped at k = 4. The sequence of digits then becomes [5, 0, 4, 2], and we need to reverse it to obtain [2, 4, 0, 5]. For reversing a sequence, the Scala library has the standard method reverse. Sergei Winitzki sergei-winitzki-11a6431
  • 9. So, a complete implementa.on for digitsOf is: def digitsOf(n: Int): Seq[Int] = if (n == 0) Seq(0) else { // n == 0 is a special case. Stream.iterate(n) { nk => nk / 10 } .takeWhile { nk => nk != 0 } .map { nk => nk % 10 } .toList.reverse } We can shorten the code by using the syntax such as (_ % 10) instead of { nk => nk % 10 }, def digitsOf(n: Int): Seq[Int] = if (n == 0) Seq(0) else { // n == 0 is a special case. Stream.iterate(n) (_ / 10 ) .takeWhile ( _ != 0 ) .map ( _ % 10 ) .toList.reverse } Sergei Winitzki sergei-winitzki-11a6431 The Scala library has a general stream-producing func8on Stream.iterate. This func8on has two arguments, the ini8al value and a func8on that computes the next value from the previous one: … The type signature of the method Stream.iterate can be wri?en as def iterate[A](init: A)(next: A => A): Stream[A] and shows a close correspondence to a defini8on by mathema8cal induc8on. The base case is the first value, init, and the induc8ve step is a func8on, next, that computes the next element from the previous one. It is a general way of crea8ng sequences whose length is not determined in advance.
  • 10. the prelude funcAon iterate returns an infinite list: π‘–π‘‘π‘’π‘Ÿπ‘Žπ‘‘π‘’ ∷ π‘Ž β†’ π‘Ž β†’ π‘Ž β†’ π‘Ž π‘–π‘‘π‘’π‘Ÿπ‘Žπ‘‘π‘’ 𝑓 π‘₯ = π‘₯ ∢ π‘–π‘‘π‘’π‘Ÿπ‘Žπ‘‘π‘’ 𝑓 (𝑓 π‘₯) In parAcular, π‘–π‘‘π‘’π‘Ÿπ‘Žπ‘‘π‘’ +1 1 is an infinite list of the posiAve integers, a value we can also write as [1..]. Richard Bird Here is how Richard Bird describes the iterate funcAon int_to_digits :: Int -> [Int] int_to_digits n = reverse ( map (n -> mod n 10) ( takeWhile (n -> n /= 0) ( iterate (n -> div n 10) n ) ) ) import LazyList.iterate def intToDigits(n: Int): Seq[Int] = iterate(n) (_ / 10 ) .takeWhile ( _ != 0 ) .map ( _ % 10 ) .toList .reverse Here on the leM I had a go at a Haskell implementaAon of the int-to-digits funcAon (we are not handling cases like n=0 or negaAve n), and on the right, the same logic in Scala. I find the Scala version slightly easier to understand, because the funcAons that we are calling appear in the order in which they are executed. Contrast that with the Haskell version, in which the funcAon invocaAons occur in the opposite order.
  • 11. While the β€˜ο¬‚uent’ chaining of funcAon calls on Scala collecAons is very convenient, when using other funcAons, we face the same problem that we saw in Haskell on the previous slide. e.g. in the following code, square appears first, even though it is executed last, and vice versa for inc. assert(square(twice(inc(3))) == 64) assert ((3 pipe inc pipe twice pipe square) == 64) To help a bit with that problem, in Scala there is a pipe funcAon which is available on all values, and which allows us to order funcAon invocaAons in the β€˜right’ order. Armed with pipe, we can rewrite the code so that funcAon names occur in the same order in which the funcAons are invoked, which makes the code more understandable. def inc(n: Int): Int = n + 1 def twice(n: Int): Int = n * 2 def square(n: Int): Int = n * n 3 pipe inc pipe twice pipe square square(twice(inc(3)) @philip_schwarz
  • 12. What about in Haskell? First of all, in Haskell there is a func8on applica8on operator called $, which we can some.mes use to omit parentheses Application operator. This operator is redundant, since ordinary application (f x) means the same as (f $ x). However, $ has low, right-associative binding precedence, so it sometimes allows parentheses to be omitted; for example: f $ g $ h x = f (g (h x)) int_to_digits :: Int -> [Int] int_to_digits n = reverse ( map (n -> mod n 10) ( takeWhile (n -> n /= 0) ( iterate (n -> div n 10) n ) ) ) int_to_digits :: Int -> [Int] int_to_digits n = reverse $ map (x -> mod x 10) $ takeWhile (x -> x > 0) $ iterate (x -> div x 10) $ n Armed with $, we can simplify our func.on as follows simplify For beginners, the $ oVen makes Haskell code more difficult to parse. In pracXce, the $ operator is used frequently, and you’ll likely find you prefer using it over many parentheses. There’s nothing magical about $; if you look at its type signature, you can see how it works: ($) :: (a -> b) -> a -> b The arguments are just a funcXon and a value. The trick is that $ is a binary operator, so it has lower precedence than the other funcXons you’re using. Therefore, the argument for the funcXon will be evaluated as though it were in parentheses.
  • 13. But there is more we can do. In addiAon to the func5on applica5on operator $, in Haskell there is also a reverse func5on applica5on operator &. hTps://www.fpcomplete.com/haskell/tutorial/operators/
  • 14. int_to_digits :: Int -> [Int] int_to_digits n = reverse $ map (x -> mod x 10) $ takeWhile (x -> x > 0) $ iterate (x -> div x 10) $ n int_to_digits :: Int -> [Int] int_to_digits s n = n & iterate (x -> div x 10) & takeWhile (x -> x > 0) & map (x -> mod x 10) & reverse def intToDigits(n: Int): Seq[Int] = iterate(n) (_ / 10 ) .takeWhile ( _ != 0 ) .map ( _ % 10 ) .toList .reverse Thanks to the & operator, we can rearrange our int_to_digits funcAon so that it is as readable as the Scala version.
  • 15. There is one school of thought according to which the choice of names for $ and & make Haskell hard to read for newcomers, that it is beTer if $ and & are instead named |> and <|. Flow provides operators for writing more understandable Haskell. It is an alternative to some common idioms like ($) for function application and (.) for function composition. … Rationale I think that Haskell can be hard to read. It has two operators for applying functions. Both are not really necessary and only serve to reduce parentheses. But they make code hard to read. People who do not already know Haskell have no chance of guessing what foo $ bar or baz & qux mean. … I think we can do better. By using directional operators, we can allow readers to move their eye in only one direction, be that left-to-right or right-to-left. And by using idioms common in other programming languages, we can allow people who aren't familiar with Haskell to guess at the meaning. So instead of ($), I propose (<|). It is a pipe, which anyone who has touched a Unix system should be familiar with. And it points in the direction it sends arguments along. Similarly, replace (&) with (|>). … square $ twice $ inc $ 3 square <| twice <| inc <| 3 3 & inc & twice & square 3 |> inc |> twice |> square Here is an example of how |> and <| improve readability
  • 16. Since & is just the reverse of $, we can define |> ourselves simply by flipping $ Ξ» "left" ++ "right" "leftright” Ξ» Ξ» (##) = flip (++) Ξ» Ξ» "left" ## "right" "rightleft" Ξ» Ξ» inc n = n + 1 Ξ» twice n = n * 2 Ξ» square n = n * n Ξ» Ξ» square $ twice $ inc $ 3 64 Ξ» Ξ» (|>) = flip ($) Ξ» Ξ» 3 |> inc |> twice |> square 64 Ξ» int_to_digits :: Int -> [Int] int_to_digits s n = n |> iterate (x -> div x 10) |> takeWhile (x -> x > 0) |> map (x -> mod x 10) |> reverse And here is how our funcAon looks using |>. @philip_schwarz
  • 17. Now let’s set ourselves the following task. Given a posiAve integer N with n digits, e.g. the five-digit number 12345, we want to compute the following: [(0,0),(1,1),(12,3),(123,6),(1234,10),(12345,15)] i.e. we want to compute a list of pairs p0 , p1 , … , pn with pk being ( Nk , Nk 𝛴 ), where Nk is the integer number formed by the first k digits of N, and Nk𝛴 is the sum of those digits. We can use our int_to_digits funcAon to convert N into its digits d1 , d2 , … , dn : Ξ» int_to_digits 12345 [1,2,3,4,5] Ξ» And we can use digits_to_int to turn digits d1 , d2 , … , dk into Nk , e.g. for k = 3 : Ξ» digits_to_int [1,2,3] 123 Ξ» How can we generate the following sequences of digits ? [ [ ] , [d1 ] , [d1 , d2 ] , [d1 , d2 , d3 ] , … , [d1 , d2 , d3 , … , dn ] ] As we’ll see on the next slide, that is exactly what the inits funcAon produces when passed [d1 , d2 , d3 , … , dn ] !
  • 18. 𝑖𝑛𝑖𝑑𝑠 π‘₯0, π‘₯1, π‘₯2 = [[ ], π‘₯0 , π‘₯0, π‘₯1 , π‘₯0, π‘₯1, π‘₯2 ] 𝑖𝑛𝑖𝑑𝑠 ∷ Ξ± β†’ Ξ± 𝑖𝑛𝑖𝑑𝑠 = [ ] 𝑖𝑛𝑖𝑑𝑠 π‘₯: π‘₯𝑠 = ∢ π‘šπ‘Žπ‘ π‘₯: (𝑖𝑛𝑖𝑑𝑠 π‘₯𝑠) Here is a definiAon for inits. So we can apply inits to [d1 , d2 , d3 , … , dn ] to generate the following: [ [ ] , [d1 ] , [d1 , d2 ] , [d1 , d2 , d3 ] , … , [d1 , d2 , d3 , … , dn ] ] e.g. Ξ» inits [1,2,3,4] [[],[1],[1,2],[1,2,3],[1,2,3,4]]] Ξ» And here is what inits produces, i.e. the list of all ini5al segments of a list.
  • 19. So if we map digits_to_int over the iniXal segments of [d1 , d2 , d3 , … , dn ], i.e [ [ ] , [d1 ] , [d1 , d2 ] , [d1 , d2 , d3 ] , … , [d1 , d2 , d3 , … , dn ] ] we obtain a list containing N0 , N1 , … , Nn , e.g. Ξ» map digits_to_int (inits [1,2,3,4,5])) [0,1,12,123,1234,12345]] Ξ» What we need now is a funcXon digits_to_sum which is similar to digits_to_int but which instead of converXng a list of digits [d1 , d2 , d3 , … , dk ] into Nk , i.e. the number formed by those digits, it turns the list into Nk𝛴, i.e. the sum of the digits. Like digits_to_int, the digits_to_sum funcXon can be defined using a leB fold: digits_to_sum :: [Int] -> Int digits_to_sum = foldl (+) 0 Let’s try it out: Ξ» map digits_to_sum [1,2,3,4]) 1234] Ξ» Now if we map digits_to_sum over the iniXal segments of [d1 , d2 , d3 , … , dn ], we obtain a list containing N0 𝛴 , N1 𝛴 , … , Nn 𝛴 , e.g. Ξ» map digits_to_sum (inits [1,2,3,4,5])) [0,1,12,123,1234,12345]] Ξ» (βŠ•) :: Int -> Int -> Int (βŠ•) n d = 10 * n + d digits_to_int :: [Int] -> Int digits_to_int = foldl (βŠ•) 0
  • 20. digits_to_sum :: [Int] -> Int digits_to_sum = foldl (+) 0 (βŠ•) :: Int -> Int -> Int (βŠ•) n d = 10 * n + d digits_to_int :: [Int] -> Int digits_to_int = foldl (βŠ•) 0 convert :: Int -> [(Int,Int)] convert n = zip nums sums where nums = map digits_to_int segments sums = map digits_to_sum segments segments = inits (int_to_digits n) int_to_digits :: Int -> [Int] int_to_digits s n = n |> iterate (x -> div x 10) |> takeWhile (x -> x > 0) |> map (x -> mod x 10) |> reverse Ξ» convert 12345 [(0,0),(1,1),(12,3),(123,6),(1234,10),(12345,15)] Ξ» So here is how our complete program looks at the moment. @philip_schwarz
  • 21. What we are going to do next is see how, by using the scan le/ funcAon, we are able to simplify the definiAon of convert which we just saw on the previous slide. As a quick refresher of (or introducAon to) the scan le/ funcAon, in the next two slides we look at how Richard Bird describes the funcAon.
  • 22. 4.5.2 Scan le/ SomeXmes it is convenient to apply a π‘“π‘œπ‘™π‘‘π‘™ operaXon to every iniXal segment of a list. This is done by a funcXon π‘ π‘π‘Žπ‘›π‘™ pronounced β€˜scan leB’. For example, π‘ π‘π‘Žπ‘›π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2 = [𝑒, 𝑒 βŠ• π‘₯0, (𝑒 βŠ• π‘₯0) βŠ• π‘₯1, ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2] In parXcular, π‘ π‘π‘Žπ‘›π‘™ + 0 computes the list of accumulated sums of a list of numbers, and π‘ π‘π‘Žπ‘›π‘™ Γ— 1 [1. . 𝑛] computes a list of the first 𝑛 factorial numbers. … We will give two programs for π‘ π‘π‘Žπ‘›π‘™; the first is the clearest, while the second is more efficient. For the first program we will need the funcXon inits that returns the list of all iniDal segments of a list. For Example, 𝑖𝑛𝑖𝑑𝑠 π‘₯0, π‘₯1, π‘₯2 = [[ ], π‘₯0 , π‘₯0, π‘₯1 , π‘₯0, π‘₯1, π‘₯2 ] The empty list has only one segment, namely the empty list itself; A list π‘₯: π‘₯𝑠 has the empty list as its shortest iniDal segment, and all the other iniDal segments begin with π‘₯ and are followed by an iniDal segment of π‘₯𝑠. Hence 𝑖𝑛𝑖𝑑𝑠 ∷ Ξ± β†’ Ξ± 𝑖𝑛𝑖𝑑𝑠 = [ ] 𝑖𝑛𝑖𝑑𝑠 π‘₯: π‘₯𝑠 = ∢ π‘šπ‘Žπ‘ (π‘₯: )(𝑖𝑛𝑖𝑑𝑠 π‘₯𝑠) The funcXon 𝑖𝑛𝑖𝑑𝑠 can be defined more succinctly as an instance of π‘“π‘œπ‘™π‘‘π‘Ÿ : 𝑖𝑛𝑖𝑑𝑠 = π‘“π‘œπ‘™π‘‘π‘Ÿ 𝑓 [ ] π‘€β„Žπ‘’π‘Ÿπ‘’ 𝑓 π‘₯ π‘₯𝑠𝑠 = ∢ π‘šπ‘Žπ‘ π‘₯: π‘₯𝑠𝑠 Now we define π‘ π‘π‘Žπ‘›π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ [𝛽] π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = π‘šπ‘Žπ‘ π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 . 𝑖𝑛𝑖𝑑𝑠 This is the clearest definiXon of π‘ π‘π‘Žπ‘›π‘™ but it leads to an inefficient program. The funcXon 𝑓 is applied k Xmes in the evaluaXon of Richard Bird
  • 23. π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 on a list of length k and, since the initial segments of a list of length n are lists with lengths 0,1,…,n, the function 𝑓 is applied about n2/2 times in total. Let us now synthesise a more efficient program. The synthesis is by an induction argument on π‘₯𝑠 so we lay out the calculation in the same way. <…not shown…> In summary, we have derived π‘ π‘π‘Žπ‘›π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ [𝛽] π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = [𝑒] π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = 𝑒 ∢ π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠 This program is more efficient in that function 𝑓 is applied exactly n times on a list of length n. Richard Bird π‘ π‘π‘Žπ‘›π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2 ⬇ [𝑒, 𝑒 βŠ• π‘₯0, (𝑒 βŠ• π‘₯0) βŠ• π‘₯1, ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2] π‘“π‘œπ‘™π‘‘π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ 𝛽 π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 = 𝑒 π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠 Note the similariXes and differences between π‘ π‘π‘Žπ‘›π‘™ and π‘“π‘œπ‘™π‘‘π‘™, e.g. the leV hand sides of their equaXons are the same, and their signatures are very similar, but π‘ π‘π‘Žπ‘›π‘™ returns [𝛽] rather than 𝛽 and while π‘“π‘œπ‘™π‘‘π‘™ is tail recursive, π‘ π‘π‘Žπ‘›π‘™ isn’t. π‘ π‘π‘Žπ‘›π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ [𝛽] π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = [𝑒] π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = 𝑒 ∢ π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠 π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2 ⬇ ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2
  • 24. On the next slide, a very simple example of using scanl, and a reminder of how the result of scanl relates to the result of inits and foldl.
  • 25. 𝑖𝑛𝑖𝑑𝑠 2,3,4 = [[ ], 2 , 2,3 , 2,3,4 ] π‘ π‘π‘Žπ‘›π‘™ + 0 2,3,4 = [0, 2, 5, 9] π‘“π‘œπ‘™π‘‘π‘™ + 0 [ ] =0 π‘“π‘œπ‘™π‘‘π‘™ + 0 2 = 2 π‘“π‘œπ‘™π‘‘π‘™ + 0 2,3 = 5 π‘“π‘œπ‘™π‘‘π‘™ + 0 2,3,4 = 9 π‘ π‘π‘Žπ‘›π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2 ⬇ [𝑒, 𝑒 βŠ• π‘₯0, (𝑒 βŠ• π‘₯0) βŠ• π‘₯1, ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2] 𝑖𝑛𝑖𝑑𝑠 π‘₯0, π‘₯1, π‘₯2 = [[ ], π‘₯0 , π‘₯0, π‘₯1 , π‘₯0, π‘₯1, π‘₯2 ] π‘ π‘π‘Žπ‘›π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ [𝛽] π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = π‘šπ‘Žπ‘ π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 . 𝑖𝑛𝑖𝑑𝑠 π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, … , π‘₯𝑛 βˆ’ 1 = … ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) … βŠ• π‘₯ 𝑛 βˆ’ 1 π‘“π‘œπ‘™π‘‘π‘™ ∷ 𝛽 β†’ 𝛼 β†’ 𝛽 β†’ 𝛽 β†’ 𝛼 β†’ 𝛽 π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 = 𝑒 π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 π‘₯: π‘₯𝑠 = π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑓 𝑒 π‘₯ π‘₯𝑠 π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 π‘₯0, π‘₯1, π‘₯2 = π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 βŠ• π‘₯0 π‘₯1, π‘₯2 = π‘“π‘œπ‘™π‘‘π‘™ βŠ• 𝑒 βŠ• π‘₯0 βŠ• π‘₯1 π‘₯2 = π‘“π‘œπ‘™π‘‘π‘™ βŠ• (( 𝑒 βŠ• π‘₯0 βŠ• π‘₯1) βŠ• π‘₯2) [ ] = ((𝑒 βŠ• π‘₯0) βŠ• π‘₯1) βŠ• π‘₯2 π‘ π‘π‘Žπ‘›π‘™ + 0 2,3,4 ⬇ [0, 0 + 2, 0 + 2 + 3, 0 + 2 + 3 + 4]
  • 26. π‘ π‘π‘Žπ‘›π‘™ 𝑓 𝑒 = π‘šπ‘Žπ‘ π‘“π‘œπ‘™π‘‘π‘™ 𝑓 𝑒 . 𝑖𝑛𝑖𝑑𝑠 convert :: Int -> [(Int,Int)] convert n = zip nums sums where nums = map digits_to_int segments sums = map digits_to_sum segments segments = inits (int_to_digits n) convert :: Int -> [(Int,Int)] convert n = zip nums sums where nums = map foldl (βŠ•) 0 segments sums = map foldl (+) 0 segments segments = inits (int_to_digits n) AVer that refresher of (introducXon to) the scanl funcXon, let’s see how it can help us simplify our definiXon of the convert funcXon. The first thing to do is to take the definiXon of convert and inline its invocaXons of digits_to_int and digits_to_sum: refactor Now let’s extract (int_to_digits n) into digits and inline segments. refactor convert :: Int -> [(Int,Int)] convert n = zip nums sums where nums = map foldl (βŠ•) 0 (inits digits) sums = map foldl (+) 0 (inits digits) digits = int_to_digits n convert :: Int -> [(Int,Int)] convert n = zip nums sums where nums = map foldl (βŠ•) 0 segments sums = map foldl (+) 0 segments segments = inits (int_to_digits n) convert :: Int -> [(Int,Int)] convert n = zip nums sums where nums = scanl (βŠ•) 0 digits sums = scanl (+) 0 digits digits = int_to_digits n convert :: Int -> [(Int,Int)] convert n = zip nums sums where nums = map foldl (βŠ•) 0 (inits digits) sums = map foldl (+) 0 (inits digits) digits = int_to_digits n refactor As we saw earlier, mapping foldl over the result of applying inits to a list, is just applying scanl to that list, so let’s simplify convert by calling scanl rather than mapping foldl.
  • 27. convert :: Int -> [(Int,Int)] convert n = zip nums sums where nums = scanl (βŠ•) 0 digits sums = scanl (+) 0 digits digits = int_to_digits n The suboptimal thing about our current definition of convert is that it does two left scans over the same list of digits. Can we refactor it to do a single scan? Yes, by using tupling, i.e. by changing the func.on that we pass to scanl from a func.on like (βŠ•) or (+), which computes a single result, to a func.on, let’s call it next, which uses those two func.ons to compute two results convert :: Int -> [(Int,Int)] convert n = scanl next (0, 0) digits where next (number, sum) digit = (number βŠ• digit, sum + digit) digits = int_to_digits n On the next slide, we inline digits and compare the resul.ng convert func.on with our ini.al version, which invoked scanl twice. @philip_schwarz
  • 28. convert :: Int -> [(Int,Int)] convert n = scanl next (0, 0) (int_to_digits n) where next (number, sum) digit = (number βŠ• digit, sum + digit) convert :: Int -> [(Int,Int)] convert n = zip nums sums where nums = map digits_to_int segments sums = map digits_to_sum segments segments = inits (int_to_digits n) Here is our first definiAon of convert And here is our refactored version, which uses a left scan. The next slide shows the complete Haskell program, and next to it, the equivalent Scala program.
  • 29. digits_to_sum :: [Int] -> Int digits_to_sum = foldl (+) 0 (βŠ•) :: Int -> Int -> Int (βŠ•) n d = 10 * n + d int_to_digits :: Int -> [Int] int_to_digits s n = n |> iterate (x -> div x 10) |> takeWhile (x -> x > 0) |> map (x -> mod x 10) |> reverse Ξ» convert 12345 [(0,0),(1,1),(12,3),(123,6),(1234,10),(12345,15)] Ξ» convert :: Int -> [(Int,Int)] convert n = scanl next (0, 0) (int_to_digits n) where next (number, sum) digit = (number βŠ• digit, sum + digit) def digitsToInt(ds: Seq[Int]): Int = ds.foldLeft(0)(`(βŠ•)`) implicit class Sugar(m: Int) { def βŠ•(n: Int) = `(βŠ•)`(m,n) } def digitsToSum(ds: Seq[Int]): Int = ds.foldLeft(0)(_+_) def intToDigits(n: Int): Seq[Int] = iterate(n) (_ / 10 ) .takeWhile ( _ != 0 ) .map ( _ % 10 ) .toList .reverse def convert(n: Int): Seq[(Int,Int)] = { val next: ((Int,Int),Int) => (Int,Int) = { case ((number, sum), digit) => (number βŠ• digit, sum + digit) } intToDigits(n).scanLeft((0,0))(next)} digits_to_int :: [Int] -> Int digits_to_int = foldl (βŠ•) 0 def `(βŠ•)`(n: Int, d: Int): Int = 10 * n + d scala> convert(12345) val res0…= List((0,0),(1,1),(12,3),(123,6),(1234,10),(12345,15)) scala> fold Ξ»
  • 30. In the next slide we conclude this deck with Sergei Winitzkiβ€˜s recap of how in func5onal programming we implement mathema5cal induc5on using folding, scanning and itera5on.
  • 31. Use arbitrary inducDve (i.e., recursive) formulas to: β€’ convert sequences to single values (aggregaDon or β€œfolding”); β€’ create new sequences from single values (β€œunfolding”); β€’ transform exisXng sequences into new sequences. Table 2.1: ImplemenXng mathemaDcal inducDon Table 2.1 shows Scala code implemenDng those tasks. IteraDve calculaDons are implemented by translaDng mathemaDcal inducDon directly into code. In the funcDonal programming paradigm, the programmer does not need to write any loops or use array indices. Instead, the programmer reasons about sequences as mathemaDcal values: β€œStarDng from this value, we get that sequence, then transform it into this other sequence,” etc. This is a powerful way of working with sequences, dicDonaries, and sets. Many kinds of programming errors (such as an incorrect array index) are avoided from the outset, and the code is shorter and easier to read than convenDonal code wriPen using loops. Sergei Winitzki sergei-winitzki-11a6431