了解嵌套Monad中的Monadic绑定

时间:2014-08-22 17:26:02

标签: haskell compiler-errors monads typeclass haskell-platform

我正在尝试使用monads(复数)解析程序参数。我想构建一个IO (Either String Parameters)Left String表示描述无效参数的错误消息。 Right Parameters代表doRealWork所需的有效计划参数。

这是程序的结构:

import System.Environment
import System.IO.Error
data Parameters = Parameters String String [Int]  

main :: IO ()
main = getArgs
   >>= processArgs
   >>= either putStrLn doRealWork

processArgs :: [String] -> IO (Either String Parameters)
processArgs args = (return $ enumerateArgs args)
               >>= (either (return . Left) parseArgs)
                -- This maybe could be improved, but it's not the focus

doRealWork     :: Parameters -> IO ()
doRealWork     = undefined -- I'll implement the real work part later

enumerateArgs  :: [String] -> Either String (String,String,String)
enumerateArgs list
  | length list == 3 = Right (a,b,c) 
  | otherwise        = Left $ "Incorrect Argument Count,\n"
                          ++  "Expected 3 parameters\n"
                          ++  "Received: " ++ show list
  where (a:b:c:[]) = list

readFileEither :: String -> IO (Either String String)
readFileEither = undefined -- it actually works, implementation is irrelevant

parseArgs'     :: String -> String -> String -> Either String Parameters
parseArgs'     = undefined -- it actually works, implementation is irrelevant

parseArgs :: (String,String,String) -> IO (Either String Parameters)
parseArgs (a,b,c) = readFileEither c >>= (\x -> return . (x >>= (parseArgs' a b)))
                          -- IO bind ^^^        Either bind ^^^

正如你在parseArgs中看到的那样,我想将readFileEither的结果绑定到一个继续解析参数的lambda。档案数据。 readFileEither结果中的值为Either String String。由于parseArgs'的结果是Either String Parameters,我想使用Either的monadic绑定将lambda的输入绑定到parseArgs',所有这些都在IO monadic绑定中readFileEither的结果和lambda。

在我的头脑中,是有意义的,但编译器不同意。

Couldn't match expected type `IO (Either String Parameters)'
            with actual type `a0 -> c0'
In the expression: return . (x >>= (parseArgs' a b))
In the second argument of `(>>=)', namely
  `(\ x -> return . (x >>= (parseArgs' a b)))'
In the expression:
  readFileEither c >>= (\ x -> return . (x >>= (parseArgs' a b)))

作为参考,这是monadic Either的工作原理:

instance Monad (Either e) where
    return = Right
    Left  l >>= _ = Left l
    Right r >>= k = k r

我错过了什么?为什么嵌套的monadic绑定无法进行类型检查?

1 个答案:

答案 0 :(得分:4)

这里的问题很简单,你已经使用了函数组合运算符.,但是没有函数可以编写! x >>= (parseArgs' a b)是一个非常精细的Either值,而不是产生一个值的函数。结果应该是IO值,而不是Kleisli函数来产生这样的值。你需要简单地写return (x >>= (parseArgs' a b)) - 或者更好,

parseArgs (a,b,c) = readFileEither c >>= \x -> return $ x >>= parseArgs' a b