Should I avoid using Monad fail?

后端 未结 3 1086
-上瘾入骨i
-上瘾入骨i 2021-02-05 00:53

I\'m fairly new to Haskell and have been slowly getting the idea that there\'s something wrong with the existence of Monad fail. Real World Haskell warns against its use (\"Once

相关标签:
3条回答
  • 2021-02-05 01:37

    Some monads have a sensible failure mechanism, e.g. the terminal monad:

    data Fail x = Fail
    

    Some monads don't have a sensible failure mechanism (undefined is not sensible), e.g. the initial monad:

    data Return x = Return x
    

    In that sense, it's clearly a wart to require all monads to have a fail method. If you're writing programs that abstract over monads (Monad m) =>, it's not very healthy to make use of that generic m's fail method. That would result in a function you can instantiate with a monad where fail shouldn't really exist.

    I see fewer objections to using fail (especially indirectly, by matching Pat <- computation) when working in a specific monad for which a good fail behaviour has been clearly specified. Such programs would hopefully survive a return to the old discipline where nontrivial pattern matching created a demand for MonadZero instead of just Monad.

    One might argue that the better discipline is always to treat failure-cases explicitly. I object to this position on two counts: (1) that the point of monadic programming is to avoid such clutter, and (2) that the current notation for case analysis on the result of a monadic computation is so awful. The next release of SHE will support the notation (also found in other variants)

    case <- computation of
      Pat_1 -> computation_1
      ...
      Pat_n -> computation_n
    

    which might help a little.

    But this whole situation is a sorry mess. It's often helpful to characterize monads by the operations which they support. You can see fail, throw, etc as operations supported by some monads but not others. Haskell makes it quite clumsy and expensive to support small localized changes in the set of operations available, introducing new operations by explaining how to handle them in terms of the old ones. If we seriously want to do a neater job here, we need to rethink how catch works, to make it a translator between different local error-handling mechanisms. I often want to bracket a computation which can fail uninformatively (e.g. by pattern match failure) with a handler that adds more contextual information before passing on the error. I can't help feeling that it's sometimes more difficult to do that than it should be.

    So, this is a could-do-better issue, but at the very least, use fail only for specific monads which offer a sensible implementation, and handle the 'exceptions' properly.

    0 讨论(0)
  • 2021-02-05 01:52

    I try to avoid Monad fail wherever possible, and there are a whole variety of ways to capture fail depending on your circumstance. Edward Yang has written a good overview on his blog in the article titled 8 ways to report errors in Haskell revisited.

    In summary, the different ways to report errors that he identifies are:

    1. Use error
    2. Use Maybe a
    3. Use Either String a
    4. Use Monad and fail to generalize 1-3
    5. Use MonadError and a custom error type
    6. Use throw in the IO monad
    7. Use ioError and catch
    8. Go nuts with monad transformers
    9. Checked exceptions
    10. Failure

    Of these, I would be tempted to use option 3, with Either e b if I know how to handle the error, but need a little more context.

    0 讨论(0)
  • 2021-02-05 01:53

    In Haskell 1.4 (1997) there was no fail. Instead, there was a MonadZero type class which contained a zero method. Now, the do notation used zero to indicate pattern match failure; this caused surprises to people: whether their function needed Monad or MonadZero depended on how they used the do notation in it.

    When Haskell 98 was designed a bit later, they did several changes to make programming simpler to the novice. For example, monad comprehensions were turned into list comprehensions. Similarly, to remove the do type class issue, the MonadZero class was removed; for the use of do, the method fail was added to Monad; and for other uses of zero, a mzero method was added to MonadPlus.

    There is, I think, a good argument to be made that fail should not be used for anything explicitly; its only intended use is in the translation of the do notation. Nevertheless, I myself am often naughty and use fail explicitly, too.

    You can access the original 1.4 and 98 reports here. I'm sure the discussion leading to the change can be found in some email list archives, but I don't have links handy.

    0 讨论(0)
提交回复
热议问题