Reactive Programming in Kotlin
上QQ阅读APP看书,第一时间看更新

Functional programming – monads

Functional programming is incomplete without monads. If you are into functional programming, then you know it very well; otherwise, you are hearing it for the first time. So, what is a monad? Let's learn about it. The concept of monad is quite abstract; the definition says monad is a structure that creates a new type by encapsulating a value and adding some extra functionalities to it. So, let's start by using a monad; take a look at the following program:

    fun main(args: Array<String>) { 
      val maybeValue: Maybe<Int> = Maybe.just(14)//1 
      maybeValue.subscribeBy(//2 
        onComplete = {println("Completed Empty")}, 
        onError = {println("Error $it")}, 
        onSuccess = { println("Completed with value $it")} 
      ) 
      val maybeEmpty:Maybe<Int> = Maybe.empty()//3 
      maybeEmpty.subscribeBy( 
        onComplete = {println("Completed Empty")}, 
        onError = {println("Error $it")}, 
        onSuccess = { println("Completed with value $it")} 
      ) 
    } 

Here, Maybe is a monad that encapsulates an Int value with some added functionalities. The Maybe monad says it may or may not contain a value, and it completes with or without a value or with an error. So, if there's an error, then it would obviously call onError; if there are no errors, and if it has a value, it will call onSuccess with the value; and, if it doesn't have a value and no error as well, it will call onComplete. The thing to note is that all three methods here, onError, onComplete, and onSuccess, are terminal methods, meaning either one of these three will get called by a Maybe monad, and others will never be called.

Let's go through the program to understand the monads better. On comment (1), we will declare a Maybe monad and assign a value of 14 to it. On comment (2), we will subscribe to the monad. On comment (3), we will again declare a Maybe monad, this time with an empty value. The subscription takes three lambdas as parameter–when the monad contains a value, onSuccess gets called; when it doesn't contain any value, onComplete gets called; and if any error occurs, then onError gets called. Let's see the output now:

Completed with value 14 
Completed Empty 

So, as we can see, for maybeValue, onSuccess gets called, but for maybeEmpty , the onComplete method gets called.