Skip to content

Latest commit

 

History

History
1732 lines (1345 loc) · 54 KB

readme.md

File metadata and controls

1732 lines (1345 loc) · 54 KB

my-scala-notes

My scala notes, because I need to store them somewhere...

Type

Every variable and expression in a Scala program has a type that is known at compile time:

// a literal value has a type:
scala> 1
res0: Int = 1

// a variable has a type and can refer only value instances of this type:
scala> val x: Int = 1
x: Int = 1

// an expression produces a type:
scala> :t 1 + 1
Int

A type restricts the possible values to which a variable can refer, or an expression can produce, at run time. A variable or expression’s type can also be referred to as a static type if necessary to differentiate it from an object’s runtime type. In other words, “type” by itself means static type.

Type is distinct from class because a class that takes type parameters can construct many types. For example, 'List' is a class, but not a type. 'List[A]' is a type with a free type parameter. 'List[Int]' and 'List[String]' are also types (called ground types because they have no free type parameters). A type can have a 'class' or 'trait.'

For example, the class of type List[Int] is List. The trait of type Set[String] is Set.

Type Constructor

A type constructor is a class or trait that takes type parameters like List[A]. So List[A] is a type constructor because it takes a type A, but for example 'List[String]' is a concrete type.

Type Parameter

A Type Parameter is a parameter to a generic class or generic method that must be filled in by a type. For example, class List is defined as 'List[A]'. The 'A' is a type parameter.

Type Parameter Variance

A type parameter of a class or trait can be marked with a variance annotation, either covariant (+) or contravariant (-). Such variance annotations indicate how subtyping works for a generic class or trait. For example, the generic class List is covariant in its type parameter so its actually List[+A], and thus List[String] is a subtype of List[Any]. By default, i.e., absent a (+) or (-) annotation, type parameters are nonvariant.

So its all about how subtyping works for the container type like eg. in this case List[String] is a subtype of List[Any]. (read this line again 5-times!)

Lets try it out. First create a simple domain of Animals:

sealed trait Animal
trait Swimmer { def swim: Unit = println("Swimming") }
class Bird extends Animal {
  def fly: Unit = println("flying")
  override val toString = "Bird"
}
class Sparrow extends Bird { override val toString = "Sparrow" }
class Duck extends Bird with Swimmer { override val toString = "Duck" }

We have created the following graph:

Duck    -> Swimmer -> Bird -> Animal -> AnyRef -> Any
Sparrow ->            Bird -> Animal -> AnyRef -> Any

Lets say we create a Cage that will hold any Bird. We have some variance options of the case namely

  • An invariant cage [A],
  • A covariant cage [+A],
  • A contravariant cage [-A].

An Invariant Cage so Cage[A]

Lets first look at the invariant relationship:

// an invariant Cage
scala> class Cage[A](val animal: A)
defined class Cage

// lets create a birdCage
scala> val birdCage = new Cage(new Bird)
birdCage: Cage[Bird] = Cage@3ca14cf4

We have created a Cage that is invariant, meaning that there is no subtype relationship between two cages. If we ask the compiler to convert the birdCage, which is of type Cage[Bird] to a Cage[Sparrow] it cannot, because the instance is invariant meaning it cannot change the type of the Cage:

scala> birdCage: Cage[Sparrow]
<console>:15: error: type mismatch;
 found   : Cage[Bird]
 required: Cage[Sparrow]
Note: Bird >: Sparrow, but class Cage is invariant in type A.
You may wish to define A as -A instead. (SLS 4.5)
       birdCage: Cage[Sparrow]
       ^

We will task about the error message later.

If we ask the compiler to convert the birdCage, which is of type Cage[Bird] to a Cage[Animal] it cannot, because the instance is invariant meaning it cannot change the type of the Cage:

scala> birdCage: Cage[Animal]
<console>:15: error: type mismatch;
 found   : Cage[Bird]
 required: Cage[Animal]
Note: Bird <: Animal, but class Cage is invariant in type A.
You may wish to define A as +A instead. (SLS 4.5)
       birdCage: Cage[Animal]
       ^

Does variance have anything to do with the instances Animal instances we can put in the birdCage? Well, we can put any Bird in the cage as we can see here:

scala> val birdCage: Cage[Bird] = new Cage(new Duck)
birdCage: Cage[Bird] = Cage@28ee0a3c

// the animal is a Duck, but its type is a Bird
scala> birdCage.animal
res0: Bird = Duck

scala> val birdCage: Cage[Bird] = new Cage(new Sparrow)
birdCage: Cage[Bird] = Cage@6221b13b

// the animal is a Sparrow, but its type is a Bird
scala> birdCage.animal
res1: Bird = Sparrow

Note that the birdCage is still of type Cage[Bird] and all the animals that it contains will be seen as a Bird.

A Covariant Cage

The birdCage we had so far is of an Invariant type so Cage[A]. When we asked the compiler to change the type of the Cage to a Cage[Animal] we couldn't. It did however gave a tip, we could change the type of the Cage to a covariant type by putting a (+) sign before the A, so Cage[+A], lets create a covariant Cage:

scala> class Cage[+A](val animal: A)
defined class Cage

// lets create a duckCage so a Cage[Duck]
scala> val duckCage: Cage[Duck] = new Cage(new Duck)
duckCage: Cage[Duck] = Cage@20b52576

// the animal it holds is of type Duck
scala> duckCage.animal
res0: Duck = Duck

Because the Cage is Covariant meaning [+A], this means that there exists a subtype relationship between Cage[+A] types. The question is what the Cage[+A] subtype relationship is. A mnemonic for the Covariant relationship is looking at the sign. The (+) points UP, in our example, the Cage is defined as Cage[Duck]. So the following relationship exists:

Cage[Duck] -> Cage[Swimmer] -> Cage[Bird] -> Cage[Animal] -> Cage[AnyRef] -> Cage[Any]

You are right if you see the subtype relationship here of the type it contains! You got it!

scala> val swimmerCage: Cage[Swimmer] = duckCage
swimmerCage: Cage[Swimmer] = Cage@20b52576

scala> swimmerCage.animal
res1: Swimmer = Duck

scala> val birdCage: Cage[Bird] = duckCage
birdCage: Cage[Bird] = Cage@20b52576

scala> birdCage.animal
res2: Bird = Duck

scala> val animalCage: Cage[Animal] = duckCage
animalCage: Cage[Animal] = Cage@20b52576

scala> animalCage.animal
res3: Animal = Duck

scala> val anyRefCage: Cage[AnyRef] = duckCage
anyRefCage: Cage[AnyRef] = Cage@20b52576

