0

次の文法を定義しました。

grammar Sample_1;


@header {
  package a;
}

@lexer::header {
  package a;
}

program
    :   
        define*
        implement*
    ;


define
    :   IDENT '=(' INTEGER',' INTEGER ')'
    ;

implement
    :IDENT '=(' (IDENT ','?)* ')'
    ;

fragment LETTER : ('a'..'z' | 'A'..'Z') ;
fragment DIGIT : '0'..'9';
INTEGER : DIGIT+ ;
IDENT : LETTER (LETTER | DIGIT)*;
WS : (' ' | '\t' | '\n' | '\r' | '\f')+ {$channel = HIDDEN;};
COMMENT : '//' .* ('\n'|'\r') {$channel = HIDDEN;};

この文法をチェックインして、例があるときに

A=(1,1)
B=(1,2)

G=(A,B)

結果は成功しますが、私が書くと

A=(1,1)
B=(1,2)

G=(A,E)

Eが定義されていないというエラーが発生します

結果:私はそれをたくさんのおかげで動作させました:

grammar Sample_1;

@members{
    int level=0;
}

@header {
  package a;
}

@lexer::header {
  package a;
}

program
    :   
        block
    ;
block   
scope {
    List symbols;
}
@init {
    $block::symbols=new ArrayList();
    level++;
}
@after { 
     System.err.println("Hello");
     level--;
 }
    : (define* implement+)
    ;

define
    :   IDENT {$block::symbols.add($IDENT.text);} '=(' INTEGER',' INTEGER ')' 
    ;

implement
    :IDENT '=(' (a=IDENT 
    {if (!$block::symbols.contains($a.text)){
    System.err.println("undefined");
    }}','?)* ')'
    ;

fragment LETTER : ('a'..'z' | 'A'..'Z') ;
fragment DIGIT : '0'..'9';
INTEGER : DIGIT+ ;
IDENT : LETTER (LETTER | DIGIT)*;
WS : (' ' | '\t' | '\n' | '\r' | '\f')+ {$channel = HIDDEN;};
COMMENT : '//' .* ('\n'|'\r') {$channel = HIDDEN;};
4

1 に答える 1

3

Antlrは、アクション、文法ファイルに埋め込まれたコードの小さなスニペットをサポートしています。

割り当てのアクションをマップに保存できます。右側のIDENTのアクションは、マップから値をプルしようとし、失敗した場合は例外をスローする可能性があります。

TerrenceParrの「TheDefinitiveANTLRReference」の第6章では、アクションについて説明しています。

于 2010-12-28T19:36:26.487 に答える