5

我有大对象:

case class BigObject(val str: String, val number: Int) {
val someVal = ...
val someVal2 = ...
    }

我想复制这个对象而不重新评估值。是否可以?现在我正在使用这种方法:

val newBigObject = oldBigObject.copy(str = newStr)

正如我从日志/调试器中看到的那样,“someVal”和“someVal2”被重新评估。有可能避免吗?由于我的 BigObject 非常大,价值重新评估需要一些时间,但性能对我来说真的很重要。

感谢您的回答!

4

1 回答 1

8

这里有一个方法:

制作也传递给构造函数的 and 字段,并为伴生对象中的这些字段提取初始化逻辑someValsomeVal2

在你的情况下:

class BigObject private(val str: String,
                        val number: Int,
                        val someVal: SomeType,
                        val someVal2: SomeType) {

   def copy(newStr: String = str, newNumber: Int = number) = {
     new BigObject(newStr, newNumber, someVal, someVal2)
   }
}

object BigObject {

  def apply(str: String, number: Int): BigObject = {
    val someVal = initialize() //your initialization logic here
    val someVal2 = initialize2()
    new BigObject(str, number, someVal, someVal2)
  }

}

现在,您可以在不重新评估内部字段的情况下进行复制:

val bigObj = BigObject("hello", 42)
val anotherBigObj = bigObj.copy(newStr = "anotherStr")

或者,如果您不喜欢伴随对象,您可以创建两个构造函数。主要字段包括所有字段(也包括不可见字段)并且是私有的。公开的只有两个可见参数:

class BigObject private(val str: String,
                        val number: Int,
                        val someVal: Any,
                        val someVal2: Any) {

  def this(str: String, number: Int) = this(str, number, initializeVal, initializeVal2)

  def copy(newStr: String = str, newNumber: Int = number) = {
    new BigObject(newStr, newNumber, someVal, someVal2)
  }

}

用法:

val bigObj = new BigObject("hello", 42)
val anotherBigObj = bigObj.copy(newStr = "anotherStr")
于 2013-03-22T10:43:46.527 回答