我必须首先说我是语法新手,而且我还在学习antlr。
我的语法允许以下操作:
grammar TEST;
file : (varDecl | functionDcl)+ ;
varDecl : type ID ('=' expression)? ';' ;
type : 'int' | 'float' | 'void' ;
functionDcl : type ID '(' formalParameters? ')' block ;
formalParameters : formalParameter (',' formalParameter)* ;
formalParameter : type ID ;
block : '{' stat* '}' ;
stat : block
| varDecl
| 'if' expression 'then' stat ('else' stat)?
| 'return' expression? ';'
| expression '=' expression ';'
| expression ';'
;
expression : unaryExprNotPlusMinus (intervalOp unaryExprNotPlusMinus)? ;
unaryExprNotPlusMinus : unaryOp expression
| INT
| FloatingPointLiteral
;
unaryOp : '~' | '!' | 'not' | 'typeof' | 'statictypeof';
intervalOp : '..' | '|..' | '..|' | '|..|' ;
INT : JavaIDDigit+ ;
ID : Letter (Letter|JavaIDDigit)* ;
fragment
Letter
: '\u0024' |
'\u0041'..'\u005a' |
'\u005f' |
'\u0061'..'\u007a' |
'\u00c0'..'\u00d6' |
'\u00d8'..'\u00f6' |
'\u00f8'..'\u00ff' |
'\u0100'..'\u1fff' |
'\u3040'..'\u318f' |
'\u3300'..'\u337f' |
'\u3400'..'\u3d2d' |
'\u4e00'..'\u9fff' |
'\uf900'..'\ufaff'
;
fragment
JavaIDDigit
: '\u0030'..'\u0039' |
'\u0660'..'\u0669' |
'\u06f0'..'\u06f9' |
'\u0966'..'\u096f' |
'\u09e6'..'\u09ef' |
'\u0a66'..'\u0a6f' |
'\u0ae6'..'\u0aef' |
'\u0b66'..'\u0b6f' |
'\u0be7'..'\u0bef' |
'\u0c66'..'\u0c6f' |
'\u0ce6'..'\u0cef' |
'\u0d66'..'\u0d6f' |
'\u0e50'..'\u0e59' |
'\u0ed0'..'\u0ed9' |
'\u1040'..'\u1049'
;
FloatingPointLiteral
: ('0'..'9')+ '.' ('0'..'9')* Exponent? FloatTypeSuffix?
| '.' ('0'..'9')+ Exponent? FloatTypeSuffix?
| ('0'..'9')+ Exponent FloatTypeSuffix?
| ('0'..'9')+ FloatTypeSuffix
| ('0x' | '0X') (HexDigit )*
('.' (HexDigit)*)?
( 'p' | 'P' )
( '+' | '-' )?
( '0' .. '9' )+
FloatTypeSuffix?
;
fragment
Exponent : ('e'|'E') ('+'|'-')? ('0'..'9')+ ;
fragment
FloatTypeSuffix : ('f'|'F'|'d'|'D'|'bd'|'BD') ;
fragment
HexDigit : ('0'..'9'|'a'..'f'|'A'..'F') ;
COMMENT
: '/*' .*? '*/' -> channel(HIDDEN)
;
WS : [ \r\t\u000C\n]+ -> channel(HIDDEN)
;
LINE_COMMENT
: '//' ~[\r\n]* '\r'? '\n' -> channel(HIDDEN)
;
我从其他语法中汲取了一些东西。
我的主要问题是我的expr规则。
给出以下输入:int aaaa = 0..|9;
我的期望是解析树会找到.. |规则,但它将0解释为浮动,并且不能正确解析其余部分。
如果我在我的0之后添加一个空格,它可以正常工作:int aaaa = 0 ..|9;
我需要这个没有空间的工作。
有什么想法吗?
谢谢!
答案 0 :(得分:0)
词法分析器在解析器执行之前对字符输入进行标记。解析器仅对令牌进行操作。将unaryOp
和intervalOp
规则转换为词法分析器规则,以便他们可以参与标记化过程。
<强>更新强>
@CoronA进行了有效的观察。仅建议的改变是不够的。将unaryOp
和intervalOp
规则移至词法分析器并将FloatingPointLiteral规则更改为
FloatingPointLiteral
: ('0'..'9')+ '.' ('0'..'9')+ Exponent? FloatTypeSuffix?
...
(要求小数点后至少有一个数字)足以使有和没有空格的输入正确匹配。有替代方案,但OP需要首先澄清是否需要允许裸小数。