5

这可能以前被问过很多次,但我没有找到任何帮助。

我有一个简单的 Scala 代码,它生成取决于一些副作用的长数字。我将事物包装在 IO monad 中,但根据最小功率原则,我实际上将我的函数声明为F[_]: Effect. 现在代码无法编译,我不明白为什么,请提出可能有问题的地方

import cats.effect.{Clock, Effect}
import cats.syntax.all._
import java.util.concurrent.TimeUnit


...

  def generateId[F[_]: Effect](rid: Long)(implicit F: Effect[F], clock: Clock[F]): F[Long] =
    for {
      currentTimeNanos <- clock.realTime(TimeUnit.NANOSECONDS)
      tid              <- F.delay(Thread.currentThread().getId)
    } yield
      (tid << 40 /*    */ & 0xFFFFFF0000000000L) |
        (rid << 16 /*  */ & 0x000000FFFFFF0000L) |
        (currentTimeNanos & 0x000000000000FFFFL)

[error] /.../package.scala:34:41: value flatMap is not a member of type parameter F[Long]
[error]       currentTimeNanos <- clock.realTime(TimeUnit.NANOSECONDS)
[error]                                         ^
[error] /.../package.scala:35:34: value map is not a member of type parameter F[Long]
[error]       tid              <- F.delay(Thread.currentThread().getId)

另外,如果您对改进代码有任何建议,请告诉我。

4

1 回答 1

8

问题是脱糖中的上下文绑定F[_]: Effect到一个隐式参数中,所以编译器看到的是这样的:

def generateId[F[_]](rid: Long)(implicit ev: Effect[F], F: Effect[F], ...): F[Long] = ...

这意味着每次尝试解析Effect[F]方法主体中的隐式时,它都会失败,因为它认为显式F和合成ev是模棱两可的。

解决方案是删除上下文绑定或显式隐式F: Effect[F]参数。我建议取消上下文绑定,因为 Scala 允许您将两者结合起来,这是很容易犯这种错误的部分原因(在我看来,这是语言设计者的严重误判,因为我之前已经说过很多次了)。

于 2019-02-18T07:49:27.473 回答