scala> anyRefCage.animal
res4: AnyRef = Duck

scala> val anyCage: Cage[Any] = duckCage
anyCage: Cage[Any] = Cage@20b52576

scala> anyCage.animal
res5: Any = Duck

The question now is, can we go the other way around? So can we convert a Cage[Animal] back to a Cage[Duck]?

scala> animalCage: Cage[Duck]
<console>:15: error: type mismatch;
 found   : Cage[Animal]
 required: Cage[Duck]
       animalCage: Cage[Duck]

No we cannot, because we have defined the Cage to be Covariant meaning Cage[+A].

A Contravariant Cage

Right, when we changed the Cage to a Covariant type so Cage[+A] we could change the type of the cage UPwards, meaning pointing to the more abstract version of the animal the Cage contained, so when we started with a Cage[Duck], we could only go to a Cage[Any] so UPwards. Can we define a Contravariant Cage so Cage[-A]?

Well, yes we can! Lets look at suck a cage:

scala> class Cage[-A](animal: A)
defined class Cage

scala> val birdCage: Cage[Bird] = new Cage(new Duck)
birdCage: Cage[Bird] = Cage@f160948

scala> val duckCage: Cage[Duck] = birdCage
duckCage: Cage[Duck] = Cage@f160948

Because the Cage is contravariant so Cage[-A], we can change type of Cage DOWNwards, meaning in Object Oriented terms, 'the more specialized type'. So in case of Cage[Bird], we can change the container type to a Cage[Duck], but what does that mean?

Well think about it, a Duck is a more specialized type of Bird, it can fly AND swim for example, something a Bird cannot do; a Bird can only fly:

// a duck can both fly and swim
scala> new Duck().fly
flying

scala> new Duck().swim
Swimming

// a bird cannot swim
scala> new Bird().swim
<console>:13: error: value swim is not a member of Bird
       new Bird().swim

So, for example, when we start out with a contravariant Cage[-A], which means that if we want to parameterized the Cage, and for example define methods and fields in it what would it mean for the members of the cage if we start out with a Cage[Bird] and specialize it to a Cage[Duck]?

For example, what if we create the following Cage:

class Cage[-A] {
  def get: A
  def put(animal: A): Unit = ???
  val animal: A = ???
  var animal: A = ???
}

The example above won't compile, but lets determine why not, what is the problem with the definition. The Cage is defined as being contravariant so Cage[-A], so the A can be specialized. What is the problem when we start out with eg. a Cage[Any] and specialize it to Cage[Duck]. Lets determine what happens with the members of the Cage?

Well, one problem is obvious, we can put 'Any' type into the Cage, so I can put a 'Car' into the Cage and that would be fine. If I can specialize the Cage to a Cage[Duck] then interesting things will happen for example, if I call get()

val anyCage: Cage[Any] = new Cage(new Car)
val duckCage: Cage[Ducl] = anyCage
val duck: Duck = duckCage.get

Well, obviously we would get a Car and cars don't have a swim method. So logically some members of an contravariant parameterized type have problems with being a member of such a type so being a member of a Cage[-A].

The other way around however is no problem, if we are a member of a Covariant parameterized type for example a Cage[+A], and we can only go UPwards, we can become more generic, then a Cage[Duck] can become a Cage[Any].

Each member of a parameterized type for example the constructor parameter, field, method parameter have a certain position in the parameterized type, here Cage. When it comes to the variance of the type parameter of the parameterized type it has some consequence. Lets look at what the consequences are.

Type Parameter Variance Positions

There seems to be a consequence for the members of a parameterized type like for example Cage when the variance of the type parameter is invariant, covariant (+) or contravariant (-). The possible positions and variance possibilities of the type parameter is as follows:

class Cage[A] {
  def get: A // covariant (+)
  def put(animal: A): Unit // contravariant (-)
  val animal: A // covariant, "you can only get"
  var animal: A // "covariant and contravariant" so invariant
}

Okay, what does the 'blueprint' above mean? Well, you can't just 'sick-on-a-variance-type' on your parameterized type like Cage! You must think what the consequences are if you choose a variance.

The type parameter is invariant:

class Cage[A] {
  def get: A = ???
  def put(animal: A): Unit = ???
  val animal: A = ???
  var _animal: A  = ???
}

When the variance is INVARIANT, so Cage[A], you can do the above, but the consequence is that the container type doesn't have a subtype relationship.

The type parameter is covariant:

class Cage[+A] {
  def get: A = ???
  val animal: A = ???
}

When the variance is COVARIANT, so Cage[+A], you can do the above, you can only get AND you can have an immutable so read-only field. For the invariant and covariant Cage examples above, I promoted the contructor parameter of the class to a field by adding the 'val' keyword, which creates a member in the class that can be made accessible by means of the uniform access principle: so thats why we could call: birdCage.animal.

The type parameter is contravariant:

class Cage[-A](animal: A) {
  def put(animal: A): Unit = ???
}

When the variance is CONTRAVARIANT, so Cage[-A], so the contained subtype relationship has been reversed, you can do the above, so only use constructor parameters and method parameters, nothing else.

We have learned the following:

  • Types of vals are in covariant position (+)
  • Types of vars are in invariant position
  • Method parameter types are in contravariant position (-)
  • Method return types are in covariant position (+)
  • Class parameters don't matter

Package objects

If you come from a programming language like Java, then you know that you can organize your code in packages. The things you would put inside a package are classes, interfaces and enums. In the Scala language, until version 2.8 this was basically the same so you could put classes, traits and objects inside a package. But from version 2.8 and up you can create a file named 'package.scala' and put that file into the package directory. Such a file is called a package object.

The contents of the file must start with the keywords package object followed by the name of the package for example, if we have the package foo.bar.baz and I want to have a 'package object' in the package 'baz' then I would do the following:

  • create a file 'package.scala' in the directory foo/bar/baz
  • put the following content in the file:
package foo.bar

package object baz {

}

Between the curly braces you can put anything you'd like such as methods and constants or types and the cool thing is all of these will be available without any imports to all types inside the 'foo.bar.baz' package; pretty neat huh!

What package object most often are used for is to hold:

  • package-wide type aliases
  • implicit conversions
  • constants
  • functions
  • methods

A good package object you should know and take a look at is the scala package object, which definitions are automatically imported into every '.scala' file you create, together with the all definitions in the scala.Predef object and of course all the definitions in java.lang.

Package chaining

