Dan*_*Dos 4 io error-handling haskell throw monoids
实例MonadPlus IO
是唯一的,因为mzero
抛出:
Prelude Control.Monad> mzero
*** Exception: user error (mzero)
Run Code Online (Sandbox Code Playgroud)
因此,相应地,这也MonadPlus IO
意味着它也适用于错误。
mzero
如果其他动作不抛出,则显然用作标识元素:
Prelude Control.Monad> mzero `mplus` return 0
0
Prelude Control.Monad> return 0 `mplus` mzero
0
Run Code Online (Sandbox Code Playgroud)
但是当两个动作都抛出时不是这样:
Prelude Control.Monad> fail "Hello, world!" `mplus` mzero
*** Exception: user error (mzero)
Prelude Control.Monad> mzero `mplus` fail "Hello, world!"
*** Exception: user error (Hello, world!)
Run Code Online (Sandbox Code Playgroud)
所以MonadPlus IO
不是monoid。
如果MonadPlus
在用户意图出错时违反法律,那么它的实际意图是什么?
IO
under mplus
是一个与等价类相对的Monoid,用于标识异常。没那么令人满意。一种替代方法可能如下所示:
m <|> n = m `catches`
[ Handler $ \ ~EmptyIO -> n
, Handler $ \ ~se@(SomeException _) ->
n `catch` \ ~EmptyIO -> throwIO se ]
Run Code Online (Sandbox Code Playgroud)
这种方法的主要问题是处理程序可以堆叠。当第一个动作失败时,我们不能仅仅执行第二个动作。一个较小的问题是,没有一种完全可靠的方法来确定异常是同步的(应该使用throwIO
抛出)还是异步的(在这种情况下,我们需要使用throwTo
自己的线程ID 将该异常抛出)。因此,这就是混乱。
归档时间: |
|
查看次数: |
88 次 |
最近记录: |