2012-10-04 12 views
17

Wie entferne ich den Shift-Reduce-Konflikt für Bison für die gegebene Grammatik?Die Grammatik reformieren, um die Verschiebung zu entfernen den Konflikt in if-then-else zu reduzieren

Eine Lösung mit der modifizierten Grammatik würde sehr geschätzt werden.

+0

Für zukünftige Personen, [diese Seite] (http://www.tldp.org/HOWTO/Lex-YACC-HOWTO-7. html) half mir durch ein ähnliches Problem. Übergeben Sie auch die Switches '--debug' und' --verbose' an bison und sehen Sie sich die generierten Dateien an. Nicht alle Informationen werden standardmäßig ausgedruckt. –

Antwort

34

Es gibt eine viel einfachere Lösung. Wenn Sie wissen, wie LR Parser arbeiten, dann wissen Sie, dass der Konflikt geschieht hier:

if (expression) statement * else statement 

wo der Stern die aktuelle Position des Cursors markiert. Die Frage, die der Parser beantworten muss, lautet: "Soll ich verschieben, oder sollte ich reduzieren?". Normalerweise möchten Sie den else an den nächsten if binden, was bedeutet, dass Sie den else Token jetzt verschieben möchten. Reduzieren würde jetzt bedeuten, dass Sie die else warten möchten, um an eine "ältere" if gebunden zu werden.

Jetzt wollen Sie Ihren Parser-Generator zu „sagen“, dass „wenn es eine Verschiebung/reduzieren Konflikt zwischen dem Token "else" und die Regel ist‚stm -> if (exp) stm‘, dann muss das Token gewinnen“. Um dies zu tun, geben Sie dem Vorrang Ihrer Regel einen Namen (z. B. "then") und geben Sie an, dass "then" weniger Vorrang hat als "else". Etwas wie:

// Precedences go increasing, so "then" < "else". 
%nonassoc "then" 
%nonassoc "else" 
%% 
stm: "if" "(" exp ")" stm   %prec "then" 
    | "if" "(" exp ")" stm "else" stm 

mit Bisonsyntax.

Eigentlich meine Lieblingsantwort ist sogar "then" und "else" die gleiche Priorität geben. Wenn die Vorbedingungen gleich sind, um die Verbindung zwischen dem Token, das verschoben werden soll, und der Regel, die reduziert werden soll, zu durchbrechen, wird Bison/Yacc auf Assoziativität schauen. Hier wollen Sie mit der rechten Assoziativität zu fördern, um zu sprechen (genauer gesagt, wollen Sie „shift“ fördern), so:

%right "then" "else" // Same precedence, but "shift" wins. 

genügt.

+0

Wie kann ich das gleiche tun, wenn es einen nicht-terminalen N nach Anweisung lautet wie folgt: IF_KEYWORD ‚(‘ Ausdruck N ‚)‘ M-Anweisung N ELSE_KEYWORD M Anweisung Hier, M und N sind Nicht-Terminal-Symbole. – Vishwas

+0

@ VishwasJain Es hängt alles von Ihrer Wenn-Dann-Regel ab. Wenn der "" sonst "M" -Teil optional ist, sollte derselbe Ansatz gelten. – akim

4

Sie müssen die Tatsache erkennen, dass die mittlere statement in der If-else-Fall kann nicht sein (oder mit enden) ein Dangling wenn (ein If mit sonst keiner.) Der einfachste Weg, dies zu tun ist die stmt Regel zu teilen in zwei:

stmt -> stmt-ending-with-dangling-if | stmt-not-ending-with-dangling-if 
stmt-not-ending-with-dangling-if -> 
    if (expression) stmt-not-ending-with-dangling-if else stmt-not-ending-with-dangling-if | 
    ...other statements not ending with dangling if... 
stmt-ending-with-dangling-if -> 
    if (expression) stmt | 
    if (expression) stmt-not-ending-with-dangling-if else stmt-ending-with-dangling-if | 
    ...other statements ending with dangling if... 

jede anderen stmt -> whatever Regel wo whatever mit einem stmt nicht in der stmt-not-ending-with-if Regel geht zu beenden, während jede stmt Regel, das Ende in stmt get Spaltung in zwei Versionen DOES; eine not-ending-with-if Version in der not-ending-with-if Regel und eine dangling-if Version in der dangling-if Regel.

bearbeitet

Eine vollständigere Grammatik mit anderen Produktionen:

stmt : stmt-ending-with-dangling-if | stmt-not-ending-with-dangling-if 
stmt-not-ending-with-dangling-if : 
    IF '(' expr ')' stmt-not-ending-with-dangling-if ELSE stmt-not-ending-with-dangling-if | 
    WHILE '(' expr ')' stmt-not-ending-with-dangling-if | 
    DO stmt WHILE '(' expr ')' ';' | 
    expr ';' | 
    '{' stmt-list '}' 
stmt-ending-with-dangling-if: 
    IF '(' expr ')' stmt | 
    IF '(' expr ')' stmt-not-ending-with-dangling-if ELSE stmt-ending-with-dangling-if | 
    WHILE '(' expr ')' stmt-ending-with-dangling-if 

Regeln wie WHILE (expr) stmt get in zwei Teile gespalten (wie sie mit stmt enden), während Regeln wie expr; nicht.

+0

Können Sie bitte auf die ... andere Aussagen ... Ich habe keine anderen Produktionen mit selection-stmt, aber ich bekomme immer noch eine Fehlermeldung mit nutzlosen nonterminals Das ist, was ich getan habe: selection-stmt \t : öffnen \t \t | geschlossen \t \t; öffnen \t \t: IF LFT_BRKT Ausdruck RGT_BRKT Anweisung \t \t | IF LFT_BRKT Ausdruck RGT_BRKT geschlossen ELSE offen \t \t; geschlossen \t \t: IF LFT_BRKT Ausdruck RGT_BRKT geschlossen ELSE geschlossen \t \t; –

+0

haben bitte einen Blick –

+0

http://stackoverflow.com/questions/12720219/bison-shift-reduce-conflict-unable-to-resolve/12720483#12720483 Die komplette Grammatik ist hier –

0

machen, wenn sonst höher als normale Aussagen, wie:

statements: 
    statements lineEnd statement 
| statements lineEnd IfStat 
| statements lineEnd IfElseStat 
| IfStat 
| IfElseStat 
; 
IfStat: 
    if (statement) 
; 
IfElse: 
    IfStat else statement 
; 
+0

Ich glaube nicht, dass dies das Problem beheben könnte. –