scala - 是什么 ?类型?

标签 scala scala-cats

我正在尝试实现一只猫 Monad具有多个类型参数的类型的实例。我看着猫 Either实例看看它是如何在那里完成的。部分Either Monad猫的实例代码复制如下:

import cats.Monad

object EitherMonad {
  implicit def instance[A]: Monad[Either[A, ?]] =
    new Monad[Either[A, ?]] {
      def pure[B](b: B): Either[A, B] = Right(b)

      def flatMap[B, C](fa: Either[A, B])(f: B => Either[A, C]): Either[A, C] =
        fa.right.flatMap(f)
    }
}

编译失败,错误:error: not found: type ?
什么是?类型以及在为我自己的类型创建实例时如何使用它?

最佳答案

它是由 kind projector plugin 添加的所谓类型 lambda 的特殊语法。 .

Either[A, ?]

是捷径
({type L[X] = Either[A, X]})#L

整个代码脱糖为
import cats.Monad

object EitherMonad {
  implicit def instance[A]: Monad[({type L[X] = Either[A, X]})#L] = new Monad[({type L[X] = Either[A, X]})#L] {
    def pure[B](b: B): Either[A, B] = Right(b)

    def flatMap[B, C](fa: Either[A, B])(f: B => Either[A, C]): Either[A, C] =
      fa.right.flatMap(f)
  }
}

类型 lambda 看起来很可怕,但它们本质上是一个非常简单的概念。你有一个需要两个类型参数的东西,比如 Either[A, B] .你想为Either 提供一个monad 实例,但是trait Monad[F[_]]只接受一个类型参数。但原则上没问题,因为你的 monad 实例只关心第二个(“正确的”)类型参数。类型 lambda 只是“修复”第一个类型参数的一种方法,因此您具有正确的形状。

如果你会在值(value)层面做同样的事情,你甚至不会考虑两次。你有两个参数的函数
val f: (Int, Int) => Int = ...

还有你想传递给 f 的东西,它只需要 1 个参数
def foo(x: Int => Int) = ...

使事情合适的唯一方法是修正其中一个论点
foo(x => f(1, x))

这正是类型 lambda 在类型级别所做的。

关于scala - 是什么 ?类型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34386103/

相关文章:

scala - 访问 DStream 集合

scala - 使用 cats-effect 的 IO monad 进行单元测试

scala - 在 IntelliJ 中使用猫库时的错误错误

scala - 认证路由中的 http4s json 处理

scala - 使用 Future[Unit] 时无法解析方法 success.unapply

scala - Spark Dataframe Join - 重复列(非连接列)

scala - 在 build.sbt 文件中使用外部库

java - 使用 AWS Java DynamoDB 流 Kinesis 适配器处理 DynamoDB 流

Scala 元组添加保持序列相同的顺序

scala - 与猫的三个列表的笛卡尔积