14

我认为这将通过一个例子更好地解释

我有以下案例类

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

而且我想创建一个通用函数来填充它,例如,一个可能没有指定所有字段的 json 消息

我知道要使用默认值,简单的答案是不要将它们传递给构造函数,但是如果我有几个字段可能会或可能不会出现在 json 中,我应该使用一个巨大的 switch 语句来覆盖所有可能的组合缺少参数。在这种情况下,在阅读了 json 之后,我应该注意 name & surname present,no name,no surname and no name or surname case ...(哎呀,我希望我说清楚了)。

更准确地说,我正在尝试开发一个函数,允许我从以下 json 值创建一个人,当缺少某些参数时使用默认值

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

这就是为什么我正在寻找一种更通用的方法来处理这个问题。

(我将展示一些伪代码来说明我想要实现的目标)

我在想类似的事情:

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

在这种情况下,姓氏应该获得默认值

或者类似的东西

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

这样它也采用姓氏的默认值

或者也许在构造函数中这样做,如果我检测到一个空值(或无),我可以分配默认值。

事实上,我试图避免重复默认值的定义。

无论如何,实现这样的事情最惯用的方法是什么?

4

2 回答 2

23

如果要使用默认值,通常只需省略该命名参数:

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

但是,由于您想明确知道一个值是否应该被默认,您可以在构造函数中实现基于 Map 的想法。如果您不想重复默认设置,那么这两个选项如何:

选项 1:默认值的外部化常量

在外部设置默认值。在主构造函数和基于 Map 的构造函数中使用它们。

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))
}

用法:

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

选项 2:可选的替代构造函数

您可能会发现这更简洁,因为它不依赖于外部常量。这里唯一的缺点是,如果你想只使用一些参数进行构造,你必须将每个参数都包装在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"))
}

用法:

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
于 2012-05-06T17:47:05.317 回答
0

我认为这可能是 Either 的一个用例。您可以指定具有两种类型的 Either。

val name: Eiter[String, Unit] = Left("name")
val surname: Either[String, Unit] = Right( () )

现在您可以检查您是否有一个 Left 或 Right 并使用您想要的参数调用构造函数。

于 2012-05-06T18:58:40.317 回答