我在反应香蕉中的“总和事件”有什么问题?

时间:2012-01-11 21:33:15

标签: haskell reactive-banana

我正在学习反应性香蕉,并打算在服务器端应用程序中使用它。我在RxJs中有一些背景,所以我习惯将事件与不同的组合器结合起来。所以我从简单的事件组合器示例开始。我试图制作一个简单的反应香蕉示例,它将两个整数事件组合成一个sum事件。我理解为了能够组合来自不同事件的值,我必须首先将它们变为行为,然后进行组合并最终将其变为新事件。我就这样做了:

-- Behaviors from Events e1, e2
let b1 = stepper 0 e1 :: Behavior Int
let b2 = stepper 0 e2 :: Behavior Int

-- Sum Behavior
let sumB = (+) <$> b1 <*> b2
-- Back to Event
let sumE = sumB <@ (e1 `union` e2)

可以在Gist 1594917中找到完整的可运行示例。

这里的问题是,当其中一个事件(e1,e2)中出现新值时,sumE事件被正确触发,但它包含过时值。这显然是由于步进器的工作原理(行为的值在事件发生后稍微改变)。我尝试用Discrete替换行为,结果相同。

是否有一种简单的方法可以使这种事件组合器正常工作?

1 个答案:

答案 0 :(得分:6)

您的诊断完全正确。以下是两个选项:您可以通过changes从Discrete返回事件,也可以创建累积事件。

从Discrete回来可能更简单(我会推荐)。只是做

-- Discretes from Events e1, e2
let d1 = stepperD 0 e1 :: Discrete Int
let d2 = stepperD 0 e2 :: Discrete Int

-- Sum Discrete
let sumD = (+) <$> d1 <*> d2
-- Back to Event
let sumE = changes sumD

现在sumE将始终在e1e2更改时更新。

备选方案仅使用事件,通过将传入事件转换为累积函数。这听起来很复杂,但代码相当简单。

--convert each input into an accumulating tuple
let e1' = (\l (_,r) -> (l,r)) <$> e1
let e2' = (\r (l,_) -> (l,r)) <$> e2

let sumE = uncurry (+) <$> accumE (0,0) (e1' `union` e2')