60

在 Scala 中,类的主构造函数没有显式的主体,而是从类主体中隐式定义的。那么,如何区分字段和本地值(即构造函数方法的本地值)?

例如,采用以下代码片段,它是“Scala 编程”中一些示例代码的修改形式:

class R(n: Int, d: Int) {
   private val g = myfunc
   val x = n / g
   val y = d / g
}

我的理解是,这将生成一个具有三个字段的类:私有“g”和公共“x”和“y”。但是,g 值仅用于计算 x 和 y 字段,超出构造函数范围没有任何意义。

所以在这个(当然是人为的)例子中,你如何为这个构造函数定义局部值?

4

4 回答 4

37

例如

class R(n: Int, d: Int) {
  val (x, y) = {
    val g = myfunc
    (n/g, d/g)
  }
}
于 2009-07-13T11:39:20.903 回答
16

有几种方法可以做到这一点。您可以在私有定义中声明此类临时变量,以便在构建期间使用。您可以在返回表达式的块内使用临时变量(例如在 Alaz 的答案中)。或者,最后,您可以在备用构造函数中使用此类变量。

以类似于替代构造函数的方式,您还可以在对象伴侣的“应用”方法中定义它们。

不能做的是声明一个字段是“临时的”。

另请注意,主构造函数接收的任何参数也是一个字段。如果您不希望这些参数成为字段,并且不想在构造函数中公开实际字段,通常的解决方案是使用实际字段将主构造函数设为私有,并使用备用构造函数或object-companion 的 apply() 作为有效的“主要”构造函数。

于 2009-07-13T12:19:23.600 回答
13

我们的另一个选择是将主对象构造函数设为私有,并使用伴随对象的 apply 方法作为构建器。如果我们将这种方法应用于您的示例(不是双关语),它将如下所示:

class R private (val x: Int, val y: Int);

object R {
  def apply(n: Int, d: Int): R = {
    val g = myfunc;
    new R(n / g, d / g);
  }
}

要创建 R 实例而不是:

val r = new R(1, 2);

写:

val r = R(1, 2);

这有点冗长,但我认为可能会更糟:)。让我们希望 private[this] val 在未来的 Scala 版本中被视为临时变量。马丁本人暗示了这一点。

于 2010-01-28T19:46:15.353 回答
6

关于这个主题的一些讨论,包括 Martin Odersky 的评论,在这里

于 2009-07-13T22:44:49.957 回答