- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在探索 Haskell 中的选项,这些选项可以让我将业务逻辑与底层系统的技术实现分开。例如,在 Web 服务器的上下文中,将 Web 服务器处理其接收的信息的方式与其读取和写入数据库的方式分开。要做到这一点,有很多选择,但有两个特别引起了我的注意:释放 Monad 和传递功能记录作为参数。我很难看出其中一种相对于另一种的优缺点。
用于说明我正在讨论的内容的代码片段:
module Lib where
import qualified Control.Monad.Free as FreeMonad
data MyGadt x
= Read (String -> x)
| Write String
x
instance Functor MyGadt where
fmap f (Read g) = Read (f . g)
fmap f (Write str x) = Write str (f x)
programWithFreeMonad :: FreeMonad.Free MyGadt ()
programWithFreeMonad = do
msg <- FreeMonad.liftF $ Read id
FreeMonad.liftF $ Write msg ()
ioInterpreter :: FreeMonad.Free MyGadt x -> IO x
ioInterpreter (FreeMonad.Pure x) = return x
ioInterpreter (FreeMonad.Free (Read f)) = getLine >>= (ioInterpreter . f)
ioInterpreter (FreeMonad.Free (Write str x)) = putStrLn str >> ioInterpreter x
runProgramWithFreeMonad :: IO ()
runProgramWithFreeMonad = ioInterpreter programWithFreeMonad
data Capabilities m = Capabilities
{ myRead :: m String
, myWrite :: String -> m ()
}
programWithCapabilities :: Monad m => Capabilities m -> m ()
programWithCapabilities capabilities = do
msg <- myRead capabilities
myWrite capabilities msg
runProgramWithCapabilities :: IO ()
runProgramWithCapabilities =
programWithCapabilities $ Capabilities {myRead = getLine, myWrite = putStrLn}
这两种解决方案的编写方式不同,因此我认为许多人对其外观以及他们更喜欢哪一种都有自己的看法。但我想知道是否有人对一种解决方案相对于另一种解决方案的优缺点有任何见解。
最佳答案
即使我们限制自己在自由 monad 和函数记录之间进行选择(不考虑涉及 monad 转换器堆栈和类似 MTL 的类型类的解决方案),仍有很多争论正在进行,而且问题还没有解决。
传统上,简单的 free monad 被指责有两个缺陷:运行时效率低下(这可能重要也可能不重要,这取决于解释器操作的速度相比而言有多慢)和缺乏可扩展性(如何将一个程序提升到另一个程序中)有更丰富的效果吗?)。
"Data types a la carte"首先尝试解决可扩展性问题。后来,"Freer Monads, More Extensible Effects"论文发表,提出了一种更复杂的自由类型来提高单子(monad)绑定(bind)的效率,同时也是定义操作集的可扩展方式。实现这种方法的主要库是:
freer-simple这群人中最容易理解的,显然也是最慢的。好像对括号类型的操作有一些限制。
fused-effects更高效的库,允许括号类型操作。但类型也比较复杂。
polysemy相对较新的库,旨在快速并支持括号类型操作,同时保留简单类型。
这些库的一个吸引人的方面是它们可以让您零碎地解释效果,挑选出一种效果,同时不解释其余的效果。您还可以将抽象效果解释为其他抽象效果,而无需立即转到 IO
。
至于功能记录方法。像 programWithCapability
这样的程序在基本 monad 上是多态的,并且记录由 monad 参数化的函数,在概念上与所谓的 van Laarhoven Free Monad 相关。 :
-- (ops m) is required to be isomorphic to (Π n. i_n -> m j_n)
newtype VLMonad ops a = VLMonad { runVLMonad :: forall m. Monad m => ops m -> m a }
instance Monad (VLMonad ops) where
return a = VLMonad (\_ -> return a)
m >>= f = VLMonad (\ops -> runVLMonad m ops >>= f' ops)
where
f' ops a = runVLMonad (f a) ops
来自链接的帖子:
Swierstra notes that by summing together functors representing primitive I/O actions and taking the free monad of that sum, we can produce values use multiple I/O feature sets. Values defined on a subset of features can be lifted into the free monad generated by the sum. The equivalent process can be performed with the van Laarhoven free monad by taking the product of records of the primitive operations. Values defined on a subset of features can be lifted by composing the van Laarhoven free monad with suitable projection functions that pick out the requisite primitive operations.
似乎不存在(?)为您提供预制 VLMonad
类型的库。确实存在的是记录函数但通过 IO 工作的库,例如 RIO
。人们仍然可以对逻辑中的基本 monad 进行抽象,然后在运行逻辑时使用 RIO。或者更喜欢简单性并撕开将 IO
隐藏在逻辑中的多态面纱。
函数记录方法可能具有更容易掌握的优点,是直接在 IO
上工作的增量升级。它也更类似于进行依赖注入(inject)的面向对象方式。
使用唱片本身的人体工程学变得至关重要。目前常用"classy lenses"使程序逻辑独立于具体的记录类型,方便程序的编写。也许有一天也可以使用可扩展记录(就像在更自由的方法中使用可扩展总和类型一样)。
关于haskell - 自由 Monad 与显式传递函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56092971/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!