Clone the above repo and import into your favorite editor.
Why FP Matters
Eli Jordan
Game Plan
- What is functional programming
- Human problem solving
- Compositionality
- Modularity / Maintainability / Testability
What Is Functional Programming?
- Programming with pure functions
What Is Functional Programming?
- A pure function is a total mapping between two sets
- The output is determined entirely by its input.
- They are mathematical functions.
Consequences
- Evaluating an expression always results in the same answer.
- We can always inline or factor out a sub expression. Since it will yield the same result.
- This property is called referential transparency
Referential Transparency
val x = ${expr}
(x, x)
(${expr}, ${expr})
are these programs the same..?
It depends on ${expr}
Referential Transparency
val x = "hello"
(x, x)
("hello", "hello")
are these programs the same..?
Yes!
Referential Transparency
val x = println("hello")
(x, x)
(println("hello"), println("hello"))
are these programs the same..?
No!
Referential Transparency
- Every expression is either referentially transparent or it's not.
- If its not, its a side effect
- If it is, its a pure expression
Functional Programs
- Are built using pure expressions.
- Can be reasoned about using substitution.
- Are composed of smaller programs.
Human Problem Solving
Human Problem Solving
- Programs are executed on a computers CPU
- However, programming languages are for humans!
- So, programming languages and paradigms should
be optimised for the human factor
Miller's Law
"the number of objects an average human can hold in short-term memory is 7 ± 2"
- Wikipedia
Miller's Law
- This inherent limitation, is reflected in the way we solve problems.
- We decompose into smaller and smaller sub-problems, until there are 7 ± 2 objects required in the
solution.
- The solutions are then composed, to solve the larger problem.
Miller's Law
- For this to be effective in reducing our cognitive burden..
- We need to be able to predict the result of composition, based on only:
- The solutions that are being composed
- The semantics of the composition
- Otherwise, there can be emergent behaviour.
- Reason locally about the composition
This is
Compositionality
Compositionality
"the meaning of a complex expression is determined by the meanings of its constituent expressions and the rules used to combine them."
- Wikipedia
Compositionality
- If we have the pieces of the program that can be individually understood and we know how to glue them together then we understand the whole program.
- If we can solve problems with A, B and C individually, we can solve any problem that is any combination of A, B, C.
val reader = new BufferedReader(
new FileReader("my-file.txt")
)
var line = reader.readLine()
var count = 0
while(line != null) {
val words = line.split("\\s")
words.foreach { w =>
count += 1
}
line = reader.readLine()
}
reader.close()
println(count)
val path = Paths.get("my-file.txt")
readAll[IO](path, blockingEc, 4096)
.through(text.utf8Decode)
.through(text.lines)
.flatMap(s => Stream.emits(s.split("\\s")))
.map(_ => 1)
.fold(0)(_ + _)
.evalMap(c => IO(println(c)))
val path = Paths.get("my-file.txt")
val bytes = readAll[IO](path, blockingEc, 4096)
type Transform[-A, +B] = Pipe[IO, A, B]
val words: Transform[String, String] =
_.flatMap(s => Stream.emits(s.split("\\s")))
val ones: Transform[String, Int] =
_.map(_ => 1)
val sum: Transform[Int, Int] =
_.fold(0)(_ + _)
val print: Transform[Int, Unit] =
_.evalMap(x => IO(println(x)))
bytes
.through(text.utf8Decode)
.through(text.lines)
.through(words)
.through(ones)
.through(sum)
.through(print)
val transforms: Transform[Byte, Unit] =
text.utf8Decode[IO] andThen
text.lines andThen
words andThen
ones andThen
sum andThen
print
bytes.through(transforms)
Functional Programming
- Pure, mathematical functions like those used in functional programming, are compositional.
- So, functional programming is really the study of compositional software.
- Compositionality has massive benefits for our ability to reason about the systems we are building.
Modularity
- Compositional programs are easily composed, and recomposed.
- When requirements change and the software needs to be adpated, the compositonality makes this very easy.
- In imperative systems, there are often hidden dependencies and emergent properties that require non-local reasoning to understand. This makes changes harder.
Maintainability
- Compositional programs are easier to reason about, since they can be understood in small chunks.
- When a change is required in an unfamiliar area, you can be confident that your change will only have a local impact.
- A new team member can reason about a small part of the application, without needing a global understanding.
Testability
- The meaning of our programs are defined by the meanings of the components and the meaning of the composition.
- So to be confident that our software is working as expected
- Test each of the components in isolation
- Test that the components are assembled in the correct manner.
- Thats it!
Summary
- Functional programming is programming with referentially transparent functions.
- Functions that are not referentially transparent, are side effects.
- Referentially transparent functions are compositional
-
Compositionality makes it much easier for people to reason about complex programs.
- Therefore, FP matters 😃
References
- Composing Programs - https://www.youtube.com/watch?v=h8aPc8sji9Q
- FP with effects - https://www.youtube.com/watch?v=30q6BkBv5MY
- Category Theory Motivations - https://www.youtube.com/watch?v=I8LbkfSSR58
- Miller's Law - https://en.wikipedia.org/wiki/The_Magical_Number_Seven,_Plus_or_Minus_Two
Why FP Matters
By Eli Jordan
Why FP Matters
- 942