是否可以在Scala中定义构造函数本地变量?

时间:2012-01-12 22:09:59

标签: scala constructor

以下代码摘自Martin Odersky等人的Scala编程书。它定义了一种理性类型:

class Rational(n: Int, d: Int) { 
  require(d != 0)
  private val g = gcd(n.abs, d.abs)
  val numer = n / g 
  val denom = d / g
  ...
  private def gcd(a: Int, b: Int): Int = if (b == 0) a else gcd(b, a % b)
}

这里的值g仅在隐式构造函数初始化字段numer和denom时使用。假设程序员知道它不会在其他任何地方使用。在上面的例子中,它在构造Rational对象后仍然可以访问。这意味着它也将占用空间,因为它是一个私有字段,而不是构造函数的局部变量。

我的问题是如何更改此代码,以便g仅在构造时使用然后扔掉?

1 个答案:

答案 0 :(得分:10)

在这种情况下,这样的事情怎么样?

class Rational(n: Int, d: Int) {
  require(d != 0)
  val (numer, denom) = {
    val g = gcd(n.abs, d.abs)
    (n / g, d / g)
  }
  private def gcd(a: Int, b: Int): Int = if (b == 0) a else gcd(b, a % b)
}

编辑: 这也会创建一个包含元组的额外字段,如在编译的类上运行javap所示(感谢Alexey): / p>

public class Rational extends java.lang.Object implements scala.ScalaObject{
    private final scala.Tuple2 x$1; // actually unwanted!
    private final int numer;
    private final int denom;
    public int numer();
    public int denom();
    private int gcd(int, int);
    public Rational(int, int);
}

在其他情况下,我有时会使用locally块来避免将每个val转换为字段:

class A {
  locally {
    val value1 = // ...
    val value2 = // ...
  }
}