我是 Haskell 的新手,但了解如何使用 Monad Transformers。然而,我仍然难以抓住他们声称的将参数传递给函数调用的优势。
基于wiki Monad Transformers Explained,我们基本上有一个配置对象定义为
data Config = Config Foo Bar Baz
并传递它,而不是用这个签名编写函数
client_func :: Config -> IO ()
我们使用 ReaderT Monad Transformer 并将签名更改为
client_func :: ReaderT Config IO ()
拉动 Config 只是对ask
.
函数调用从client_func c
变为runReaderT client_func c
美好的。
但是为什么这会使我的应用程序更简单?
1-当您将许多功能/模块拼接在一起以形成应用程序时,我怀疑 Monad Transformers 有兴趣。但这就是我的理解停止的地方。有人可以阐明一下吗?
2-我找不到任何关于如何在 Haskell 中编写大型模块化应用程序的文档,其中模块公开某种形式的 API 并隐藏它们的实现,以及(部分)隐藏它们自己的状态和环境对其他模块。请问有什么指点吗?
(编辑:真实世界的 Haskell 声明“.. 这种方法 [Monad Transformers] ......适用于更大的程序。”,但没有明确的例子证明这种说法)
编辑以下克里斯泰勒的回答
Chris 完美地解释了为什么在 Transformer Monad 中封装 Config、State 等会带来两个好处:
- 它防止更高级别的函数必须在其类型签名中维护它调用的(子)函数所需的所有参数,但它自己使用不需要(参见
getUserInput
函数) - 因此,使更高级别的函数对 Transformer Monad 内容的更改更具弹性(例如,您想在其中添加一个
Writer
以在较低级别的函数中提供 Logging)
这是以更改所有函数的签名为代价的,以便它们“在”Transformer Monad 中运行。
因此,问题 1 已完全涵盖。谢谢克里斯。
问题 2 现在已在此 SO 帖子中得到解答