Beautiful World of Monads

Sergiy Yevtushenko
7 min readJul 14, 2020

--

Practical Introduction to Monads for Java Developers

Let me start with the disclaimer. The explanation below is in no way pretends to be precise or absolutely accurate from Functional Programming's perspective. Instead, I’m focusing on clarity and simplicity of the explanation to let as many Java developers get into this beautiful world.

When I started digging into Functional Programming a few years ago, I’ve quickly discovered that there is an overwhelming amount of information, but very little of it is understandable for the average Java developer with almost exclusively imperative background. These days situation is slowly changing. There are countless articles which explain, for example, basic FP concepts and how they are applicable to Java. Or articles explaining how to use Java streams properly. But Monads still remain out of the focus of these articles. I don’t know why this happens, but I’ll try to fill this gap.

What Is Monad, Anyway?

The Monad is … a design pattern. As simple as that. This design pattern consists of two parts:

  • Monad is a container for some value. For every Monad, there are methods which allow wrap value into Monad.
  • Monad implements “Inversion of Control” for the value contained inside. To achieve this, Monad provides methods which accept functions. These functions take a value of the same type as stored in Monad and return a transformed value. The transformed value is wrapped into the same kind of Monad as source one.

To understand the second part of the pattern, it will be convenient to look at the imaginable Monad interface:

interface Monad<T> {
<R> Monad<R> map(Function<T, R> mapper);
<R> Monad<R> flatMap(Function<T, Monad<R>> mapper);
}

Of course, a particular Monad usually has a far more rich interface, but these two methods definitely should be present.

At first look, accepting functions instead of providing access to value is not a big difference. In fact, this enables Monad to retain full control on how and when to apply the transformation function. When you call a getter, you expect to get value immediately. In the case of Monad, transformation can be applied immediately or not applied at all, or its application can be delayed. Lack of direct access to value inside enables the monad to represent value which is even not yet available!

Below I’ll show some examples of Monads and which problems they can address.

The Story of The Missing Value or Option/Maybe Monad

This Monad has many names — Maybe, Option, Optional. The last one sounds familiar, isn’t it? Well, since Java 8 Optional is a part of the Java Platform.

Unfortunately Java Optional implementation makes too much reverences to traditional imperative approaches and this makes it less useful than it can be. In particular Optional allows application to get value using .get() method. And even throw NPE if value is missing. As a consequence Optional usage is often limited to the cases when we need to return potentially missing value, although this is only small part of the potential usages.

The purpose of the Maybe Monad is to represent a value which potentially can be missing. Traditionally, this role in Java is reserved for null. Unfortunately, this causes numerous various issues, including the famous NullPointerException.

If you expect that, for example, some parameter or some return value can be null, you ought to check it before use:

Looks familiar? Sure it does.

Let's take a look how Option Monad changes this (with one static import for brevity):

Note that code is much more concise and contains much less “distraction” from business logic.

This example shows how convenient monadic “inversion of control”: transformations don’t need to check for null, they will be called only when value is actually available.

The “do something if/when value is available” is a key mindset to start conveniently using Monads.

Note that the example above retains original API’s intact. But it makes sense to use the approach wider and change API’s, so they will return Optional instead of null:

Few observations:

  • The code even more concise and contains nearly zero boilerplate
  • All types are automatically derived. This is not always so, but in the vast majority of cases types are derived by compiler despite weaker type inference in Java comparing to, for example, Scala
  • There are no explicit error processing, instead we can focus on “happy day scenario”.
  • All transformations are composing and chaining conveniently, no breaks or distractions from main business logic.

In fact, the properties above are common for all Monads.

To Throw Or Not To Throw, That Is The Question

Things not always going as we would like and our applications living in real world, full of suffering, errors and mistakes. Sometimes we can do something with them, sometimes not. If we can’t do anything, we would like at least notify caller that things went not as we anticipated.

In Java, we traditionally have two mechanisms to notify the caller about a problem:

  • Return special value (usually null)
  • Throw an exception

Instead of returning null we can also return Option Monad (see above), but often this is not enough as more detailed information about the error is necessary. Usually, we throw an exception in this case. There is an issue with this approach, though. Actually, even few issues.

  • Exceptions break execution flow
  • Exceptions add a lot of mental overhead

