Как решить бизон уменьшить / уменьшить конфликт в моем коде?
Я новичок в бизоне... Я написал грамматическое правило для оператора if, else if и else... Я получил уменьшить конфликт, хотя... Кто-нибудь может помочь? Я перепробовал все, что нашел, но, как я уже сказал, я новичок, и я точно не понимаю, что происходит..
Вот мой код:
ifinstr: KW_IF expr_decl KW_THEN statements elseifinstr elseinstr KW_END
;
elseifinstr : %empty {$$ = "";}
| elseifinstr KW_ELSE KW_IF expr_decl KW_THEN statement
;
elseinstr : %empty {$$ = "";}
| KW_ELSE statement
;
Я тоже пробовал это решение, но вместо этого получил конфликт сдвиг / уменьшение:
ifinstr: KW_IF expr_decl KW_THEN statements elseifinstr KW_END
;
elseifinstr : %empty {$$ = "";}
| elseifinstr KW_ELSE KW_IF expr_decl KW_THEN statement
| KW_ELSE statement
;
1 ответ
Проблема в том, что предлагаемая вами грамматика неоднозначна. (Я имею в виду первую грамматику. Предлагаемое второе решение - это другой язык.)
Языки, которые используют конструкцию "else if" для сокращения вложенности скобок "if … end", используют специальный токен:
Ruby Python VB Shell
---------------- ------------------- ----------------- --------------------
if n > 0 if n > 0: If n > 0 Then if ((n>0)); then
puts "Greater" print ("Greater") Print "Greater" echo Greater
elsif n == 0 elif n == 0: ElseIf n = 0 Then elif ((n==0)); then
puts "Equal" print ("Equal") Print "Equal" echo Equal
else else: Else else
puts "Less" print ("Less") Print "Less" echo Less
end End fi
Без слитного "else if" эти выражения были бы намного более громоздкими:
Ruby Python VB Shell
---------------- ------------------- ----------------- --------------------
if n > 0 if n > 0: If n > 0 Then if ((n>0)); then
puts "Greater" print ("Greater") Print "Greater" echo Greater
else else: Else else
if n == 0 if n == 0: If n = 0 Then if ((n==0)); then
puts "Equal" print ("Equal") Print "Equal" echo Equal
else else: Else else
puts "Less" print ("Less") Print "Less" echo Less
end end End fi
end End fi
Это контрастирует с такими языками, как C, Java и многими другими, которые не требуют, чтобы операторы "если" были завершены, и, таким образом, демонстрируют конфликт "висячий еще", сдвиг-уменьшение, который всегда разрешается в пользу сдвига. В этих языках, так как предложение "else" просто присоединяется к ближайшему непревзойденному предложению "if", нет необходимости предоставлять специальный токен "else-if".
Теперь вы пытаетесь объединить эти два подхода, используя объединенное "else-if", не смешивая его, и это просто приводит к той же висячей проблеме else, которую пытался исправить явный брекетинг. Рассмотрим, например:
if C1 then S1 else if C2 then S2 else if C3 then S3 else S4 end S5 end
Теперь, что из следующих двух это представляет?
if C1 then if C1 then
S1 S1
else if C2 then else
S2 if C2 then
else S2
if C3 then else if C3 then
S3 S3
else else
S4 S4
end end
S5 S5
end end
(Со временем я мог бы найти более простой пример. Приведенные выше две интерпретации отличаются в обстоятельствах, в которых S5
выполняется.)
Самое простое решение - использовать какой-то токен "else if", как в приведенных выше примерах языков. (Или придумай свое:-))