如何初始化子特征中的特征值?

时间:2015-01-05 14:44:15

标签: scala traits

我尝试在特征中使用抽象val来初始化另一个值。我得到了NullPointerException。我将这种行为归结为一个最小的测试用例:

trait MessagePrinter {
  val message: String
  println(message)
}

class HelloPrinter extends MessagePrinter {
  val message = "Hello World"
}

val obj = new HelloPrinter()
println(obj.message)

这个小程序产生以下结果:

null
Hello World

我的印象是val可能永远不会改变。这是预期的行为还是编译器错误?如何解决此问题并在初始化期间打印Hello World

2 个答案:

答案 0 :(得分:11)

根据Scala specification的第5.1节,首先初始化超类。即使vals通常无法重新实例化,它们也会在构造期间以默认初始值开始。您可以使用def,它具有不同的语义:

trait MessagePrinter {
   def message: String
   println(message)
}

class HelloPrinter extends MessagePrinter {
   def message = "Hello World"
}

或者你可以考虑改变这样的事情:

class HelloPrinter extends { val message = "Hello World" } with MessagePrinter 

在这种情况下,按顺序评估超类,以便MessagePrinter初始化应该按照需要工作。

答案 1 :(得分:3)

在这两种情况下都应该使用def

描述此行为的其中一个来源是“Scala Puzzlers”Puzzler 4:

  

以下规则控制初始化和覆盖行为   vals:

     
      
  1. 超类在子类之前完全初始化。
  2.   
  3. 会员按照宣布的顺序进行初始化。
  4.   
  5. 当覆盖val时,它仍然只能初始化一次。
  6.   
  7. 与抽象val一样,在构造超类时,重写的val将具有默认的初始值。
  8.