我在 Haskell 中使用了一个名为Threepenny-GUI的图形库。在这个库中,主函数返回一个UI
monad 对象。这让我很头疼,因为当我尝试将IO
值解压缩到局部变量中时,我收到抱怨不同 monad 类型的错误。
这是我的问题的一个例子。这是标准 main 函数的略微修改版本,如 Threepenny-GUI 的代码示例所示:
main :: IO ()
main = startGUI defaultConfig setup
setup :: Window -> UI ()
setup w = do
labelsAndValues <- shuffle [1..10]
shuffle :: [Int] -> IO [Int]
shuffle [] = return []
shuffle xs = do randomPosition <- getStdRandom (randomR (0, length xs - 1))
let (left, (a:right)) = splitAt randomPosition xs
fmap (a:) (shuffle (left ++ right))
请注意第五行:
labelsAndValues <- shuffle [1..10]
返回以下错误:
Couldn't match type ‘IO’ with ‘UI’
Expected type: UI [Int]
Actual type: IO [Int]
In a stmt of a 'do' block: labelsAndValues <- shuffle [1 .. 10]
至于我的问题,我如何IO
使用标准箭头符号 ( <-
) 解包函数,并继续将这些变量作为IO ()
而不是UI ()
,以便我可以轻松地将它们传递给其他函数。
目前,我找到的唯一解决方案是使用liftIO
,但这会导致转换为UI
monad 类型,而我实际上想继续使用该IO
类型。