gpt4 book ai didi

scala - 如何让 Scalaz ZIO 变得懒惰?

转载 作者:行者123 更新时间:2023-12-01 09:43:40 24 4
gpt4 key购买 nike

我有一个严重的副作用函数(想想数据库调用),我想将其用作惰性值,这样它只会在第一次使用时被调用(如果从未使用过则根本不会被调用)。

我如何使用 ZIO 做到这一点?

如果我的程序看起来像这样,函数只被调用一次(但即使结果根本没有被使用):

import scalaz.zio.IO
import scalaz.zio.console._

object Main extends scalaz.zio.App {

def longRunningDbAction: IO[Nothing, Integer] = for {
_ <- putStrLn("Calling the database now")
} yield 42

def maybeUseTheValue(x: Integer): IO[Nothing, Unit] = for {
_ <- putStrLn(s"The database said ${x}")
} yield ()

def maybeNeedItAgain(x: Integer): IO[Nothing, Unit] = for {
_ <- putStrLn("Okay, we did not need it again here.")
} yield ()

override def run(args: List[String]): IO[Nothing, Main.ExitStatus] = for {
valueFromDb <- longRunningDbAction
_ <- maybeUseTheValue(valueFromDb)
_ <- maybeNeedItAgain(valueFromDb)
} yield ExitStatus.ExitNow(0)

}

我想我必须传递一个生成 IntIO 而不是已经具体化的 Int,但是如果我传递原始的刚刚调用数据库的IO,会被反复调用:

object Main extends scalaz.zio.App {

def longRunningDbAction: IO[Nothing, Integer] = for {
_ <- putStrLn("Calling the database now")
} yield 42


def maybeUseTheValue(x: IO[Nothing, Integer]): IO[Nothing, Unit] = for {
gettingItNow <- x
_ <- putStrLn(s"The database said ${gettingItNow}")
} yield ()

def maybeNeedItAgain(x: IO[Nothing, Integer]): IO[Nothing, Unit] = for {
gettingItNow <- x
_ <- putStrLn(s"Okay, we need it again here: ${gettingItNow}")
} yield ()

override def run(args: List[String]): IO[Nothing, Main.ExitStatus] = for {
_ <- maybeUseTheValue(longRunningDbAction)
_ <- maybeNeedItAgain(longRunningDbAction)
} yield ExitStatus.ExitNow(0)

}

有没有办法将 longRunningDbAction“包装”成使其变得懒惰的东西?

最佳答案

我想到了以下内容:

 def lazyIO[E,A](io: IO[E,A]): IO[Nothing, IO[E, A]] = {
for {
barrier <- Promise.make[Nothing, Unit]
fiber <- (barrier.get *> io).fork
} yield barrier.complete(()) *> putStrLn("getting it") *> fiber.join
}

ZIO 1.0-RC4 的更新版本(带有环境支持):

def lazyIO[R, E, A](io: ZIO[R, E, A]): ZIO[R, Nothing, ZIO[R, E, A]] = {
for {
barrier <- Promise.make[Nothing, Unit]
fiber <- (barrier.await *> io).fork
} yield barrier.succeed(()) *> fiber.join
}

所以这是一个接受 IO 并返回它的惰性版本的 IO。

它通过启动一个 fiber 来工作运行原始 io , 但只有在 Promise ( barrier) 完成之后。

惰性IO首先完成barrier (如果它是第一个执行此操作的人,它将解锁 fiber,后者又运行包装的 io)然后加入 fiber检索计算结果。

有了这个,我可以做到

override def run(args: List[String]): IO[Nothing, Main.ExitStatus] = for {
valueFromDb <- lazyIO(longRunningDbAction)
_ <- maybeUseTheValue(valueFromDb)
_ <- maybeNeedItAgain(valueFromDb)
} yield ExitStatus.ExitNow(0)

并且控制台输出显示延迟值确实被拉取了两次,但只有第一个触发了“数据库访问”:

getting it
Calling the database now
The database said 42
getting it
Okay, we need it again here: 42

关于scala - 如何让 Scalaz ZIO 变得懒惰?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53851349/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com