DEV Community

Cover image for Builder pattern in Kotlin
bright inventions
bright inventions

Posted on • Edited on • Originally published at brightinventions.pl

Builder pattern in Kotlin

Builder pattern might be used in regular code, but I personally find it the most useful in tests. This is where you need to create objects which on one hand need to meet certain circumstances and on the other hand you do not want to be bothered with other parameters. The builder pattern is the answer.

Let’s jump straight to a code example. We have a basic class representing a person that can have a number of friends.

class Person (var name: String,
    var surname: String,
    var age: Int,
    var gender: Gender,
    var friends: Array<Person>)
Enter fullscreen mode Exit fullscreen mode

There is also an enum:

enum Gender {
    Male,
    Female
}
Enter fullscreen mode Exit fullscreen mode

Now we will create the builder class. First of all we need a private field for every field of our Person class together with a meaningful and valid default value.

class PersonBuilder {
    private var name: String = "John",
    private var surname: String = "Doe",
    private var age: Int = 21
    private var gender: Gender = Gender.Mail,
    private var friends: MutableArray<Person> = mutableArrayOf()
}
Enter fullscreen mode Exit fullscreen mode

Now we need a method for each parameter to make possible to overwrite it. Each method is returning this so that we can chain the methods and therefore achieve the fluent interface.

fun withName(name: String) {
    this.name = name
    return this
}
Enter fullscreen mode Exit fullscreen mode

To get the actual object we need to have one more method - the build method.

fun build() : Person = Person(name, surname, age, gender, friends.toTypedArray())
Enter fullscreen mode Exit fullscreen mode

We can use our builder class in a test method.

val person = PersonBuilder()
    .withName("Jane")
    .withGender(Gender.Female)
    .build()
Enter fullscreen mode Exit fullscreen mode

In the test methods I often need just one element in the array, so I find it useful to create another method which allows for adding a single person to the array:

fun withFriend(friend: Person) : PersonBuilder {
    this.friends.add(friend)
    return this
}
Enter fullscreen mode Exit fullscreen mode

Another great thing you can use along with the builder pattern are domain specific methods. Say you need an under-aged person for your tests. You could just pass the age of say 15 into the withAge method:

val underAgedPerson = PersonBuilder().whitAge(15).build()
Enter fullscreen mode Exit fullscreen mode

Does it clearly show the purpose? Not really. This is a kind of a magic number. Let’s add two more methods to our builder class:

fun withAgeOfMaturity() {
    this.age = 19
    return this
}
fun withAgeOfUnderAged() {
    this.age = 15
    return this
}
Enter fullscreen mode Exit fullscreen mode

And now we can use it as so:

val underAgedPerson = PersonBuilder().withAgeOfUnderAged().build()
Enter fullscreen mode Exit fullscreen mode

Another typical case is when you need a set of parameter values mixed together to create a meaningful object. Say you want to create a woman person - not only do you want to set the gender but also add some feminine name.

val woman = PersonBuilder().withGender(Gender.Female).withName("Jane").build()
Enter fullscreen mode Exit fullscreen mode

You can add a build method for this purpose:

fun buildWoman() : Person = Person("Jane", surname, age, Gender.Female, friends.toTypedArray())
Enter fullscreen mode Exit fullscreen mode

Creating another build method for every case you can think of is a waste of time and energy. There are two main conditions to be met to decide that you need a specific build method:

  • a set of values tend to repeat in your tests,
  • the set of values together create a specific domain object.

You will know the second condition is met if you can name the build method with the domain specific language :)

Kotlin Named And Default Arguments

There is another approach I’ve come along recently which made me realize I was using Java approach poorly adapted in Kotlin. You can make a great use of Kotlin default arguments as well as named arguments in the constructor. You can read the whole story here or move on to an example.

Instead of writing each with method you just declare the private fields as optional constructor parameters. The build method stays the same as before.

class PersonBuilder (
    var name: String = John,
    var surname: String = Doe,
    var age: Int = 21
    var gender: Gender = Gender.Mail,
    var friends: MutableArray<Person> = mutableArrayOf()) {

    fun build() : Person = Person(name, surname, age, gender, friends.toTypedArray())
}
Enter fullscreen mode Exit fullscreen mode

In the forementioned blog post, the author uses immutable properties. However, if we use mutable ones, we can still make use of the additional domain specific methods. This is how the builder class looks now:

class PersonBuilder (var name: String = "John",
    var surname: String = "Doe",
    var age: Int = 21
    var gender: Gender = Gender.Mail,
    var friends: MutableArray<Person> = mutableArrayOf()) {

    fun build() : Person = Person(name, surname, age, gender, friends.toTypedArray())

    fun buildWoman() : Person = Person("Jane", surname, age, Gender.Female, friends.toTypedArray())

    fun withFriend(friend: Person) : PersonBuilder {
        this.friends.add(friend)
        return this
    }

    fun withAgeOfMaturity() : PersonBuilder {
        this.age = 19
        return this
    }

    fun withAgeUnderAged() : PersonBuilder {
        this.age = 15
        return this
    }
}
Enter fullscreen mode Exit fullscreen mode

And this is how we can use it in tests:

val underAgedPerson = PersonBuilder(name = "Mickey").withAgeUnderAged().build()
val woman = PersonBuilder(surname = "Smith").buildWoman()
Enter fullscreen mode Exit fullscreen mode

Summary

Although I am an enthusiast of TDD and clearly see all the advantages of this approach, sometimes it feels really tempting to ignore the tests and go straight to coding. The more convenient tests are, the more willing you are to write them. The builder pattern is an example of what makes the tests easier to write. And easier to read, which after all might be even more important :)

Image of Bright Data

Scale Your Data Needs Effortlessly – Expand your data handling capacities seamlessly.

Leverage our scalable solutions to meet your growing data demands without compromising performance.

Scale Effortlessly

Top comments (1)

Collapse
 
jmfayard profile image
Jean-Michel 🕵🏻‍♂️ Fayard • Edited

Well you don't need this
The builder pattern is integrated in Kotlin in the language itself.
Define a normal data class with immutable properties, that's it, you have exactly what you needed in Java when you needed a builder class.
The only case for using the actual Java builder pattern is if you release a Kotlin library and want to provide this feature to Java programmers.

Billboard image

Use Playwright to test. Use Playwright to monitor.

Join Vercel, CrowdStrike, and thousands of other teams that run end-to-end monitors on Checkly's programmable monitoring platform.

Get started now!