您可以使用延续插件。在插件完成翻译工作后,它与来自 Oleg的Cont
monad 和 the shift
and有相似之处。reset
棘手的部分是找出类型。所以这是我的翻译:
import util.continuations._
import collection.mutable.ListBuffer
sealed trait Zipper[A] { def zipUp: Seq[A] }
case class ZDone[A](val zipUp: Seq[A]) extends Zipper[A]
case class Z[A](current: A, forward: Option[A] => Zipper[A]) extends Zipper[A] {
def zipUp = forward(None).zipUp
}
object Zipper {
def apply[A](seq: Seq[A]): Zipper[A] = reset[Zipper[A], Zipper[A]] {
val coll = ListBuffer[A]()
val iter = seq.iterator
while (iter.hasNext) {
val a = iter.next()
coll += shift { (k: A=>Zipper[A]) =>
Z(a, (a1: Option[A]) => k(a1.getOrElse(a)))
}
}
ZDone(coll.toList)
}
}
continuation 插件支持 for while
loop 但不支持map
or flatMap
,所以我选择使用while
和 mutableListBuffer
来捕获可能更新的元素。该make_zipper
函数被翻译成伴侣Zipper.apply
——一个典型的 Scala 位置,用于创建新对象或集合。数据类型被翻译成一个密封的特征,两个案例类对其进行了扩展。我已经将 zip_up 函数作为一种方法,Zipper
对每个案例类都有不同的实现。也很典型。
这是在行动:
object ZipperTest extends App {
val sample = for (v <- 1 to 5) yield (v, (1 to v).reduceLeft(_ * _))
println(sample) // Vector((1,1), (2,2), (3,6), (4,24), (5,120))
def extract134[A](seq: Seq[A]) = {
val Z(a1, k1) = Zipper(seq)
val Z(a2, k2) = k1(None)
val Z(a3, k3) = k2(None)
val Z(a4, k4) = k3(None)
List(a1, a3, a4)
}
println(extract134(sample)) // List((1,1), (3,6), (4,24))
val updated34 = {
val Z(a1, k1) = Zipper(sample)
val Z(a2, k2) = k1(None)
val Z(a3, k3) = k2(None)
val Z(a4, k4) = k3(Some(42 -> 42))
val z = k4(Some(88 -> 22))
z.zipUp
}
println(updated34) // List((1,1), (2,2), (42,42), (88,22), (5,120))
}
我是如何计算的类型shift
和k
/reset
或如何翻译的T.mapM
?
我看了看,mapM
我知道它可以让我得到一个Cont
,但我不确定里面是什么,Cont
因为它取决于班次。所以我开始轮班。return
忽略构造 a的 haskell Cont
, shift 返回 a Zipper
。我还想我需要在A
我的集合中添加一个类型的元素来构建。因此,转变将在A
预期类型元素的“洞”中,因此k
将是一个A=>?
函数。让我们假设这一点。我在我不太确定的类型后面加上问号。所以我开始:
shift { (k: A?=>?) =>
Z(a, ?)
}
接下来我(努力地)看着(k . maybe a id)
。该函数maybe a id
将返回一个A
,因此这与k
作为参数的内容一致。它相当于a1.getOrElse(a)
。另外因为我需要填写Z(a, ?)
,所以我需要弄清楚如何从选项到 Zipper 获取函数。最简单的方法是假设k
返回 a Zipper
。另外,看看如何使用 zipperk1(None)
或k1(Some(a))
,我知道我必须为用户提供一种方法来选择性地替换元素,这就是forward
函数的作用。它继续使用原始a
的或更新的a
. 它开始变得有意义。所以现在我有:
shift { (k: A=>Zipper[A]) =>
Z(a, (a1: Option[A]) => k(a1.getOrElse(a)))
}
接下来我mapM
再看一遍。我看到它是由return . ZDone
. 再次忽略return
(因为它仅适用于Cont
monad),我看到这ZDone
将获取结果集合。所以这是完美的,我只需要把coll
它放进去,当程序到达那里时,它就会有更新的元素。此外, 中的表达式类型reset
现在与返回类型k
一致Zipper[A]
。
最后我填写了编译器可以推断的重置类型,但是当我猜对时,它给了我一种(错误的)自信感,我知道发生了什么。
我的翻译不像 Haskell 那样通用或漂亮,因为它不保留集合中的类型并使用突变,但希望它更容易理解。
编辑:这是保留类型并使用不可变列表的版本,以便z.zipUp == z.zipUp
:
import util.continuations._
import collection.generic.CanBuild
import collection.SeqLike
sealed trait Zipper[A, Repr] { def zipUp: Repr }
case class ZDone[A, Repr](val zipUp: Repr) extends Zipper[A, Repr]
case class Z[A, Repr](current: A,
forward: Option[A] => Zipper[A, Repr]) extends Zipper[A, Repr] {
def zipUp = forward(None).zipUp
}
object Zipper {
def apply[A, Repr](seq: SeqLike[A, Repr])
(implicit cb: CanBuild[A, Repr]): Zipper[A, Repr] = {
type ZAR = Zipper[A, Repr]
def traverse[B](s: Seq[A])(f: A => B@cps[ZAR]): List[B]@cps[ZAR] =
if (s.isEmpty) List()
else f(s.head) :: traverse(s.tail)(f)
reset {
val list = traverse(seq.toSeq)(a => shift { (k: A=>ZAR) =>
Z(a, (a1: Option[A]) => k(a1.getOrElse(a)))
})
val builder = cb() ++= list
ZDone(builder.result): ZAR
}
}
}
顺便说一下,这里有关于 scala 中 continuation monad 的额外资源: