5

おい。私はANTLRを初めて使用します。ANTLRWorksウィザードは私のために次のコードを書きました:

grammar test;

ID  :   ('a'..'z'|'A'..'Z'|'_') ('a'..'z'|'A'..'Z'|'0'..'9'|'_')*
    ;

INT :   '0'..'9'+
    ;

FLOAT
    :   ('0'..'9')+ '.' ('0'..'9')* EXPONENT?
    |   '.' ('0'..'9')+ EXPONENT?
    |   ('0'..'9')+ EXPONENT
    ;

COMMENT
    :   '//' ~('\n'|'\r')* '\r'? '\n' {$channel=HIDDEN;}
    |   '/*' ( options {greedy=false;} : . )* '*/' {$channel=HIDDEN;}
    ;

WS  :   ( ' '
        | '\t'
        | '\r'
        | '\n'
        ) {$channel=HIDDEN;}
    ;

STRING
    :  '"' ( ESC_SEQ | ~('\\'|'"') )* '"'
    ;
CHAR:  '\'' ( ESC_SEQ | ~('\''|'\\') ) '\''
    ;

fragment
EXPONENT : ('e'|'E') ('+'|'-')? ('0'..'9')+ ;

fragment
HEX_DIGIT : ('0'..'9'|'a'..'f'|'A'..'F') ;

fragment
ESC_SEQ
    :   '\\' ('b'|'t'|'n'|'f'|'r'|'\"'|'\''|'\\')
    |   UNICODE_ESC
    |   OCTAL_ESC
    ;

fragment
OCTAL_ESC
    :   '\\' ('0'..'3') ('0'..'7') ('0'..'7')
    |   '\\' ('0'..'7') ('0'..'7')
    |   '\\' ('0'..'7')
    ;

fragment
UNICODE_ESC
    :   '\\' 'u' HEX_DIGIT HEX_DIGIT HEX_DIGIT HEX_DIGIT
    ;

デバッグすると、次のエラーがスローされます。

[22:45:49] error(100): C:\Documents and Settings\user\Desktop\test.g:0:0: syntax error: codegen: <AST>:0:0: unexpected end of subtree

誰かがエラーとは何か、どこにあるのか、どうすれば修正できるのか説明してもらえますか?

ありがとう。

4

2 に答える 2

7

ANTLRでは、大文字で始まるすべてのルールはレクサールールです。小文字で始まるものはパーサールールです。ご覧のとおり、字句解析ルールしかありません。問題があります。少なくとも1つのパーサールールが必要です。次のルールを追加した場合:

parse
  :  ID
  |  INT
  |  // ...
  ;

レクサー/パーサーのソースファイルを生成すると、エラーは消えます。

于 2009-12-17T08:27:16.640 に答える
2

免責事項: ANTLR ウィザードについては何も知りません。

Google 検索で次の引用が表示されます。

通常、「サブツリーの予期しない終了」は、パーサーで何かをルートにするのを忘れたことを意味します。

ファイルが字句解析のルールだけでなく、文法を指定することになっている場合、これは私には理にかなっています。ファイルの最初の行は「grammar test」なので、おそらくこれは文法です。

文法を使用すると、一連の終端記号を 1 つの非終端記号に減らすことができます。たとえば、完全に括弧で囲まれた式を表す非常に単純な文法は次のようになります。

P : E
E : (X)
  | E E
  | (E)
X : 'x'

ここで、すべての文が最終的に P に還元されるため、P がルートです。文が P に還元できない場合、その文はこの文法に一致しません。したがって、文法のルートを見つける必要があり、他のすべてのプロダクションはルートのコンテキストでのみ (つまり、直接的または間接的な派生を介して) 出現する必要があります。

于 2009-12-16T21:05:15.440 に答える