MonadPlus IO不是一个半身像

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在用户意图出错时违反法律,那么它的实际意图是什么?

dfe*_*uer 7

IOunder 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 将该异常抛出)。因此,这就是混乱。