私は式のパーサーを構築しています。右から左へ構文解析
expr ::= term (+ expr | - expr | null)
term ::= expo (* term |/term | null)
expo ::= factor (^ expo | null)
factor ::= (expr) | int
と対応するコード:
$ eval "3*1/3"
:expr :: Parser Int expr = do t <- term do _ <- symbol "+" e <- expr return (t + e) <|> do _ <- symbol "-" e <- expr return (t - e) <|> return t term :: Parser Int term = do ep <- expo do _ <- symbol "*" t <- term return (ep * t) <|> do _ <- symbol "/" t <- term return (ep `div` t) <|> return ep expo :: Parser Int expo = do f <- factor do _ <- symbol "^" e <- expo return (f^e) <|> return f factor :: Parser Int factor = do _ <- symbol "(" e <- expr _ <- symbol ")" return e <|> integer
それは確かに失敗するほとんどの場合に適していますが、
は、ここに私の文法規則です
3 * 1/3
が4 - 3 - 2
ので3 * (1/3)
(*)
/\
3 (/)
/\
1 3
と
$ eval "4-3-2"
に解析されるためには4 - (3 - 2)
(-)
/\
4 (-)
/\
3 2
私はそれがすべての方向性(正しい連想性)について理解しています。私は何を期待
は私がright-left
を解析し、left-right
それを解釈する(または再帰的にそれを解析)う意味(4 - 3) - 2
(-)
/\
(-) 2
/\
4 3
です。
どのように達成するかわかりません。今のところ私の心にはfoldl
以外何も来ません。
誰かがそれを修正するために何をすべきか提案できますか?
総プログラム:
{-# OPTIONS_GHC -Wall #-}
import Control.Applicative
import Data.Char
newtype Parser a = P (String -> [(a, String)])
parse :: Parser a -> String -> [(a, String)]
parse (P p) inp = p inp
instance Functor Parser where
fmap g p = P (\inp -> case parse p inp of
[] -> []
[(v, out)] -> [(g v, out)]
_ -> undefined)
instance Applicative Parser where
pure v = P (\inp -> [(v, inp)])
pg <*> px = P (\inp -> case parse pg inp of
[] -> []
[(g, out)] -> parse (fmap g px) out
_ -> undefined)
instance Monad Parser where
p >>= f = P (\inp -> case parse p inp of
[] -> []
[(v, out)] -> parse (f v) out
_ -> undefined)
instance Alternative Parser where
empty = P (\_ -> [])
p <|> q = P (\inp -> case parse p inp of
[] -> parse q inp
[(v, out)] -> [(v, out)]
_ -> undefined)
many x = some x <|> pure []
some x = pure (:) <*> x <*> many x
item :: Parser Char
item = P (\inp -> case inp of
[] -> []
(x : xs) -> [(x, xs)])
sat :: (Char -> Bool) -> Parser Char
sat p = do x <- item
if p x
then return x
else empty
digit :: Parser Char
digit = sat isDigit
char :: Char -> Parser Char
char x = sat (== x)
string :: String -> Parser String
string [] = return []
string (x : xs) = do _ <- char x
_ <- string xs
return (x : xs)
space :: Parser()
space = do _ <- many (sat isSpace)
return()
nat :: Parser Int
nat = do xs <- some digit
return (read xs)
int :: Parser Int
int = do _ <- char '-'
n <- nat
return (-n)
<|> nat
token :: Parser a -> Parser a
token p = do _ <- space
v <- p
_ <- space
return v
integer :: Parser Int
integer = token int
symbol :: String -> Parser String
symbol = token . string
expr :: Parser Int
expr = do t <- term
do _ <- symbol "+"
e <- expr
return (t + e)
<|> do _ <- symbol "-"
e <- expr
return (t - e)
<|> return t
term :: Parser Int
term = do ep <- expo
do _ <- symbol "*"
t <- term
return (ep * t)
<|> do _ <- symbol "/"
t <- term
return (ep `div` t)
<|> return ep
expo :: Parser Int
expo = do f <- factor
do _ <- symbol "^"
e <- expo
return (f^e)
<|> return f
factor :: Parser Int
factor = do _ <- symbol "("
e <- expr
_ <- symbol ")"
return e
<|> integer
eval :: String -> Int
eval xs = case (parse expr xs) of
[(n, [])] -> n
[(_, out)] -> error ("Unused input " ++ out)
[] -> error "Invalid input"
_ -> undefined
['Text.Parsec.Expr'](https://hackage.haskell.org/package/parsec-3.1.11/docs/Text-Parsec-Expr.html)をご覧ください –