Mental overhead caused by exceptions depends on types of exceptions:

  • Checked exceptions, forcing you either to take care of them right here or declare them in signature and shift headache to the caller
  • Unchecked exceptions cause the same level of issues, but you don’t have support from compiler

Don’t know which one is worse.

Here Comes Either Monad

Let’s analyze the issue for the moment. What we want to return is a some special value, which can be exactly one of two possible things:

  • the result value in case of success
  • the error in case of failure

Note that these things are mutually exclusive — if we return the value, there is no need to carry error and vice versa.

The above is almost exact description of Either Monad: any given instance contains exactly one value, and this value has one of two possible types. The interface of Either Monad can be described like this:

interface Either<L, R> {
<T> Either<T, R> mapLeft(Function<L, T> mapper);
<T> Either<T, R> flatMapLeft(Function<L, Either<T, R>> mapper);
<T> Either<L, T> mapRight(Function<T, R> mapper);
<T> Either<L, T> flatMapRight(Function<R, Either<L, T>> mapper);
}

The interface is rather verbose, as it’s symmetric regarding left and right values. For the narrower use case when we need to deliver success or error, it means that we have to agree on some convention — which type (first or second) will hold error and which will hold value. The symmetric nature of Either makes it more error-prone in this use case, as it’s easy to unintentionally swap error and success values in code. While most likely this problem will be caught by the compiler, it’s better to tailor Either for this particular use case. This can be done if we fix one of the types. Obviously, it’s more convenient to fix the error type, as Java programmers are already used to having all errors and exceptions derived from the single Throwable type.

Result Monad — Either Monad Specialized for Error Handling & Propagation

So, let’s assume that all errors implement the same interface and let’s call it Failure. Now we can simplify and reduce the interface:

interface Result<T> {
<R> Result<R> map(Function<T, R> mapper);
<R> Result<R> flatMap(Function<T, Result<R>> mapper);
}

The Result Monad API looks very similar to the API of Maybe Monad.

Using this Monad, we may rewrite the previous example:

Well, it’s basically identical to the example above, the only change is the kind of Monad — Result instead of Optional. Unlike the previous example, here we have full information about error, so we can do something with that at the upper level. But still, despite full error handling, code remains simple and focused on the business logic.

“Promise is a big word. It either makes something or breaks something.”

The next Monad I’d like to show will be the Promise Monad.

Must admit that I’ve not found authoritative answer if Promise is a monad or not. Different authors have different opinion in regard to it. I’m looking at it from purely pragmatic point of view: it looks and behaves a lot like other monads, so I consider them a monad.

The Promise Monad represents a (potentially not yet available) value. In some sense, it’s very similar to Maybe Monad.

The Promise Monad can be used to represent, for example, the result of a request to an external service or database, file read or write, etc. Basically, it can represent anything which requires I/O and time to perform it. The Promise supports the same mindset as we’ve observed with other Monads — “do something if/when value is available”.

Note that since it’s impossible to predict whether an operation will be successful or not, it’s convenient to implement Promise as a container for the Result Monad. This way we can conveniently handle both, success and error resolutions.

To see how it works, let's take a look example below:

To bring the whole context, I’ve included both necessary interfaces, but actually interesting part is the userTopicHandler() method. Despite suspicious simplicity, this method does the following:

  • Calls TopicService and retrieve list of topics created by provided user
  • When the list of topics is successfully obtained, the method extracts topic ID’s and then calls ArticleService and retrieves the list of articles created by the user for specified topics
  • Performs end-to-end error handling, stops processing immediately as any error is encountered and propagates error back to caller

Quite a lot of work for one (albeit long) line of code. Such an expressiveness and conciseness is a consequence of using Monads.

Afterword

The Monads are a powerful and convenient tool. Writing code using “do when value is available” mindset requires some time to get used to, but once you start getting it, it will allow you to simplify your life a lot. It allows offloading a lot of mental overhead to the compiler and make many errors impossible or detectable at compile time rather than at run time.

--

--

Sergiy Yevtushenko
Sergiy Yevtushenko

Written by Sergiy Yevtushenko

Writing code for 35+ years and still enjoy it…

Responses (1)