gpt4 book ai didi

haskell - 管道广播

转载 作者:行者123 更新时间:2023-12-04 03:52:15 26 4
gpt4 key购买 nike

几天前的一个观点,我问了这个question .现在我需要这个函数的纯单线程版本:

重复一遍,我需要一个函数将每个接收到的值发送到每个接收器并收集它们的结果。函数的类型签名应该是这样的:

broadcast :: [Sink a m b] -> Sink a m [b]

最佳斯文

附言不是 sequence ,我试过了:
> C.sourceList [1..100] $$ sequence [C.fold (+) 0, C.fold (+) 0]
[5050, 0]

预期结果:
[5050, 5050]

附言 zipSinks给出了想要的结果,但它只适用于元组:
> C.sourceList [1..100] $$ C.zipSinks (C.fold (+) 0) (C.fold (+) 0)
(5050, 5050)

最佳答案

基本上我们需要做的就是重新实现sequence , 但使用 zipSinks而不是原来的排序操作:

import Data.Conduit as C
import Data.Conduit.List as C
import Data.Conduit.Util as C

fromPairs
:: (Functor f)
=> f [a] -- ^ an empty list to start with
-> (f a -> f [a] -> f (a, [a])) -- ^ a combining function
-> [f a] -- ^ input list
-> f [a] -- ^ combined list
fromPairs empty comb = g
where
g [] = empty
g (x:xs) = uncurry (:) `fmap` (x `comb` g xs)

现在创建 broadcast正在申请 fromPairszipSinks :
broadcast :: (Monad m) => [Sink a m b] -> Sink a m [b]
broadcast = fromPairs (return []) zipSinks

我们可以做类似的事情
main = C.sourceList [1..100] $$ broadcast [C.fold (+) 0, C.fold (*) 1]

更新:我们可以看到 fromPairs看起来只是 sequenceA所以我们可以进一步插入这个想法。让我们在管道上定义一个 zipping applicative functor,类似于 ZipList :
import Control.Applicative
import Control.Monad
import Data.Conduit
import Data.Conduit.Util
import Data.Traversable (Traversable(..), sequenceA)

newtype ZipSink i m r = ZipSink { getZipSink :: Sink i m r }

instance Monad m => Functor (ZipSink i m) where
fmap f (ZipSink x) = ZipSink (liftM f x)
instance Monad m => Applicative (ZipSink i m) where
pure = ZipSink . return
(ZipSink f) <*> (ZipSink x) =
ZipSink $ liftM (uncurry ($)) $ zipSinks f x

然后 broadcast变得如此简单
broadcast :: (Traversable f, Monad m) => f (Sink i m r) -> Sink i m (f r)
broadcast = getZipSink . sequenceA . fmap ZipSink

关于haskell - 管道广播,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18252064/

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