Аннотация: scala.annotation.tailrec
.Он вызывает ошибку компилятора, если метод не может быть оптимизирован с помощью хвостового вызова, что происходит, если:
- Рекурсивный вызов не находится в хвостовой позиции
- Метод может быть переопределен
- Метод не является окончательным (особый случай предыдущего)
Он помещается непосредственно перед def
в определении метода.Это работает в REPL.
Здесь мы импортируем аннотацию и пытаемся пометить метод как @tailrec
.
scala> import annotation.tailrec
import annotation.tailrec
scala> @tailrec def length(as: List[_]): Int = as match {
| case Nil => 0
| case head :: tail => 1 + length(tail)
| }
<console>:7: error: could not optimize @tailrec annotated method: it contains a recursive call not in tail position
@tailrec def length(as: List[_]): Int = as match {
^
Упс!Последний вызов 1.+()
, а не length()
!Давайте переформулируем метод:
scala> def length(as: List[_]): Int = {
| @tailrec def length0(as: List[_], tally: Int = 0): Int = as match {
| case Nil => tally
| case head :: tail => length0(tail, tally + 1)
| }
| length0(as)
| }
length: (as: List[_])Int
Обратите внимание, что length0
автоматически является частным, поскольку оно определено в области действия другого метода.