Running the code below:
class Parent {
val value = {
println("Setting value in parent")
"ParentVal"
}
println(s"Parent value is ${value}")
}
class Child extends Parent {
override val value = {
println("Setting value in child")
"ChildVal"
}
println(s"Child value is ${value}")
}
new Child
Produces this output:
Setting value in parent
Parent value is null
Setting value in child
Child value is ChildVal
So the code associated with the parent value assignment is run, however the value is not really assigned at the parent. Afterwards the child code runs and it assigns the value as expected.
Could someone explain the chain of events here at a lower level?
You can think about
val
as if it were a combination of a private member variable without setters + a getter method. If you rewrite your code withoutval
s, it would be something like this:Just as the original, it prints:
This is because the parent constructor invokes the method
value
, but this method is overridden bydef value = childValue
. The returnedchildValue
isnull
, because when the parent constructor is invoked, the child constructor has not been invoked yet.You can read more about object initialization order here.
Related answers:
Why does implement abstract method using val and call from superclass in val expression return NullPointerException.
Assertion with
require
in abstract superclass creates NPE