Package chaining or chained package clause is a way to get parent packages into scope of your '.scala' file. Say for example that I want to create an object named 'BazObject' in the package foo.bar.baz, and say that all the packages 'foo', 'bar' and 'baz' all three of them have a package object defined so they would all have a file called 'package.scala', with the appropriate content and say I want to bring the contents of all these three packages into scope of the BazObject, then the package statement at the top of the '.scala' file will contain multiple 'chained' entries of each of the package names's content you wish to bring in scope, so for example for the BazObject the definition will be:

package foo
package bar
package baz

// the contents of all the three packages are now in scope

object BazObject {
}

Scala Functions

Scala functions are instances of the scala.FunctionN[-T1, +R] trait. We can manually create instances of a function and call it:

scala> val f: Function1[Int, Int] = new Function1[Int, Int] { def apply(x: Int): Int = x + 1 }
f: Int => Int = <function1>

scala> f(1)
res0: Int = 2

Wow, that is a lot of boilerplate syntax. Fortunately Scala has some syntactic sugar for us to use that generates this code for us:

scala> val f: Int => Int = (x: Int) => x + 1
f: Int => Int = $$Lambda$1191/1289462509@2c1f8dbd

scala> f(1)
res1: Int = 2

Or very concise:

scala> val f = (_: Int) + 1
f: Int => Int = $$Lambda$1192/1869813593@1f84327b

scala> f(1)
res2: Int = 2

0-Arity Functions

0-Arity functions are functions that are instances of the scala.Function0[+R] trait:

scala> val f: Function0[Int] = new Function0[Int] { def apply(): Int = 1 }
f: () => Int = <function0>

scala> f()
res0: Int = 1

Scala has special syntax for these kinds of functions:

scala> val f: Function0[Int] = () => 1
f: () => Int = $$Lambda$1195/1872158052@4d7cac24

scala> f()
res1: Int = 1

Scala has syntactic sugar for the Function0 literal type:

scala> val f: () => Int = () => 1
f: () => Int = $$Lambda$1194/413763859@7dbae40

scala> f()
res2: Int = 1

Scala and Thunks

A thunk is a 'call-by-name' parameter. It is most often used as a lazy evaluated parameter on methods. Thunks are implemented (under the hood) as Function0 functions by the Scala compiler but the thunk itself cannot be called with a 0-Arity function. Also, the thunk must be called parameter-less, lets look at an example:

scala> :paste
// Entering paste mode (ctrl-D to finish)

def foo[A](x: => A): Unit = {
  println("Calling the thunk")
  val result: A = x
  println("Received from thunk: " + x)
}

// Exiting paste mode, now interpreting.

foo: [A](x: => A)Unit

The method 'foo' can be called in the following ways:

scala> def bar(): Int = 42
bar: ()Int

scala> foo(bar())
Calling the thunk
Received from thunk: 42

scala> foo (42 - 12 + 20)
Calling the thunk
Received from thunk: 50

How to unify method parameters and tuples

Scala does not unify method parameters and tuples, for example, the following method:

def addNumbers(x: Int, y: Int): Int = x + y

Doesn't work with tuples. It would be nice if it did, but that is not the case. Also, changing the method API to tuples doesn't work as nice:

def addNumbers(numbers: (Int, Int)): Int = numbers match {
  case (x, y) => x + y
}

What is the alternative? Well, we can 'convert' the method to a function and change its interface to a tuple:

scala> def addNumbers(x: Int, y: Int): Int = x + y
addNumbers: (x: Int, y: Int)Int

scala> addNumbers _
res0: (Int, Int) => Int = $$Lambda$1495/60466312@3a332eed

scala> res0.tupled
res1: ((Int, Int)) => Int = scala.Function2$$Lambda$229/1860944798@3c0178f

scala> val numbers = (1, 2)
numbers: (Int, Int) = (1,2)

scala> res1(numbers)
res2: Int = 3

The pattern of converting a method to a function and then calling '.tupled' can be useful because most operations we do in practise return pairs or list of pairs:

scala> val xs = Vector(1, 2, 3, 4)
xs: scala.collection.immutable.Vector[Int] = Vector(1, 2, 3, 4)

scala> val ys = xs.zip(xs.tail)
ys: scala.collection.immutable.Vector[(Int, Int)] = Vector((1,2), (2,3), (3,4))

scala> ys.map(res1)
res3: scala.collection.immutable.Vector[Int] = Vector(3, 5, 7)

scala> ys.map((addNumbers _).tupled)
res4: scala.collection.immutable.Vector[Int] = Vector(3, 5, 7)

Curried functions

Methods can be converted to functions like so:

scala> def addNumbers(x: Int, y: Int) = x + y
addNumbers: (x: Int, y: Int)Int

scala> val x = addNumbers _
x: (Int, Int) => Int = $$Lambda$1530/265754685@1e63cd7e

The function can then be applied by applying the function with all the arguments:

scala> x(1, 2)
res0: Int = 3

We can also curry the function, which means that we've converted the function that takes multiple arguments into a sequence of functions each taking a single argument:

scala> x.curried
res1: Int => (Int => Int) = scala.Function2$$Lambda$1529/1670650711@5c7bbf0a

scala> x(1)
scala> val y = x.curried
y: Int => (Int => Int) = scala.Function2$$Lambda$1529/1670650711@60e784de

scala> y(1)
res2: Int => Int = scala.Function2$$Lambda$1531/1050434890@6fae8b4d

scala> res3(2)
res3: Int = 3

Scalaz Validation: Parsing user input

Parsing user input is easy with Scala and is provided by the class StringLike and provides methods like 'toBoolean', 'toInt', 'toLong', 'toDouble', but all of these methods can throw exceptions like NumberFormatException and IllegalArgumentException.

scala> "foo".toBoolean
java.lang.IllegalArgumentException: For input string: "foo"

scala> "foo".toDouble
java.lang.NumberFormatException: For input string: "foo"

Of course, we can wrap the operation in a scala.util.Try, but this is not ideal:

scala> Try("foo".toBoolean)
res0: scala.util.Try[Boolean] = Failure(java.lang.IllegalArgumentException: For input string: "foo")

Scalaz provides scalaz.syntax.std.StringOps and provides the methods:

  • def parseBoolean: Validation[IllegalArgumentException, Boolean] = s.parseBoolean(self)
  • def parseByte: Validation[NumberFormatException, Byte] = s.parseByte(self)
  • def parseShort: Validation[NumberFormatException, Short] = s.parseShort(self)
  • def parseInt: Validation[NumberFormatException, Int] = s.parseInt(self)
  • def parseLong: Validation[NumberFormatException, Long] = s.parseLong(self)
  • def parseFloat: Validation[NumberFormatException, Float] = s.parseFloat(self)
  • def parseDouble: Validation[NumberFormatException, Double] = s.parseDouble(self)
  • def parseBigInt: Validation[NumberFormatException, BigInt] = s.parseBigInt(self)
  • def parseBigDecimal: Validation[NumberFormatException, BigDecimal] = s.parseBigDecimal(self)
