如何在Scala中的主构造函数中定义局部var / val?

时间:2009-07-13 10:18:00

标签: scala scala-primary-constructor

在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 个答案:

答案 0 :(得分:37)

E.g。

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

答案 1 :(得分:16)

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

以类似于备用构造函数的方式,您还可以在object-companion的“apply”方法中定义它们。

无法做的是将字段声明为“临时”。

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

答案 2 :(得分:12)

我们的另一个选择是将主对象构造函数设为私有,并使用随播对象的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);

这有点冗长,但可能会更糟,我认为:)。让我们希望私有[this] vals将在未来的Scala版本中被视为临时变量。马丁自己暗示说。

答案 3 :(得分:6)

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