Scala: how to initialize an object using default values

27,092

If you want the default value to be used, you normally just leave off that named argument:

scala> val p = Person(name = "new person name")
p: Person = Person(new person name,no surname)

But, since you want to explicitly know whether a value should be defaulted or not, you could implement your Map-based idea in a constructor. If you don't want to repeat the defaults, how about these two options:

Option 1: Externalized constants for defaults

Set the defaults externally. Use them in both the main constructor and the Map-based constructor.

val nameDefault = "no name"
val surnameDefault = "no surname"

case class Person(name: String = nameDefault, surname: String = surnameDefault) {
  def this(m: Map[String, String]) =
    this(m.getOrElse("name", nameDefault), m.getOrElse("surname", surnameDefault))
}

Usage:

new Person(name = "new person name", surname = "new person surname")
new Person(Map("name" -> "new person name"))
new Person(name = "new person name")

Option 2: Optionified alternate constructor

You may find this a little cleaner since it doesn't rely on externalized constants. The only downside here is that if you want to construct with only some of the parameters, you have to wrap each one in Some().

case class Person(name: String, surname: String) {
  def this(name: Option[String] = None, surname: Option[String] = None) =
    this(name.getOrElse("no name"), surname.getOrElse("no surname"))

  def this(m: Map[String, String]) = this(m.get("name"), m.get("surname"))
}

Usage:

new Person(name = "new person name", surname = "new person surname")
new Person(Map("name" -> "new person name"))
new Person(name = Some("new person name"))
new Person(name = "new person name") // can't do this
Share:
27,092
opensas
Author by

opensas

Updated on July 29, 2022

Comments

  • opensas
    opensas almost 2 years

    I think this will be better explained with an example

    I have the following case class

    case class Person(name: String = "no name", surname: String = "no surname")
    

    And I want to make a general function to populate it from, for example, a json message, that might not specify all fields

    I know that to use the default values the simple answer is not to pass them to the constructor, but if I have several fields that may or may not appear in the json, I should have to use a huge switch sentence covering every possible combination of missing parameters. In this case, after reading the json, I should take care of name & surname present, no name, no surname and no name nor surname case... (Gee, I hope I made myself clear).

    To be more precise, I'm trying to develop a function that allows me to create a person from the following json values, using the default values when there's some parameter missing

    { "name": "john", "surname": "doe" }
    { "surname": "doe" }
    { "name": "john" }
    { }
    

    That's why I'm looking for a more general way to handle this.

    (I'll show some pseudo code to give and idea of what I'm trying to achieve)

    I was thinking about something like:

    val p = Person(name= "new person name", surname= Unit)
    

    And in that case surname should get the default value

    Or something like

    val p = Person( Map( "name" -> "new person name" ) _* )
    

    So that it also takes the default value for surname

    Or maybe doing it in the constructor, if I detect a null value (or None) I could assign the default value.

    In fact, I'm trying to avoid repeating the definition of the default values.

    Anyway, what would be the most idiomatic way to achieve such a thing?