3

我有以下代码:

parseExpr :: [String] -> (Ast,[String])
parseExpr [] = error "Incorrect"
parseExpr (s:ss) | all isDigit s = (Tall (read s),ss)
              | s == "-" = let (e,ss') = parseExpr ss in (Min e,ss')
              | s == "+" = (Sum e e',ss'') where
                           (e,ss') = parseExpr ss
                           (e',ss'') = parseExpr ss'
              | s == "*" = (Mult e e',ss'') where
                           (e,ss') = parseExpr ss
                           (e',ss'') = parseExpr ss'

当我尝试运行它时,我收到一条错误消息“输入'|'上的解析错误。它正在谈论的行是这一行:

...
                           (e',ss'') = parseExpr ss'
->            | s == "*" = (Mult e e',ss'') where
                           (e,ss') = parseExpr ss
...

我想我明白为什么了。我想这是因为我上面有两行没有被守卫屏蔽,当下面突然出现另一个守卫时,haskell 会感到困惑。但是在这些情况之后,我怎样才能添加额外的警卫呢?

我尝试使用分号分隔新行,但我对 Haskell 很陌生,因此对它的语法了解不多。

4

1 回答 1

2

没有必要使用大括号(当然你可以)。但是你确实需要小心缩进。它并不像起初看起来那么难。

话虽如此,您确实不能where在受保护的表达式中使用。您必须let在这种情况下使用,如下所示:

parseExpr :: [String] -> (Ast,[String])
parseExpr [] = error "Incorrect"
parseExpr (s:ss) | all isDigit s = (Tall (read s),ss)
              | s == "-" = let (e,ss') = parseExpr ss in (Min e,ss')
              | s == "+" = let (e,ss') = parseExpr ss
                               (e',ss'') = parseExpr ss'
                           in (Sum e e',ss'')
              | s == "*" = let (e,ss') = parseExpr ss
                               (e',ss'') = parseExpr ss'
                           in (Mult e e',ss'')

但是,在您的情况下,保护表达式非常简单,可以用模式匹配替换。(并且建议利用这个机会。模式匹配是您在 Haskell 中最好的朋友。)在这种情况下,您使用where.

parseExpr :: [String] -> (Ast,[String])
parseExpr [] = error "Incorrect"
parseExpr (s:ss) | all isDigit s = (Tall (read s),ss)
parseExpr ("-":ss) = (Min e,ss') where (e,ss') = parseExpr ss
parseExpr ("+":ss) = (Sum  e e',ss'') where (e, ss' ) = parseExpr ss
                                            (e',ss'') = parseExpr ss'

parseExpr ("*":ss) = (Mult e e',ss'') where (e, ss' ) = parseExpr ss
                                            (e',ss'') = parseExpr ss'
于 2019-10-10T17:09:19.930 回答