scala> import scalaz._
import scalaz._

scala> import Scalaz._
import Scalaz._

scala> "foo".parseBoolean
res0: scalaz.Validation[IllegalArgumentException,Boolean] = Failure(java.lang.IllegalArgumentException: For input string: "foo")

scala> "true".parseBoolean
res1: scalaz.Validation[IllegalArgumentException,Boolean] = Success(true)

All the methods provided by scalaz.syntax.std.StringOps return a scalaz.Validation with a Throwable on the failure side and the expected type on the success side.

Creating Validation instances

The companion object of scalaz.Validation provides the following ways to create Validation instances:

// create a success instance
scala> Validation.success[String, Int](1)
res0: scalaz.Validation[String,Int] = Success(1)

// create a failure instance
scala> Validation.failure[String, Int]("My Error")
res1: scalaz.Validation[String,Int] = Failure(My Error)

// create a FailureNel instance
scala> Validation.failureNel[String, Int]("My Error")
res2: scalaz.ValidationNel[String,Int] = Failure(NonEmpty[My Error])

// create a Validation, based on a predicate, the instance will either be a
// success or a failure
scala> Validation.lift(17)(age => age < 18, "You must be at least 18 to buy beer")
res3: scalaz.Validation[String,Int] = Failure(You must be at least 18 to buy beer)

// create a ValidationNel instance
scala> Validation.liftNel(17)(age => age < 18, "You must be at least 18 to buy beer")
res4: scalaz.ValidationNel[String,Int] = Failure(NonEmpty[You must be at least 18 to buy beer])

// create a validation from a thunk that can throw
scala> Validation.fromTryCatchThrowable[Int, Throwable](1/0)
res5: scalaz.Validation[Throwable,Int] = Failure(java.lang.ArithmeticException: / by zero)

// create a validation from a thunk that throws nonfatal exceptions
scala> Validation.fromTryCatchNonFatal[Int](1/0)
res6: scalaz.Validation[Throwable,Int] = Failure(java.lang.ArithmeticException: / by zero)

// create a validation from an either
scala> val answer = scala.util.Right[String, Int](42)
answer: scala.util.Right[String,Int] = Right(42)

scala> Validation.fromEither(answer)
res7: scalaz.Validation[String,Int] = Success(42)

The most interesting constructors are 'lift' and 'liftNel' that take a predicate, and if that predicate resolves to true, it will return a Failure with the contents of the second argument of the 'lift' method. These two methods are really handy for encoding business rule validations:

scala> def validateAge(age: Int): ValidationNel[String, Int] = Validation.liftNel(age)(age => age < 18, "You must be at least 18 to buy beer")
validateAge: (age: Int)scalaz.ValidationNel[String,Int]

scala> def validateName(name: String): ValidationNel[String, String] = Validation.liftNel(name)(name => name.isEmpty, "Name must not be empty")
validateName: (name: String)scalaz.ValidationNel[String,String]

scala> validateAge(17)
res0: scalaz.ValidationNel[String,Int] = Failure(NonEmpty[You must be at least 18 to buy beer])

scala> validateName("")
res1: scalaz.ValidationNel[String,String] = Failure(NonEmpty[Name must not be empty])

Using Validation results in an imperative way

The scalaz.Validation instances contain two properties 'isSuccess' and 'isFailure' that return a boolean, so we could do the following:

scala> if(validateAge(17).isSuccess && validateName("Dennis").isSuccess) "yeah success!" else "darn, a failure"
res0: String = darn, a failure

Of course there are better ways, lets go functional!

Using Validation results the functional way

Lets do the following, parse an electronic form that takes the user name and the user age:

scala> val formName = ""
formName: String = ""

scala> val formAge = "17"
formAge: String = 17

// convert it to a ValidationNel[String, Int]
scala> val parsedAge: ValidationNel[String, Int] =
     |   formAge.parseInt.leftMap(_.toString).toValidationNel
parsedAge: scalaz.ValidationNel[String,Int] = Success(17)

// define some validation methods
scala> def validateName(name: String): ValidationNel[String, String] =
     |   Validation.liftNel(name)(name => name.isEmpty, "Name must not be empty")
validateName: (name: String)scalaz.ValidationNel[String,String]

scala> def validateAge(age: Int): ValidationNel[String, Int] =
     |   Validation.liftNel(age)(age => age < 18, "You must be at least 18 to buy beer")
validateAge: (age: Int)scalaz.ValidationNel[String,Int]

// create a case class to capture a validated person
scala> case class Person(name: String, age: Int)
defined class Person

// define a validation to validate name and age, accrue the failures and create
// a person that captures a valid person
scala> def validateNameAndAge(name: String, age: Int): ValidationNel[String, Person] =
     |   (validateName(name) |@| validateAge(age))(Person.apply)
validateNameAndAge: (name: String, age: Int)scalaz.ValidationNel[String,Person]

// Validation from a form has two steps:
// 1. converting from the String types to the expected types so String => Int
// 2. Validating the business rules and accrue the results

// we can use Validation is a for-expression by importing the following:
scala> import scalaz.Validation.FlatMap.ValidationFlatMapRequested
import scalaz.Validation.FlatMap.ValidationFlatMapRequested

scala> for {
     |   age <- parsedAge
     |   person <- validateNameAndAge(formName, age)
     | } yield person
res0: scalaz.Validation[scalaz.NonEmptyList[String],Person] =
  Failure(NonEmpty[Name must not be empty,You must be at least 18 to buy beer])

Validating a list of business rules

We can use the 'traverse' method on a List of business rules to validate the person eg:

scala> val formName = "Dennis42"
formName: String = Dennis42

scala> val formAge = "17"
formAge: String = 17

scala> val parsedAge: ValidationNel[String, Int] =
     |   formAge.parseInt.leftMap(_.toString).toValidationNel
parsedAge: scalaz.ValidationNel[String,Int] = Success(17)

// lets create some validations
scala> def validateNonEmpty(name: String): ValidationNel[String, String] =
     |   Validation.liftNel(name)(name => name.isEmpty, "Name must not be empty")
validateNonEmpty: (name: String)scalaz.ValidationNel[String,String]

scala> def validateLength(name: String): ValidationNel[String, String] =
     |   Validation.liftNel(name)(name => name.length > 7, "Maximum length of 7 chars exceeded")
validateLength: (name: String)scalaz.ValidationNel[String,String]

