Reputation: 15284
I have an abstract class with an unimplemented method numbers
that returns a list of numbers, and this method is used in another val property initialization:
abstract class Foo {
val calcNumbers = numbers.map(calc)
def numbers: List[Double]
}
The implementing class implements using a val expression:
class MainFoo extends Foo {
val numbers = List(1,2,3)
}
This compiles fine, but at run time it throws a NullPointerException and it points to the line of val calcNumbers
:
[error] (run-main-0) java.lang.ExceptionInInitializerError
[error] java.lang.ExceptionInInitializerError
...
[error] Caused by: java.lang.NullPointerException
...
However when I changed the implemented method to def, it works:
def numbers = List(1,2,3)
Why is that? Does it have something to do with initialization order? How can this be avoided in the future as there is no compile time error/warning? How does Scala allow this unsafe operation?
Upvotes: 1
Views: 311
Reputation: 44908
Here is what your code attempts to do when it initializes MainFoo
:
val calcNumbers
and val numbers
, initially set to 0
.Foo
, where it attempts to invoke numbers.map
while initializing calcNumbers
.MainFoo
, where it initializes numbers
to List(1, 2, 3)
.Since numbers
is not initialized yet when you try to access it in val calcNumbers = ...
, you get a NullPointerException
.
Possible workarounds:
numbers
in MainFoo
a def
numbers
in MainFoo
a lazy val
calcNumbers
in Foo
a def
calcNumbers
in Foo
a lazy val
Every workaround prevents that an eager value initialization invokes numbers.map
on a non-initialized value numbers
.
The FAQ offers a few other solutions, and it also mentions the (costly) compiler flag -Xcheckinit
.
You might also find these related answers useful:
Upvotes: 3