Как явно вызывать и использовать Functor для функции

import scala.language.higherKinds
import cats.Functor
import cats.instances.list._
import cats.instances.function._

val list1 = List(1, 2)

val list2 = Functor[List].map(list1)(i => i + 1)

Но вещи не работают так гладко для функций,

val f1 = (i: Int) => i.toString
val f2 = (s: String) => s

И мы должны прибегнуть к обману типа,

scala> type TTT[A] = Int => A
// defined type alias TTT

scala> val ff = Functor[TTT].map(f1)(f2)
// ff: TTT[String] = scala.Function1$$Lambda$607/69160933@3a689738

Хорошо... Есть ли более прямой способ решить эту проблему, так как это становится очень утомительным для сложных функций.

2 ответа

Решение

Ты можешь написать

val ff = Functor[Int => ?].map(f1)(f2)

если вы добавите addCompilerPlugin("org.spire-math" %% "kind-projector" % "0.9.7") в build.sbt,

Или вы можете импортировать синтаксис функтора import cats.syntax.functor._ а затем написать с типом надписи

val ff = (f1: TTT[String]).map(f2)

или с явной аннотацией типа

val f1: TTT[String] = (i: Int) => i.toString
val ff = f1.map(f2)

Зачем вам явно вызывать экземпляр класса типа? Cat включает в себя поддержку синтаксиса для функтора, поэтому в этом нет необходимости.

import cats.syntax.functor._
import cats.instances.function._
val mul4 = ((a: Int) => a * 2).map(_ * 2)

Чтобы это работало, вам понадобится -Ypartial-unification флаг компилятора (scalacOptions += "-Ypartial-unification" если вы используете SBT). Этот флаг, вероятно, будет включен по умолчанию в будущих версиях Scala.

Другие вопросы по тегам