Устранение левой рекурсии — различия между версиями
Shagal (обсуждение | вклад) (→Устранение произвольной левой рекурсии) |
Shagal (обсуждение | вклад) (Отмена правки 27816 участника Shagal (обсуждение)) |
||
Строка 45: | Строка 45: | ||
<div> | <div> | ||
for все нетерминалы <tex>A_i</tex> | for все нетерминалы <tex>A_i</tex> | ||
− | for все нетерминалы <tex>A_j</tex>, такие, что <tex> 1 \leq j < i </tex> | + | for все нетерминалы <tex>A_j</tex>, такие, что <tex> 1 \leq j < i </tex> и |
− | + | рассмотреть все правила вывода из <tex>A_j</tex>: <tex>A_j \rightarrow \delta_1 | \ldots | \delta_k</tex>. | |
− | + | заменить каждое правило <tex>A_i \rightarrow A_j \gamma</tex> на <tex>A_i \rightarrow \delta_1\gamma | \ldots | \delta_k\gamma</tex>. | |
устранить непосредственную левую рекурсию для <tex>A_i</tex>. | устранить непосредственную левую рекурсию для <tex>A_i</tex>. | ||
</div> | </div> | ||
− | + | Таким образом, после применения алгоритма все правила вывода имеют вид: | |
+ | *<tex>A \rightarrow c \alpha </tex>, где <tex>c</tex> {{---}} терминал, <tex>A</tex> {{---}} произвольный нетерминал; | ||
+ | *<tex>A_i \rightarrow A_j \alpha </tex>, где <tex>i < j</tex>, <tex>A_i , A_j</tex> {{---}} нетерминалы из исходной грамматики; | ||
+ | *<tex>A_i^{\prime} \rightarrow A_j \alpha </tex>, где <tex>A_i^{\prime}</tex> {{---}} новый нетерминал, <tex>A_j</tex> {{---}} нетерминал из исходной грамматики. | ||
− | + | Если теперь перенумеровать нетерминалы, сохранив порядок для старых и присвоив всем новым меньшие номера, то все правила будут иметь вид: | |
+ | *<tex>B_i \rightarrow c \alpha </tex>, где <tex>c</tex> {{---}} терминал; | ||
+ | *<tex>B_i \rightarrow B_j \alpha </tex>, где <tex>i < j</tex>. | ||
==Алгоритм устранения левой рекурсии== | ==Алгоритм устранения левой рекурсии== |
Версия 12:33, 3 января 2013
Определение: |
Говорят, что контекстно-свободная(КС) грамматика содержит непосредственную левую рекурсию, если она содержит правило вида . |
Определение: |
Говорят, что КС-грамматика | содержит левую рекурсию(left recursion), если в ней существует вывод вида .
Методы нисходящего разбора(top=down parsers) не в состоянии работать с леворекурсивными грамматиками. Проблема в том, что продукция вида может применяться бесконечно долго, так и не выработав некий терминальный символ, который можно было бы сравнить со строкой. Поэтому требуется преобразование грамматики, которое бы устранило левую рекурсию.
Левая рекурсия может быть:
- непосредственной(immidiate)
- произвольной(indirect)
Содержание
Устранение непосредственной левой рекурсии
Опишем процедуру, устраняющую все правила вида
, для фиксированного нетерминала .- Запишем все правила вывода из
- — непустая последовательность терминалов и нетерминалов ( );
- — непустая последовательность терминалов и нетерминалов, не начинающаяся с .
в виде:
, где
- Заменим правила вывода из на .
- Создадим новый нетерминал .
Этот алгоритм не устраняет произвольную левую рекурсию,вызванную двумя или более шагами порождения.
Пример
S леворекурсивен, так как
, но эта рекурсия не является непосредственной.Устранение произвольной левой рекурсии
Пусть
— упорядоченное множество всех нетерминалов.for все нетерминалыfor все нетерминалы , такие, что и рассмотреть все правила вывода из : . заменить каждое правило на . устранить непосредственную левую рекурсию для .
Таким образом, после применения алгоритма все правила вывода имеют вид:
- , где — терминал, — произвольный нетерминал;
- , где , — нетерминалы из исходной грамматики;
- , где — новый нетерминал, — нетерминал из исходной грамматики.
Если теперь перенумеровать нетерминалы, сохранив порядок для старых и присвоив всем новым меньшие номера, то все правила будут иметь вид:
- , где — терминал;
- , где .
Алгоритм устранения левой рекурсии
Для произвольной грамматики
левую рекурсию можно устранить следующим образом:- Воспользуемся алгоритмом удаления . Получим грамматику без -правил -правил для языка .
- Воспользуемся алгоритмом устранения произвольной левой рекурсии.
- Если присутствовал в языке исходной грамматики, добавим новый начальный символ и правила .
Литература
- Хопкрофт Д., Мотвани Р., Ульман Д. — Введение в теорию автоматов, языков и вычислений, 2-е изд. : Пер. с англ. — Москва, Издательский дом «Вильямс», 2002. — 528 с. : ISBN 5-8459-0261-4 (рус.)
- Robert C. Moore — Removing Left Recursion from Context-Free Grammars