Functional Programming on the JVM
Think about expressions and not instructions

Functional Programming on the JVM

Functional Programming is a programming paradigm which involves breaking down a "giant" functionality into smaller, composable, higher-order, pure functions.

  • Pure Functions have no side effect, i.e. doesn't change or depend on any mutable external state and can be run multiple times for the same input to get the same output.
  • Higher-Order Functions can take other functions as parameters and also return other functions.

In this article we will talk about the three basic things one needs to know to start with Functional Programming.

Mindset shift

Quoting Daniel from the podcast (link below)

To learn functional programming, start with a mindset shift where you think of code as "expressions" instead of "instructions".

In the world of imperative programming, a programmer thinks of a set of instructions to run, which is a lower level of abstraction. A programmer can very well work a higher level and not think "instructional".

For example: Lets say you want to print a List of numbers.

List<Integer> numbers = List.of(1,2,3,4,5,6,7,8,9);        

In the imperative programming world, you would think in terms of instructions.

  • create a variable to hold the index.
  • write a for loop and call a print statement inside the loop.
  • you also have to remember and increment the index to go to the next item in the list.

for(int i = 0; i < numbers.size() ; i++) {
  System.out.println(numbers.get(i));
}        

In the world of functional programming, you don't think about iteration, but you express yourself.

numbers.stream().forEach(number -> System.out.println(number);        

Please note that you just "expressed" what you wanted to do, but you didn't think about how the numbers are taken out of the list and given to the `println` function.

This mindset is very important so lets take another simple example.

"Given a list of numbers, return a list of even numbers"

In the imperative world you would think instructionally as following:

  • Create a new output list to store the list of even numbers to be returned.
  • Write a for loop to iterate over the input list
  • Check if the current number is even and store it in the output list.
  • Once the loops ends, return the output list.

Note how you created an output list even before you started the iteration, you did a bunch of mutations and dealt with low level details.

In the functional programming world, you would express yourself

return numbers.stream().filter(number -> number%2==0).collect(Collectors.toList());        

  • You just say, for each number "filter" on the given criteria.
  • And collect into a list (this gets even better with languages like Scala, for example just `numbers.filter(predicate)` would work)
  • You don't worry (care) about how the filtering is done, how the list is populated etc. These are low level details and distracts you from focusing on the core business logic that is "check if a number if even or not".

Ok, so this makes it easier to understand what is the mental shift that is required.

Immutability

The second important aspect of Functional programming is Immutability.

Quoting Daniel from the podcast (link below)

Immutability works wonders in Distributed Systems. Tracing modifications in your code is bad enough in a single threaded piece of code, but in the world of multi threaded applications and distributed systems its a nightmare.

Since there is NO mutation made to the data structures, there are no concurrency issues where two threads are modifying the same state differently. It is simpler to work with.

There is one common concern with Immutability though, that is higher number of allocations. But is that really a problem? Well, modern languages handles allocations smartly. For ex, Scala keeps the references to the used Objects, so you don't have to reallocate memory. For short-lived data structures, Garbage collectors help with reducing memory footprint. Garbage collectors solves these problems very nicely for majority of the applications.

Recursion

Another important aspect of Functional Programming is Recursion.

There is no such thing as "iteration" in Functional programming. The concept of iteration itself is "instructional", so you have to think of iteration as recursion instead, which is "expressional". But as we know with recursion, every function call puts a little stack frame on the stack and we all know Stack memory is limited. So for larger recursion calls (recursion over a million elements for example) it will throw Stack overflow exception.

Is it a problem in modern languages? Well not really. Tail recursion comes to the rescue.So if you call a function recursively, using Tail call optimization, the language runtime will reuse the stack frame by replacing the parameters instead of allocating a new stack frame. Since we are reusing the stack frame and not creating a new one for each recursive call, it doens't go out of Stack space.

The only catch is that you should be able to model your recursive function such that the last expression being evalauted is a recursive call the same function with just different parameters.

Tail recursive implementation of factorial function:

def factorial(n: Int): Int {
???@tailrec 
   def factorialInner(acc: Int, n: Int): Int = {
       if (n <= 1)
?????????acc
???????else
?????????factorialInner(n * acc, n - 1)
???}
??factorialInner(1, n)
}        

As you can see, the inner recursive function is marked as @tailrec so if the function isn't really tail recursive the compiler will give you an error. This adds a little safety mechanism to ensure your function is really tail recursive.

Understanding these three patterns of Functional Programming is very important and for a beginner it will need some practice to think in terms of "expressions" and remove the baggage of thinking instructionally.

I highly recommend watching the podcast with Daniel Cioc?rlan where he has explained the what, why and the how part of Functional Programming. It will give you a great (and the right) start to Functional programming.

Like, share and subscribe. Keep learning.

Cheers,

The GeekNarrator

Ankit Dwivedi

Software Engineer

1 年

Worth a read!

Solving problems using expression and not instructions. Never thought in this way. Interesting

Kartik S.

SDE @Amazon | GSoC @RedHat | Open Source and Coding Mentor |Ex @Nagarro|Ex @Coding Blocks|System Design Content Creator|20k+ linkedin followers|3 million views|open for collaborations

1 年

Very well explained, I love the way how the imperative vs functional programming is explained with examples.? Reach++

要查看或添加评论,请登录

Kaivalya Apte的更多文章

  • Cassandra 5.0 : ACID Transactions, Vector Search and much more

    Cassandra 5.0 : ACID Transactions, Vector Search and much more

    Introduction In a recent podcast discussion with Patrick McFadin, VP of Developer Relations at DataStax, we delved into…

    4 条评论
  • Capture the Change, you want to see in the database.

    Capture the Change, you want to see in the database.

    Change Data Capture As the name suggests, it is about capturing changes to your data. By capturing, I mean reacting to…

    14 条评论
  • Be Resilient - Humans and Servers

    Be Resilient - Humans and Servers

    What is Resiliency? Ability to recover quickly to a “normal” working state from degradations/problems/failures. Like in…

    4 条评论
  • Distributing SQL Databases Globally

    Distributing SQL Databases Globally

    There are several reasons, why you would want to distributed your database. Keep your data closer to your customers.

    9 条评论
  • Why is DynamoDB AWSome?

    Why is DynamoDB AWSome?

    What is DynamoDB? A cloud NoSQL database service that guarantees consistent performance at any scale. Consistent…

    4 条评论
  • Designing Instagram, Linkedin, Facebook like applications

    Designing Instagram, Linkedin, Facebook like applications

    Hey Everyone, Welcome to the first article of The GeekNarrator newsletter, I am excited to start this newsletter along…

  • 5 things I learned from Hack-Week

    5 things I learned from Hack-Week

    Last few days we spent on hacking things and building something interesting, useful and which could not be done as part…

  • Readable Code : Just like a fairy tale

    Readable Code : Just like a fairy tale

    Most software today is very much like an Egyptian pyramid with millions of bricks piled on top of each other, with no…

    7 条评论
  • Work smart(less), achieve big. After all you are an engineer.

    Work smart(less), achieve big. After all you are an engineer.

    In the current world, where IT industry is booming, people often miss the bigger picture.Everyone is running, to…

社区洞察

其他会员也浏览了