Cats Effect IO:使用 Scala 集合组合 IO

pat*_*rit 4 functional-programming scala scala-cats cats-effect

这是一个玩具 Scala 程序,它从控制台读取 10 个数字并将每个数字加 1 打印出来:

import cats._
import cats.effect._
import cats.implicits._

object Main extends IOApp {
  def printLine[A](x: A)(implicit show: Show[A]): IO[Unit] =
    IO(println(show.show(x)))

  def readLine(): IO[String] =
    IO(scala.io.StdIn.readLine())

  def readInts(n: Int): IO[List[Int]] =
    List.fill(n)(readLine().map(_.toInt)).sequence

  override def run(args: List[String]): IO[ExitCode] = {
    for {
      list <- readInts(10)
      i <- list
      _ <- printLine(i + 1)
    } yield ExitCode.Success
  }
}
Run Code Online (Sandbox Code Playgroud)

上面的代码不能编译。我得到:

[error]  found   : cats.effect.IO[cats.effect.ExitCode]
[error]  required: scala.collection.GenTraversableOnce[?]
[error]       _ <- printLine(i + 1)
Run Code Online (Sandbox Code Playgroud)

我究竟做错了什么?

Las*_*asf 5

你的理解是你的IO,但你把它和List. 为了让它更清楚一点,我们可以扩展 for comprehension,它不会编译的原因与您不能将 a 展平List为 an 的原因相同IO

readInts(10).flatMap(list => list.flatMap(i => printLine(i + 1).map(_ => ExitCode.Success)))
Run Code Online (Sandbox Code Playgroud)

相反,试试这个:

for {
  list <- readInts(10)
  _ <- list.traverse(i => printLine(i + 1))
} yield ExitCode.Success
Run Code Online (Sandbox Code Playgroud)

请注意,这与以下内容基本相同:

for {
  list <- readInts(10)
  _ <- list.map(i => printLine(i + 1)).sequence
} yield ExitCode.Success
Run Code Online (Sandbox Code Playgroud)

traverse只是将mapsequence步骤压缩为一。无论哪种情况,现在您的理解都被适当地限制为IO.