Problem
You want to create an API so developers can write code in a fluent programming style, also known as method chaining.
Solution
A fluent style of programming lets users of your API write code by chaining method calls together, as in this example:
person.setFirstName("Leonard") .setLastName("Nimoy") .setAge(82)
.setCity("Los Angeles") .setState("California")
To support this style of programming:
• If your class can be extended, specify this.type as the return type of fluent style methods.
• If you’re sure that your class won’t be extended, you can optionally return this from your fluent style methods.
The following code demonstrates how to specify this.type as the return type of the set* methods:
class Person {
protected var fname = ""
protected var lname = ""
def setFirstName(firstName: String): this.type = { fname = firstName
this }
def setLastName(lastName: String): this.type = { lname = lastName
this } }
class Employee extends Person { protected var role = ""
def setRole(role: String): this.type = { this.role = role
this }
override def toString = {
"%s, %s, %s".format(fname, lname, role) }
}
The following test object demonstrates how these methods can be chained together:
object Main extends App { val employee = new Employee // use the fluent methods employee.setFirstName("Al") .setLastName("Alexander") .setRole("Developer") println(employee)
}
Discussion
If you’re sure your class won’t be extended, specifying this.type as the return type of your set* methods isn’t necessary; you can just return the this reference at the end of each fluent style method. This is shown in the addTopping, setCrustSize, and setCrustType methods of the following Pizza class, which is declared to be final:
final class Pizza {
import scala.collection.mutable.ArrayBuffer private val toppings = ArrayBuffer[String]() private var crustSize = 0
private var crustType = ""
def addTopping(topping: String) = { toppings += topping
this }
def setCrustSize(crustSize: Int) = { this.crustSize = crustSize
this }
def setCrustType(crustType: String) = { this.crustType = crustType
this }
def print() {
println(s"crust size: $crustSize") println(s"crust type: $crustType") println(s"toppings: $toppings") }
}
This class is demonstrated with the following driver program:
object FluentPizzaTest extends App { val p = new Pizza
p.setCrustSize(14) .setCrustType("thin") .addTopping("cheese") .addTopping("green olives") .print()
}
This results in the following output:
crust size: 14 crust type: thin
toppings: ArrayBuffer(cheese, green olives)
Returning this in your methods works fine if you’re sure your class won’t be extended, but if your class can be extended—as in the first example where the Employee class extended the Person class—explicitly setting this.type as the return type of your set* methods ensures that the fluent style will continue to work in your subclasses. In this example, this makes sure that methods like setFirstName on an Employee object return an Employee reference and not a Person reference.
See Also
• Definition of a fluent interface
• Method chaining
• Martin Fowler’s discussion of a fluent interface
CHAPTER 6
Objects
Introduction
The word “object” has a dual meaning in Scala. As with Java, you use it to refer to an instance of a class, but in Scala, object is also a keyword.
The first three recipes in this chapter look at an object as an instance of a class, show how to cast objects from one type to another, demonstrate the Scala equivalent of Java’s .class approach, and show how to determine the class of an object.
The remaining recipes demonstrate how the object keyword is used for other purposes.
You’ll see how to use it to launch Scala applications and to create Singletons. There’s also a special type of object known as a package object. Using a package object is entirely optional, but it provides a nice little out-of-the-way place where you can put code that’s common to all classes and objects in a particular package level in your application. For instance, Scala’s root-level package object contains many lines of code like this:
type Throwable = java.lang.Throwable type Exception = java.lang.Exception type Error = java.lang.Error
type Seq[+A] = scala.collection.Seq[A]
val Seq = scala.collection.Seq
Declaring those type definitions in Scala’s root package object helps to make the rest of the code a little bit cleaner, and also keeps these definitions from cluttering up other files.
You’ll also see how to create a companion object to solve several problems. For instance, one use of a companion object is to create the equivalent of Java’s static members. You can also use a companion object so consumers of its corresponding class won’t need to use the new keyword to create an instance of the class. For example, notice how the new keyword isn’t required before each Person instance in this code:
val siblings = List(Person("Kim"), Person("Julia"), Person("Kenny"))
These solutions, and a few more, are presented in this chapter.