A1

附錄 I

這裡是 Lox 的完整文法。介紹語言各部分的章節中包含了該部分的文法規則,但這裡將它們全部收集到一處。

A1.1語法文法

語法文法用於將線性的標記序列解析為巢狀的語法樹結構。它從匹配整個 Lox 程式(或單個 REPL 輸入)的第一條規則開始。

programdeclaration* EOF ;

A1.1.1宣告

程式是一系列的宣告,這些宣告是綁定新識別符號的語句或任何其他語句類型。

declarationclassDecl
               | funDecl
               | varDecl
               | statement ;

classDecl"class" IDENTIFIER ( "<" IDENTIFIER )?
                 "{" function* "}" ;
funDecl"fun" function ;
varDecl"var" IDENTIFIER ( "=" expression )? ";" ;

A1.1.2語句

剩餘的語句規則會產生副作用,但不會引入綁定。

statementexprStmt
               | forStmt
               | ifStmt
               | printStmt
               | returnStmt
               | whileStmt
               | block ;

exprStmtexpression ";" ;
forStmt"for" "(" ( varDecl | exprStmt | ";" )
                           expression? ";"
                           expression? ")" statement ;
ifStmt"if" "(" expression ")" statement
                 ( "else" statement )? ;
printStmt"print" expression ";" ;
returnStmt"return" expression? ";" ;
whileStmt"while" "(" expression ")" statement ;
block"{" declaration* "}" ;

請注意,block 是一個語句規則,但也用作其他一些規則(例如函數體)中的非終端符號。

A1.1.3表達式

表達式會產生值。Lox 有許多具有不同優先級的一元和二元運算符。一些語言的文法並未直接編碼優先級關係,而是在其他地方指定。在這裡,我們為每個優先級使用單獨的規則使其明確。

expressionassignment ;

assignment     → ( call "." )? IDENTIFIER "=" assignment
               | logic_or ;

logic_orlogic_and ( "or" logic_and )* ;
logic_andequality ( "and" equality )* ;
equalitycomparison ( ( "!=" | "==" ) comparison )* ;
comparisonterm ( ( ">" | ">=" | "<" | "<=" ) term )* ;
termfactor ( ( "-" | "+" ) factor )* ;
factorunary ( ( "/" | "*" ) unary )* ;

unary          → ( "!" | "-" ) unary | call ;
callprimary ( "(" arguments? ")" | "." IDENTIFIER )* ;
primary"true" | "false" | "nil" | "this"
               | NUMBER | STRING | IDENTIFIER | "(" expression ")"
               | "super" "." IDENTIFIER ;

A1.1.4輔助規則

為了使上述規則更簡潔一些,一些文法被拆分成了幾個可重複使用的輔助規則。

functionIDENTIFIER "(" parameters? ")" block ;
parametersIDENTIFIER ( "," IDENTIFIER )* ;
argumentsexpression ( "," expression )* ;

A1.2詞法文法

詞法文法被掃描器用於將字元分組為標記。語法是上下文無關的,而詞法文法是正規的請注意,這裡沒有遞迴規則。

NUMBERDIGIT+ ( "." DIGIT+ )? ;
STRING"\"" <any char except "\"">* "\"" ;
IDENTIFIERALPHA ( ALPHA | DIGIT )* ;
ALPHA"a" ... "z" | "A" ... "Z" | "_" ;
DIGIT"0" ... "9" ;