scala> def validateOnlyLetters(name: String): ValidationNel[String, String] =
     |   Validation.liftNel(name)(name => name.exists(char => !char.isLetter), "name must contain only letters")
validateOnlyLetters: (name: String)scalaz.ValidationNel[String,String]

scala> def validateAge(age: Int): ValidationNel[String, Int] =
     |   Validation.liftNel(age)(age => age < 18, "You must be at least 18 to buy beer")
validateAge: (age: Int)scalaz.ValidationNel[String,Int]

scala> def validateName(name: String): ValidationNel[String, String] = {
     |   NonEmptyList(
     |     validateNonEmpty _,
     |     validateLength _,
     |     validateOnlyLetters _)
     |     .traverseU(_.apply(name))
     |     .rightMap(_.head)
     | }
validateName: (name: String)scalaz.ValidationNel[String,String]

scala> case class Person(name: String, age: Int)
defined class Person

scala> def validateNameAndAge(name: String, age: Int): ValidationNel[String, Person] =
     |   (validateName(formName) |@| validateAge(age)) (Person.apply)
validateNameAndAge: (name: String, age: Int)scalaz.ValidationNel[String,Person]

// we can use Validation is a for-expression by importing the following:
scala> import scalaz.Validation.FlatMap.ValidationFlatMapRequested
import scalaz.Validation.FlatMap.ValidationFlatMapRequested

scala> for {
     |   age <- parsedAge
     |   person <- validateNameAndAge(formName, age)
     | } yield person
res1: scalaz.Validation[scalaz.NonEmptyList[String],Person] =
  Failure(NonEmpty[Maximum length of 7 chars exceeded,name must contain only letters,You must be at least 18 to buy beer])

Case Class Tricks

A case class is a scala.Product, which is the base trait for all case classes and tuples. Because both case classes and tuples are a Product type, they have access to the methods:

method description
productArity: Int The size of this product ie. the number of types it contains eg. Person(name: String, age: Int) has an arity of 2
productElement(n: Int): Any Returns the n-th element of the product, and is zero-based
productIterator: Iterator[Any] An iterator over the elements of the product
productPrefix: String A String used in the toString() methods of derived classes
scala> case class Person(name: String, age: Int)
defined class Person

scala> val person = Person("foo", 42)
person: Person = Person(foo,42)

scala> person.isInstanceOf[Product]
res0: Boolean = true

scala> val values = person.productIterator.toList
values: List[Any] = List(foo, 42)

scala> person.productElement(0)
res1: Any = foo

scala> person.productElement(1)
res2: Any = 42

scala> person.productArity
res3: Int = 2

scala> person.productPrefix
res4: String = Person

Getting the fields of the case class:

import scala.reflect.runtime.universe._

scala> def getMethods[T <: Product : TypeTag] : List[MethodSymbol] = typeOf[T].members.collect {
     |     case m: MethodSymbol if m.isCaseAccessor => m
     |   }.toList
getMethods: [T <: Product](implicit evidence$1: reflect.runtime.universe.TypeTag[T])List[reflect.runtime.universe.MethodSymbol]

scala> getMethods[Person]
res5: List[reflect.runtime.universe.MethodSymbol] = List(value age, value name)

scala> getMethods[Person].map(_.name.toString)
res6: List[String] = List(age, name)

Getting the names and values of the case class:

scala> :paste
// Entering paste mode (ctrl-D to finish)

getMethods[Person]
  .map(_.name.toString)
  .iterator.zip(Person("foo", 42).productIterator)
  .toMap

// Exiting paste mode, now interpreting.

res7: scala.collection.immutable.Map[String,Any] = Map(age -> foo, name -> 42)

Scalaz ReaderT usage

A monad transformer for the Reader Monad

import scalaz._
import Scalaz._
import scala.language.higherKinds
import scala.language.implicitConversions

case class StringStats(length: Int, palindrome: Boolean)

def stringStats(calcLength: String => Int, isPalindrome: String => Boolean): String => StringStats = for {
  length <- calcLength
  palindrome <- isPalindrome
} yield StringStats(length, palindrome)

def stringStatsCtx[F[_]: Monad]
  (calcLength: ReaderT[F, String, Int],
   isPalindrome: ReaderT[F, String, Boolean]): ReaderT[F, String, StringStats] = for {
  length <- calcLength
  palindrome <- isPalindrome
} yield StringStats(length, palindrome)

def compose[F[_]: Monad, A: Monoid](fx: F[A], fy: F[A]): F[A] = for {
  x <- fx
  y <- fy
} yield x |+| y

compose[Id, Int](1, 2) == 3

val statsFunction: (String) => StringStats = 
  stringStats(_ => 4, _ => true)

statsFunction("abba") == StringStats(4, true)

// actually use the function and define the effect
val statsFunctionCtx: ReaderT[Option, String, StringStats] =
  stringStatsCtx[Option](
    ReaderT[Option, String, Int](_ => Option(4)),
    ReaderT[Option, String, Boolean](_ => Option(true))
  )

statsFunctionCtx("abba") == Some(StringStats(4, true))

// there must be an implicit execution context
// available at the call site
import scala.concurrent.duration._
import scala.concurrent.{Await, Future}
def callStrLengthService(str: String): Future[Int] =
  Future.successful(4)

def callStrPalindromeService(str: String): Future[Boolean] =
  Future.successful(str.reverse == str)

import scala.concurrent.ExecutionContext.Implicits.global
val statsFunctionAsync: ReaderT[Future, String, StringStats] =
  stringStatsCtx[Future](
    ReaderT[Future, String, Int](callStrLengthService),
    ReaderT[Future, String, Boolean](callStrPalindromeService)
  )

Await.result(statsFunctionAsync("abba"), 1.second) == 
  StringStats(4, true)

Creating and Parsing a binary String

Whats really fun is converting a text String to a binary String and back again:

"Hello World".getBytes.map(Integer.toBinaryString(_)).mkString(" ")

"1001000 1100101 1101100 1101100 1101111 100000 1010111 1101111 1110010 1101100 1100100"
.split("\\s").map(Integer.parseInt(_, 2)).map(_.toChar).mkString

Scala object equality

In Scala we check for object equality using '==' operator. The '==' operator delegates to the '.equals' method:

scala> 1 == 1
res0: Boolean = true

scala> new String("foo") == new String("foo")
res1: Boolean = true

When creating our own value objects manually, we should override the equals method, but there is a better way, using case classes. Case classes create a valid '.equals' and 'hashcode' methods:

scala> case class Person(name: String, age: Int)
defined class Person

scala> Person("foo", 42) == Person("foo", 42)
res2: Boolean = true

