IO monad和记录语法的初学者错误

时间:2011-11-15 05:43:04

标签: haskell io monads

以下是我的代码的简化版本:

data Bookmark = Bookmark {
   url :: String
 , label :: String
 } deriving (Show)

genBookmark :: String -> String -> IO Bookmark
genBookmark u l = return ( Bookmark { url = u, label = l } )

但是Strings(url,label)在数据库中,所以我必须处理IO String。我必须有一个非常简单的解决方案,但我没有看到它(并且广泛的网络搜索没有让我到处找。)

基本上我想将我的代码更改为:

genBookmark :: IO String -> IO String -> IO Bookmark

这是另一个说明问题的版本:

genBookmark2 :: String -> String -> Bookmark
genBookmark2 u = return ( Bookmark { url = u, label = newlabel } )
                   where newlabel = getLine

错误“Expected type:String,Actual type:IO String”。

======编辑=======

以下是出现错误的“真实代码”(我从答案中得到的解决方案):

getSkosConceptRight :: String -> IO SkosConcept
getSkosConceptRight catName = do
                      suConcepts <- getSubConcepts catName
                      concept <- getMainConcept catName 
                      return ( concept { subConcepts = suConcepts })

getSkosConceptWrong :: String -> IO SkosConcept
getSkosConceptWrong catName = return ( concept { subConcepts = suConcepts })
                        where suConcepts = getSubConcepts catName
                              concept = getMainConcept catName 

getMainConcept :: ShortUrl -> IO SkosConcept
getSubConcepts :: ShortUrl -> IO [SkosConcept]

1 个答案:

答案 0 :(得分:7)

您遗漏了代码中从db获取字符串的部分。无论哪种方式,您都需要从String“解压缩”IO String

一般来说,你应该尽早摆脱IO。所以你应该尝试一种String -> String -> Bookmark

这是一个简单的例子:

-- These will represent your 'get from the db' calls
ioUrl :: IO String
ioUrl = return "http://..."

ioLabel :: IO String
ioLabel = return "my label"

genBookmark :: String -> String -> Bookmark
genBookmark url label = Bookmark { url = url, label = label }

现在,从main IO开始,我们进行“db”调用以获取字符串,解压缩值并发送到genBookmark函数。

main = do
    url <- ioUrl     -- here, we're removing the "http://..." from the IO String
    label <- ioLabel -- same here
    print (genBookmark url label)

genBookmark2失败的原因是因为您尝试在纯函数中使用getLine,其类型为IO String;因此expected String, got IO String错误消息。此外,该类型提到两个String作为输入,但您只需要一个。