我正在尝试使用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绑定无法进行类型检查?
答案 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