Checking for reference equality

In Scala we check for reference equality using the 'eq' and 'ne' operators:

scala> new String("foo") eq new String("foo")
res0: Boolean = false

scala> new String("foo") ne new String("foo")
res1: Boolean = true

Simple FizzBuzz example

FizzBuzz can be solved in many ways, you could do the following:

def evaluate(x: Int): String = {
  if(x % 15 == 0)
    "FizzBuzz"
  else if(x % 3 == 0) 
    "Fizz"
  else if(x % 5 == 0)
    "Buzz"
  else x.toString
}

(1 to 100).map(evaluate)

FizzBuzz with pattern matching

If you like pattern matching approach:

Using guards:

(1 to 100)
  .map {
    case x if x % 15 == 0 => "FizzBuzz"
    case x if x % 3 == 0 => "Fizz"
    case x if x % 5 == 0 => "Buzz"
    case x => x.toString
  }

Using extractors:

object FizzBuzz {
  def unapply(arg: Int): Option[String] =
    Option(arg).find(_ % 15 == 0).map(_ => "FizzBuzz")
}

object Fizz {
  def unapply(arg: Int): Option[String] =
    Option(arg).find(_ % 3 == 0).map(_ => "Fizz")
}

object Buzz {
  def unapply(arg: Int): Option[String] =
    Option(arg).find(_ % 5 == 0).map(_ => "Buzz")
}

(1 to 100)
  .map {
    case FizzBuzz(x) => x
    case Fizz(x) => x
    case Buzz(x) => x
    case x => x.toString
  }

Using tuples:

(1 to 100)
  .map(x => (x, x % 3, x % 5))
  .map {
    case (_, 0, 0) => "FizzBuzz"
    case (_, 0, _) => "Fizz"
    case (_, _, 0) => "Buzz"
    case (nr,_, _) => nr
  }

Using a Stream:

Stream.from(1)
  .map(x => (x, x % 3, x % 5))
  .map {
    case (_, 0, 0) => "FizzBuzz"
    case (_, 0, _) => "Fizz"
    case (_, _, 0) => "Buzz"
    case (nr,_, _) => nr
  }
.take(35)
.toList

A FizzBuzz example

Of course, the FizzBuzz can be solved in many ways, this is just one of many that uses an aggregator approach:

val fizzPred: Int => Boolean = (_: Int) % 3 == 0
val buzzPred: Int => Boolean = (_: Int) % 5 == 0
val woofPred: Int => Boolean = (_: Int) % 7 == 0

val fizzTxt = (x: Boolean) => if(x) Option("Fizz") else None
val buzzTxt = (x: Boolean) => if(x) Option("Buzz") else None
val woofTxt = (x: Boolean) => if(x) Option("Woof") else None

val fizz = fizzPred andThen fizzTxt
val buzz = buzzPred andThen buzzTxt
val woof= woofPred andThen woofTxt

val rules = List(fizz, buzz, woof)

val ys = (1 to 35).map(x => (x, rules))
.map {
  case (x, xs) => (x, xs.flatMap(_(x)))
}
.map {
  case (x, Nil) => x.toString
  case (x, xs) => xs.mkString
}

ys.foreach(println)

1
2
Fizz
4
Buzz
Fizz
Woof
8
Fizz
Buzz
11
Fizz
13
Woof
FizzBuzz
16
17
Fizz
19
Buzz
FizzWoof
22
23
Fizz
Buzz
26
Fizz
Woof
29
FizzBuzz
31
32
Fizz
34
BuzzWoof

FizzBuzz with Scalaz

Also using an aggregator:

import scalaz._
import Scalaz._

val fizzPred = (_: Int) % 3 == 0
val buzzPred = (_: Int) % 5 == 0
val woofPred = (_: Int) % 7 == 0
  
val fizzTxt = (x: Boolean) => if(x) Option("Fizz") else None
val buzzTxt = (x: Boolean) => if(x) Option("Buzz") else None
val woofTxt = (x: Boolean) => if(x) Option("Woof") else None

val fizz = fizzPred andThen fizzTxt
val buzz = buzzPred andThen buzzTxt
val woof= woofPred andThen woofTxt

val rules = List(fizz, buzz, woof)

val ys = (1 to 35).map { x =>
  // apply the value to the list of rules
  (List(x) <*> rules)
    // the resulting List[Option] will be flatten'd
    // and converted to an Option[NonEmptyList]
    .flatten.toNel
    // if its a Some[NonEmptyList], convert that NEL to a String
    .map(_.fold)
    // get the value, or put the x' value here as a String
    .getOrElse(x.toString)
}

ys.foreach(println)

1
2
Fizz
4
Buzz
Fizz
Woof
8
Fizz
Buzz
11
Fizz
13
Woof
FizzBuzz
16
17
Fizz
19
Buzz
FizzWoof
22
23
Fizz
Buzz
26
Fizz
Woof
29
FizzBuzz
31
32
Fizz
34
BuzzWoof

FizzBuzz using Scalaz Validation

A solution (of many) that involves using Scalaz Validation:

import scalaz._
import Scalaz._

val fizz = (x: Int) =>
  Option(x).filterNot(_ % 3 == 0).map(_.toString).toSuccessNel("Fizz")
val buzz = (x: Int) =>
  Option(x).filterNot(_ % 5 == 0).map(_.toString).toSuccessNel("Buzz")
val woof = (x: Int) =>
  Option(x).filterNot(_ % 7 == 0).map(_.toString).toSuccessNel("Woof")

val rules = List(fizz, buzz, woof)

val ys = (1 to 35).toList.map { x =>
  val ys = List(x) <*> rules
  ys.sequenceU.rightMap(_.head).valueOr(_.fold)
}

ys.foreach(println)

1
2
Fizz
4
Buzz
Fizz
Woof
8
Fizz
Buzz
11
Fizz
13
Woof
FizzBuzz
16
17
Fizz
19
Buzz
FizzWoof
22
23
Fizz
Buzz
26
Fizz
Woof
29
FizzBuzz
31
32
Fizz
34
BuzzWoof

Finding a line in a file that is double

Say, we want to find a line in a file that is double, so the line must be somewhere in the file but is double. Because the solution revolves around aggregating results, or 'appending', Monoids play a big role. Using a library like Scalaz for example, you can use the Monoids from these libraries and saves a lot of boilerplate code so the code now just specifies what to do, and not how to do it:

import java.security.MessageDigest

