LR(k)-грамматики — различия между версиями
Shersh (обсуждение | вклад)  (→Управляющая программа анализатора)  | 
				Shersh (обсуждение | вклад)  м (→Алгоритм)  | 
				||
| Строка 103: | Строка 103: | ||
               s = top()  |                s = top()  | ||
               push(<tex>A</tex>)  |                push(<tex>A</tex>)  | ||
| − |                push(goto   | + |                push(goto(s, <tex>A</tex>))    | 
               Вывод правила (<tex> A \to \beta</tex>)  |                Вывод правила (<tex> A \to \beta</tex>)  | ||
           '''else''' '''if''' <tex>\mathtt{T}</tex>[curState, curToken] == Accept    |            '''else''' '''if''' <tex>\mathtt{T}</tex>[curState, curToken] == Accept    | ||
Версия 20:03, 3 сентября 2015
Восходящий разбор (англ. Bottom-up parsing) предназначен для построения дерева разбора. Мы можем представить себе этот процесс как "свертку" исходной строки к стартовому нетерминалу грамматики. Каждый шаг свертки заключается в сопоставлении некоторой подстроки и правой части какого-то правила грамматики, затем происходит замена этой подстроки на нетерминал, являющийся левой частью правила. Восходящий разбор менее интуитивный, чем нисходящий, но зато позволяет разбирать большее множество грамматик.
Содержание
LR(k)-грамматика
Определение
| Определение: | 
| Пусть — контекстно-свободная грамматика. Пополненной грамматикой (англ. augmented grammar), полученной из , назовем грамматику , где | 
| Определение: | 
Пусть  — пополненная грамматика для КС-грамматики . Грамматика  является LR(k)-грамматикой, если из того, что для любых двух правосторонних выводов верно, что:
 следует, что , тогда и . | 
Говоря неформально, мы делаем правостороннюю свёртку нашей строки в стартовый нетерминал. Если по не более чем символам неразобранной части строки мы можем однозначно определить, во что сворачивается хвост выведенного правила, то грамматика будет LR(k).
LR(k) означает, что
- входная цепочка обрабатывается слева направо (англ. left-to-right parse),
 - выполняется правый вывод (англ. rightmost derivation),
 - не более символов цепочки (англ. k-token lookahead) используются для принятия решения.
 
Замечание о пополненной грамматике
Использование в определении LR(k)-грамматики пополненной грамматики существенно для однозначного определения конца анализа. Действительно, если грамматика использует в правых частях правил, то свертка основы в не может служить сигналом приема входной цепочки. Свертка же в в пополненной грамматике служит таким сигналом, поскольку нигде, кроме начальной сентенциальной формы, не встречается.
Существенность использования пополненной грамматики в определении LR(k)-грамматик продемонстрируем на следующем конкретном примере. Пусть пополненная грамматика имеет следующие правила:
Если игнорировать -е правило, то, не заглядывая в правый контекст основы , можно сказать, что она должна сворачиваться в . Аналогично основа безусловно должна сворачиваться в . Создается впечатление, что данная грамматика без -го правила есть LR(0)-грамматика. Но на самом деле это не так, в этом можно убедиться рассмотрев процесс LR(0)-разбора.
LR-разборщик
Принцип переноса-свёртки
При LR(k)-анализе применяется метод перенос-свертка (англ. shift-reduce). Суть метода сводится к следующему:
- Программа анализатора читает последовательно символы входной строки до тех пор, пока не накопится цепочка, совпадающая с правой частью какого-нибудь из правил. Рассмотренные символы переносим в стек (операция перенос).
 - Далее все символы совпадающей цепочки извлекаются из стека и на их место помещается нетерминал, находящийся в левой части этого правила (операция свертка).
 
Структура
Метод перенос-свертка использует следующие компоненты:
- входная строка,
 - стек (для запоминания рассмотренных символов),
 - управляющая таблица (для определения, какое действие применить — перенос или свертку),
 - автомат (для запоминания информации о текущем состоянии стека).
 
Управляющая программа анализатора
Управляющая программа одинакова для всех LR-анализаторов, а таблица и автомат изменяются от одного анализатора к другому.
Для запоминания строки запись в стек имеет вид: , где — вершина стека. Каждый — символ грамматики (терминал или нетерминал), а — состояние автомата. Каждое состояние суммирует информацию, cодержащуюся в стеке перед ним. — стартовое состояние автомата. Комбинация символа состояния на вершине стека и текущего входного символа используется для индексирования управляющей таблицы и определения операции переноса-свертки. При реализации грамматические символы не обязательно располагаются в стеке, однако, мы будем использовать их при обсуждении для лучшего понимания поведения LR-анализатора.
Обращение к таблице происходит следующим образом , где
- — состояние автомата,
 - — входной символ.
 
Полученное значение в таблице должно информировать о текущем действии, то есть о переносе или свертке. В этих двух случаях, необходима дополнительная информация: к какому состоянию происходит переход (при переносе) и по какому правилу происходит свертка. В случае некорректного входного символа происходит ошибка, а свертка в стартовое состояние идентифицируется как допуск:
enum Cell
    Shift 
    Reduce 
    Error    // ошибка
    Accept   // допуск 
struct Shift 
    state: int  // переход в стостояние state
struct Reduce 
    rule: int   // свертка по правилу rule
Рузультатом работы управляющей программы будет:
enum Result
    Accept   // допуск 
    Error    // ошибка
Алгоритм
- Программа читает символ из входной цепочки.
 - Обращается к управляющей таблице.
 - Совершает соответствующее действие.
 - Возвращается к первому пункту, пока входная цепочка не закончится.
 
| 
 
  Result algorithmLR(w: string)
     // curToken — указатель на первый символ в строке w
     while hasTokens()
         curState = top()
         if [curState, curToken] == Shift s 
             push(curToken)
             push(s)
             nextToken()
         else if [curState, curToken] == Reduce  
             for j = 1 to   
                 pop()
                 pop()
             s = top()
             push()
             push(goto(s, )) 
             Вывод правила ()
         else if [curState, curToken] == Accept 
             return Accept
         else 
             return Error     
  | 
Функция получает состояние и символ грамматики и выдает состояние. Функция , строящаяся по грамматике , есть функция переходов детерминированного магазинного автомата, который распознает язык, порождаемый грамматикой .
См. также
Источники информации
- Альфред Ахо, Рави Сети, Джеффри Ульман. Компиляторы. Принципы, технологии, инструменты. Издательство Вильямс, 2003. Стр. 301-326.
 - Терехов Ан.А., Вояковская Н., Булычев Д., Москаль А. Разработка компиляторов на платформе .NET — Восходящие анализаторы
 - Б.К.Мартыненко. Языки и трансляции. Стр. 198-223