Я пытаюсь что-то сделать с монадами в scala, используя библиотеку scalaz, и у меня возникают проблемы с тем, чтобы он хорошо работал с подтипами.
Я начал с определения моей собственной монады. Пусть это будет монада идентичности ради простоты:
import scalaz._
import Scalaz._
class Id[+A] (val value : A) { }
implicit object IdMonad extends Monad[Id] {
override def pure[A](a : => A) = new Id(a)
override def bind[A, B](a : Id[A], f : A => Id[B]) = f(a.value)
}
Далее я расширил его некоторыми дополнительными функциями:
class ExtendedId[A] (value : A, val printer : A => String) extends Id[A](value) { }
С этой дополнительной функциональностью ExtendedId
больше не является монадой.
Теперь я хочу использовать объект типа ExtendedId[A]
как Id[A]
:
def increment1(v : ExtendedId[Int]) : Id[Int] = {
for(v <- v) yield v + 1;
// ^
// error: could not find implicit value for parameter t: scalaz.Functor[test.package.ExtendedId]
}
Обратите внимание, что я понимаю, что, поскольку ExtendedId
не является монадой, лучшее, что я могу получить в качестве результата, это Id[Int]
, и я согласен с этим! Но, к сожалению, этот код все еще не компилируется.
Однако этот делает:
def asId[A](a : ExtendedId[A]) : Id[A] = a
def increment2(v : ExtendedId[Int]) {
for(v <- asId(v)) yield v + 1;
}
Здесь, функция asId
не делает ничего, кроме как передвигает свой аргумент с ExtendedId[A]
до Id[A]
. Кажется, что он должен быть полностью избыточным, но это не так.
Почему это происходит? Существует неявное преобразование из Id[A]
в объект, содержащий map
, и, очевидно, существует тривиальное неявное преобразование из ExtendedId[A]
в Id[A]
. Итак, почему компилятор не может объединить их?