val lines =
  """
    |Now look at them yo-yos thats the way you do it
    |You play the guitar on the mtv
    |That aint workin thats the way you do it
    |Money for nothin and chicks for free
    |Now that aint workin thats the way you do it
    |Lemme tell ya them guys aint dumb
    |Maybe get a blister on your little finger
    |Maybe get a blister on your thumb
    |
    |We gotta install microwave ovens
    |Custom kitchen deliveries
    |We gotta move these refrigerators
    |We gotta move these colour tvs
    |
    |See the little faggot with the earring and the makeup
    |Yeah buddy thats his own hair
    |That little faggot got his own jet airplane
    |That little faggot hes a millionaire
    |
    |We gotta install microwave ovesns
    |Custom kitchens deliveries
    |We gotta move these refrigerators
    |We gotta move these colour tvs
    |
    |I shoulda learned to play the guitar
    |I shoulda learned to play them drums
    |Look at that mama, she got it stickin in the camera
    |Man we could have some fun
    |And hes up there, whats that? hawaiian noises?
    |Bangin on the bongoes like a chimpanzee
    |That aint workin thats the way you do it
    |Get your money for nothin get your chicks for free
    |
    |We gotta install microwave ovens
    |Custom kitchen deliveries
    |We gotta move these refrigerators
    |We gotta move these colour tvs, lord
    |
    |Now that aint workin thats the way you do it
    |You play the guitar on the mtv
    |That aint workin thats the way you do it
    |Money for nothin and your chicks for free
    |Money for nothin and chicks for free
  """.stripMargin.trim.replace(",", "").replace("'", "").replace("?", "").split("\n")

def hasher(str: String, algo: String = "MD5"): String = {
  val HexChars = "0123456789abcdef".toCharArray
  val digest = MessageDigest.getInstance(algo)
  val bytes: Array[Byte] = digest.digest(str.getBytes)
  val buffer = new StringBuilder(bytes.length * 2)
  bytes.foreach { byte =>
    buffer.append(HexChars((byte & 0xF0) >> 4))
    buffer.append(HexChars(byte & 0x0F))
  }
  buffer.toString
}


import scalaz._
import Scalaz._

// if we append two maps using then we get the following result
(Map("a" -> 1) |+| Map("a" -> 1)) == Map("a" -> 2)

// if we have a List[Map[String, Int]] and we fold the list using suml
// using the available monoids we get the following
List(Map("a" -> 1), Map("a" -> 1)).suml == Map("a" -> 2)

// so if we can just digest the text and create a map then we're all good
// if the key is unique like say a hash, and the value is a '1' then we
// can use the monoid strategy out of the box of scalaz

// what we can do is the following
// 1. the key is still a hash
// 2. The value is a pair
// 3. The pair contains the '1' as the key and a List of rownumbers as the value
List(Map("a" -> (1 -> List(1))), Map("a" -> (1 -> List(2))), Map("a" -> (1 -> List(3))), Map("b" -> (1 -> List(1)))).suml

// so what we must encode is the following:
// 1. A List[Map[String, (Int, List[Int])]
// 2. Each element is a Map
// 3. The map uses the hash as the key
// 4. The value points to a pair of (1, List[Int])
// 5. The value of the pair is a list of rownumber

val hashedLines: Map[String, (Int, List[Int])] = lines.zipWithIndex.map {
  case (str, index) => Map(hasher(str) -> (1 -> List(index)))
}.toList.suml

hashedLines.filter(_._2._1 == 2).flatMap(_._2._2).headOption.map(row => lines(row))

The result is:

res0: Option[String] = Some(Now that aint workin thats the way you do it)

b

What is the difference between a var, a val, lazy val and def?

A var is a variable. It’s a mutable reference to a value. Since it’s mutable, its value may change through the program lifetime, making it a magnet for errors. Keep in mind that the variable type cannot change in Scala. You may say that a var behaves similarly to Java variables.

scala> var x = 1
x: Int = 1

scala> x = 2
x: Int = 2

scala> x = "foo"
<console>:12: error: type mismatch;
 found   : String("foo")
 required: Int
       x = "foo"
           ^

A val is a value. It’s an immutable reference, meaning that its value never changes. Once assigned it will always keep the same value. It’s similar to constants in another languages.

scala> val x = 10
x: Int = 10

scala> x = 42
<console>:12: error: reassignment to val
       x = 42

A def creates a method. It is evaluated on call.

scala> def x: Int = { println("evaluating..."); 42 }
x: Int

scala> x
evaluating...
res0: Int = 42

scala> x
evaluating...
res1: Int = 42

A lazy val is like a val, but its value is only computed when needed. It’s specially useful to avoid heavy computations at the point of the expression, but delaying it until the value is actually needed. It is mostly used when doing dependency injection like with the cake pattern (lite). Lazy vals aren't cheap so they must be used sparingly in your code base:

scala> lazy val x: Int = { println("Evaluating..."); 42 }
x: Int = <lazy>

scala> x
Evaluating...
res2: Int = 42

scala> x
res3: Int = 42

Lazy val deadlock problem

The lazy val deadlock problem is introduced when we have a cyclic dependency so a dependency that goes both ways when using lazy vals. For example, say we have a Foo that depends on Bar and we have a Bar that depends on Foo so Foo <-> Bar:

case class Foo(b: Bar)
case class Bar(f: Foo)

lazy val foo: Foo = Foo(bar)
lazy val bar: Bar = Bar(foo)

defined class Foo
defined class Bar
foo: Foo = <lazy>
bar: Bar = <lazy>

Everything seems fine, until we access a member of either of them:

scala> foo.b
java.lang.StackOverflowError
 at ....

or

scala> bar.f
java.lang.StackOverflowError
  at ...

The best ways to fix this is by breaking the cyclic dependency, the Foo <-> Bar dependency. We can do that by introducing a third type, lets call it Quz and put it between the two so Foo <- Quz -> Bar which means:

case class Foo()
case class Bar()
case class Quz(b: Bar, f: Foo)

lazy val quz: Quz = Quz(bar, foo)
lazy val foo: Foo = Foo()
lazy val bar: Bar = Bar()

Of course that means that we also have refactored the code in Foo and in Bar so that the logic that caused the cyclic dependency is now in Quz. This of course means some refactoring and a change to our model. This goes to show how important it is to not start coding immediately and first create a model...

Now its safe to call quz.b

scala> quz.b
res0: Bar = Bar()

Of course, cyclic dependencies are sometimes bad, like with object graphs as we saw here, and sometimes a good thing like when creating a virtual object graph when referencing eg. entities by UUID when we want to reference an ID in another DDD bounded context for example.

Methods aren't functions

Methods or 'def' aren't functions because methods aren't values. You can however construct a function that delegates to a method via 'eta-expansion' by appending an underscore after a method name or by coersing the expression by using the type system (just stating that the value must be a function) eg:

