看看我的语法
grammar protocol;
options {
language = Java;
output = AST;
}
//imaginary tokens
tokens{
BOOL;
CHAR;
STRING;
}
parse
: declaration
;
declaration
: variable
;
variable
: locals
;
locals
: (bool
| char
| string)+
;
bool
:'bool' ID -> ^(BOOL ID)
;
char
: 'char' ID -> ^(CHAR ID)
;
string
:'string' ID -> ^(STRING ID)
;
ID
: (('a'..'z' | 'A'..'Z'|'_')('a'..'z' | 'A'..'Z'|'0'..'9'|'_'))*
;
INT
: ('0'..'9')+
;
WHITESPACE
: ('\t' | ' ' | '\r' | '\n' | '\u000C')+ {$channel = HIDDEN;}
;
对于以下输入,
bool boolVariable
char charVariable
string stringVariable
我的语法创建了以下 AST
我不能多次声明一个变量。我不想立即声明相同类型的变量,并用逗号分隔,但我想要这样
bool boolVariable1
bool boolVariable2
bool boolVariable3
string stringVariable1
string stringVariable2
完成此操作后,我希望所有变量都是两种主要类型。共享和本地。在Java中,共享变量(静态)是所有对象都有一个副本的变量,而局部变量为每个对象都有单独的副本。我希望用户在定义变量集之前显式指定变量的范围。喜欢,
locals:
bool boolVariable1
bool boolVariable2
bool boolVariable3
string stringVariable1
string stringVariable2
shared:
bool boolVariable4
bool boolVariable5
bool boolVariable6
string stringVariable3
string stringVariable4
char charVariable1
此外,有什么方法可以检查用户不能有两个同名的变量吗?就像,
bool boolVariable
bool boolVariable
应该给出某种错误或类似的错误。
有什么想法/帮助吗?
谢谢你
编辑 - 解决方案
grammar protocol;
options {
language = Java;
output = AST;
}
//imaginary tokens
tokens{
BOOL;
CHAR;
STRING;
SBOOL;
SCHAR;
SSTRING;
}
parse
: declaration
;
declaration
: variable
;
variable
: (locals
| shared)*
;
locals
: 'locals:' (bool| char| string)*
;
bool
:'bool' ID -> ^(BOOL ID)
;
char
: 'char' ID -> ^(CHAR ID)
;
string
:'string' ID -> ^(STRING ID)
;
shared
: 'shared:' (sbool| schar| sstring)*
;
sbool
:'bool' ID -> ^(SBOOL ID)
;
schar
: 'char' ID -> ^(SCHAR ID)
;
sstring
:'string' ID -> ^(SSTRING ID)
;
ID
: (('a'..'z' | 'A'..'Z'|'_')('a'..'z' | 'A'..'Z'|'0'..'9'|'_'))*
;
INT
: ('0'..'9')+
;
WHITESPACE
: ('\t' | ' ' | '\r' | '\n' | '\u000C')+ {$channel = HIDDEN;}
;
正如巴丹在他的回答中提到的那样,您需要维护已使用过的名称的集合。这是一个基于您更新的语法的简单示例(还有一些其他更改,如本答案底部所述)。新规则
var
是使用新会员代码的地方。请注意,这里没有进行真正的错误处理,只是进行名称检查。
grammar protocol;
options {
language = Java;
output = AST;
}
//imaginary tokens
tokens{
BOOL;
CHAR;
STRING;
SBOOL;
SCHAR;
SSTRING;
}
@parser::header {
import java.util.ArrayList;
}
@members {
private ArrayList<String> variableNames = new ArrayList<String>();
private boolean variableDefined(String name){
return variableNames.contains(name);
}
private void defineVariable(String name){
variableNames.add(name);
}
}
parse
: declaration
;
declaration
: variable
;
variable
: (locals | shared)*
;
locals
: 'locals:' (bool| char_ | string)*
;
bool
:'bool' var -> ^(BOOL var)
;
char_
: 'char' var -> ^(CHAR var)
;
string
:'string' var -> ^(STRING var)
;
shared
: 'shared:' (sbool| schar| sstring)*
;
sbool
:'bool' var -> ^(SBOOL var)
;
schar
: 'char' var -> ^(SCHAR var)
;
sstring
:'string' var -> ^(SSTRING var)
;
var
: ID
{!variableDefined($ID.text)}? //This rule is only satisfied if the variable is new.
{defineVariable($ID.text);} //we made it here, so it's new. Add it for future reference.
;
ID
: ('a'..'z' | 'A'..'Z'|'_')('a'..'z' | 'A'..'Z'|'0'..'9'|'_')*
;
INT
: ('0'..'9')+
;
WHITESPACE
: ('\t' | ' ' | '\r' | '\n' | '\u000C')+ {$channel = HIDDEN;}
;
我可以声明两个变量,如 bool boolVariable bool boolVariable,但我不能声明两个变量,如 bool boolVariable bool boolVariable12
请参阅我对上面
ID
的更改。有一组额外的括号扰乱了规则。我还将 char
重命名为 char_
以使解析器为我正确编译。
你可以试试这个:
locals
: bool* char* string*
;
它应该允许您声明具有相同类型的不同变量。 禁止声明具有相同名称的不同变量的最佳(我认为)方法是使用先前声明的变量保留集合(在 Java 中)。只需调用 Java 函数,如here