我正在尝试转换以下语法生成
callExpr:
primaryExpr
| callExpr primaryExpr
到Haskell中的Parsec表达式。
很明显,问题是它是递归的,所以我试图用递归上升的方式来解析它。我试图实现的伪代码是:
e = primaryExp
while(true) {
e2 = primaryExp
if(e2 failed) break;
e = CallExpr(e, e2)
}
我尝试将其翻译成Haskell是:
callExpr :: IParser Expr
callExpr = do
e <- primaryExpr
return $ callExpr' e
where
callExpr' e = do
e2m <- optionMaybe primaryExpr
e' <- maybe e (e2 -> callExpr' (CallExpr e e2)) e2m
return e'
其中primaryExpr
具有类型IParser Expr
IParser被定义为
type IParser a = ParsecT String () (State SourcePos) a
然而,这给了我以下类型的错误:
Couldn't match type `ParsecT String () (State SourcePos) t0'
with `Expr'
Expected type: ParsecT String () (State SourcePos) Expr
Actual type: ParsecT
String
()
(State SourcePos)
(ParsecT String () (State SourcePos) t0)
In a stmt of a 'do' block: return $ callExpr' e
In the expression:
do { e <- primaryExpr;
return $ callExpr' e }
In an equation for `callExpr':
callExpr
= do { e <- primaryExpr;
return $ callExpr' e }
where
callExpr' e
= do { e2m <- optionMaybe primaryExpr;
.... }
如何修复此类型错误?
使用chainl1
。chainl1 p op
以左关联方式解析一个或多个由op
-s分隔的p
-s。op
返回一个二进制函数,用于将两侧的p
-的结果组合为一个结果。
由于您的语法似乎没有分隔符,因此可以将chainl1
与只返回组合函数的op
一起使用:
callExpr :: IParser Expr
callExpr = chainl1 primaryExpr (return CallExpr)
关于您的callExpr
实现,我可以发现两个错误。
首先,您使用return $ callExpr' e
,但callExpr' e
已经是一个一元值,所以只有callExpr' e
是正确的。
其次,在maybe e (e2 -> callExpr' (CallExpr e e2)) e2m
中,默认的e
应该是一元的(否则我们如何将其绑定到e'
?),所以它应该是return e
。