// define a method
scala> def add(x: Int): Int = x + 1
add: (x: Int)Int

// using coersion
scala> val f: Int => Int = add
f: Int => Int = $$Lambda$1121/1506648430@5dd903be

// using eta-expansion
scala> val g = add _
g: Int => Int = $$Lambda$1122/1966787205@2e645fbd

Methods can also be automatically promoted to a function by just referencing the name of the method eg. in a processing pipeline like the following:

scala> List(1, 2, 3).map(add)
res0: List[Int] = List(2, 3, 4)

What is the difference between a trait and an abstract class?

The first difference is that a class can only extend one other class, but an unlimited number of traits:

scala> class Foo
defined class Foo

scala> class Bar
defined class Bar

scala> class Baz extends Foo with Bar
<console>:13: error: class Bar needs to be a trait to be mixed in
       class Baz extends Foo with Bar

While traits only support type parameters, abstract classes can support both type parameters and constructor parameters:

scala> trait Box[A]
defined trait Box

scala> trait Box[A](a: A)
<console>:1: error: traits or objects may not have parameters
trait Box[A](a: A)

scala> abstract class Box[A](a: A)
defined class Box

Also, abstract classes are interoperable with Java, while traits are only interoperable with Java if they do not contain any implementation. IMHO, for most projects this is a non-issue as the dependency mostly goes from Scala -> Java, ie. we are using Java libraries; not the other way around. If you do, please move from Java to Scala and start having more fun!

What is the difference between an object and a class?

An object is a singleton instance of a class. It cannot be instantiated by the developer. The Scala runtime handles all the necessary initialization issues of the singleton issue of the class.

If an object has the same name that a class, and is in the same scala file, then the object is called a companion object.

scala> :paste
// Entering paste mode (ctrl-D to finish)

class Foo
object Foo {
def apply(): Foo = new Foo
}

// Exiting paste mode, now interpreting.

defined class Foo
defined object Foo

scala> Foo()
res5: Foo = Foo@b3fc6d8

What is a case class?

A case class is syntactic sugar for a class that is immutable and decomposable through pattern matching. This is because the companion object of a case class has an apply and unapply method. Being decomposable means it is possible to extract its constructors parameters in the pattern matching.

Case classes contain a companion object which holds the apply method. This fact makes possible to instantiate a case class without the new keyword. They also come with some helper methods like the .copy method, that eases the creation of a slightly changed copy from the original.

Finally, case classes are compared by structural equality instead of being compared by reference, i.e., they come with a method which compares two case classes by their values/fields, instead of comparing just the references.

Case classes are specially useful to be used as Value Objects, Data Transfer Object definitions and so on.

scala> import io.circe._, io.circe.generic.auto._, io.circe.parser._, io.circe.syntax._

scala> Person("dennis", 42).asJson.noSpaces
res0: String = {"name":"dennis","age":42}

Or a more generic representation of the case class:

scala> import shapeless._

scala> Generic[Person].to(Person("dennis", 42))
res1: String :: Int :: shapeless.HNil = dennis :: 42 :: HNil

Or just do some pattern matching:

scala> Person("dennis", 42) match {
     | case Person(fn, age) => s"Hi $fn, you are $age years old!"
     | }
res2: String = Hi dennis, you are 42 years old!

What is the difference between a Java future and a Scala future?

The Scala implementation is asynchronous without blocking, while in Java you can’t get the future value without blocking. Scala provides an API to manipulate the future as a monad or by attaching callbacks for completion. Unless you decide to use the Await, you won’t block your program using Scala futures.

import scala.concurrent.ExecutionContext.Implicits.global
import scala.concurrent.duration._
import scala.concurrent._

val result: Future[Int] = for {
  x <- Future(1)
  y <- Future(1 + x)
  z <- Future(1 + x + y)
} yield x + y + z

Await.result(result, 1.second)

What is the difference between unapply and apply, when would you use them?

The method 'unapply' is a method that needs to be implemented by an object in order for it to be an extractor. Extractors are used in pattern matching to access an object constructor parameters. It’s the opposite of a constructor.

For example:

object Even {
  def unapply(arg: Int): Option[Int] =
    Option(arg).find(_ % 2 == 0)
}

object Odd {
  def unapply(arg: Int): Option[Int] =
    Option(arg).find(_ % 2 != 0)
}

(1 to 10).map {
  case Even(x) => s"$x is even"
  case Odd(x) => s"$x is odd"
}.foreach(println)

The apply method is a special method that allows you to write someObject(params) instead of someObject.apply(params). This usage is common in case classes, which contain a companion object with the apply method that allows the nice syntax to instantiate a new object without the new keyword:

case class Person(id: Long, name: String, age: Int)

object PersonRepository {
  def apply(): PersonRepository = new InMemoryPersonRepository(Map.empty)
}
trait PersonRepository {
  def getById(id: Long): Option[Person]
}
class InMemoryPersonRepository(repo: Map[Long, Person]) extends PersonRepository {
  override def getById(id: Long): Option[Person] = repo.get(id)
}
PersonRepository().getById(1L)

What is a companion object?

If an object has the same name that a class, and is in the same scala file then the object is called a companion object. A companion object has access to methods of private visibility of the class, and the class also has access to private methods of the object. Doing the comparison with Java, companion objects hold the “static methods” of a class.

Note that the companion object has to be defined in the same source file that the class.

What is the difference between the following terms and types in Scala: Nil, Null, None, Nothing?

The None is the empty representation of the Option Monad which is represented as an Algebraic Data Type (ADT) Option[A] is either a Some[A] or a None.

Null is a Scala trait, where null is its only instance. The null value comes from Java and it’s an instance of any object, i.e., it is a subtype of all reference types, but not of value types. It exists so that reference types can be assigned null and value types (like Int or Long) can’t.

Nothing is another Scala trait. It’s a subtype of any other type, and it has no subtypes. It exists due to the complex type system Scala has. It has zero instances. It’s the return type of a method that never returns normally, for instance, a method that always throws an exception. The reason Scala has a bottom type is tied to its ability to express variance in type parameters. For example, the None is of type Option[Nothing] and the '.get' method of None always throws a 'NoSuchElementException'.

Finally, Nil represents an empty List of anything of size zero and is an object, so there is a single instance. Nil is of type List[Nothing]. If you call '.head' or '.tail' on the Nil object, the methods throw an exception.

What is Unit?

Unit is a type which represents the absence of value, just like Java void. It is a subtype of 'scala.AnyVal'. There is only one value of type Unit, represented by '()', and it is not represented by any object in the underlying runtime system.

Resources