monad 变压器和多个 monad 的堆叠

发布于 2024-08-18 19:35:23 字数 598 浏览 11 评论 0原文

我有带有签名 f :: [a] -> 的函数 f StateT Int Reader b [c] 和带有签名 f' :: a -> 的 f' StateT Int Reader b [c]

f 中的计算(非常简化)如下所示:

f [] = return []
f (s:st) = f' s >>= \x ->
           f st >>= \y ->
           return $ ...

并且代替 ... 我想返回 < 的 [c] 部分code>x ++ y[c] 部分,其中包含 monad 内容。
是否有可能在不手动解开 xy 并再次手动将结果放在一起的情况下实现这一目标?我是否需要在 monad 堆栈底部添加一个 List monad 才能获得简单的代码? Reader Monad 显然不是 MonadPlus 类的实例。

I have function f with signature f :: [a] -> StateT Int Reader b [c], and f' with signature f' :: a -> StateT Int Reader b [c]

The computation in f (very simplified) looks like that:

f [] = return []
f (s:st) = f' s >>= \x ->
           f st >>= \y ->
           return $ ...

And in place of the ... I would like to return the [c] part of x ++ the [c] part of y with the monad stuff wrapped around.
Is there a possibility to achieve that without manually unwrapping x and y and manually put the result together again? Do I need a List monad at the bottom of my monad stack to get simple code? The Reader Monad is obviously not an instance of the MonadPlus class.

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(3

浪菊怪哟 2024-08-25 19:35:23

我不明白你打开 xy 的意思。

我会将最后一行作为

return (x ++ y)

我是否误解了你想要的?

I don't get what you mean by unwrapping x and y.

I would have the last line as

return (x ++ y)

Do I misunderstand what you want?

浊酒尽余欢 2024-08-25 19:35:23

您还可以简单地定义

f = fmap concat . mapM f'

(mapM f' xs 生成一个 m [[c]] 类型的值,其中 xs :: [a]和 m = StateT Int (Reader b),然后 fmap concat 连接列表“在 monad 内部”。)

You can also simply define

f = fmap concat . mapM f'

(mapM f' xs produces a value of type m [[c]], where xs :: [a] and m = StateT Int (Reader b), and then fmap concat concatenates the lists "inside the monad".)

护你周全 2024-08-25 19:35:23

f' sf st 都是一个 monad 中的值,即 StateT Int Reader b。因此,您已经有了 x :: [c]y :: [c],只需编写 return (x ++ y) >,正如戴夫·辛顿所说。

Both f' s and f st are values in one monad, namely StateT Int Reader b. So you already have x :: [c] and y :: [c] and you just need to write return (x ++ y), as Dave Hinton said.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文