Функции FIRST и FOLLOW
При построении таблицы предсказывающего анализатора нам потребуются две функции - FIRST и FOLLOW.
Пусть G = (N, T, P, S) - КС-грамматика. Для - произвольной цепочки, состоящей из символов грамматики, определим как множество терминалов, с которых
Рис. 4.3.
начинаются строки, выводимые из Если , то e также принадлежит .
Определим FOLLOW(A) для нетерминала A как множество терминалов a, которые могут появиться непосредственно справа отA в некоторой сентенциальной форме грамматики, то есть множество терминалов a таких, что существует вывод вида для некоторых . Заметим, что между A и a в процессе вывода могут находиться нетерминальные символы, из которых выводится e. Если A может быть самым правым символом некоторой сентенциальной формы, то $ также принадлежит FOLLOW(A).
Рассмотрим алгоритмы вычисления функции FIRST.
Алгоритм 4.3. Вычисление FIRST для символов КС- грамматики.
Вход. КС-грамматика G = (N, T, P, S).
Выход. Множество FIRST(X) для каждого символа .
Метод. Выполнить шаги 1-3:
(1) Если X - терминал, то положить FIRST(X) = {X}; если X - нетерминал, положить .
(2) Если в P имеется правило X -> e, то добавить e к FIRST(X).
(3) Пока ни к какому множеству FIRST(X) нельзя уже будет добавить новые элементы, выполнять:
do { continue = false; Для каждого нетерминала X Для каждого правила X -> Y1Y2...Yk {i=1; nonstop = true; while (i <= k && nonstop) {добавить FIRST(Yi) n {e} к FIRST(X); if (Были добавлены новые элементы) continue = true; if (e != FIRST (Yi)) nonstop = false; else i+ = 1; } if (nonstop) {добавить e к FIRST(X); continue = true; } } }while (continue);Алгоритм 4.4. Вычисление FIRST для цепочки.
Вход. КС-грамматика G = (N, T, P, S).
Выход. Множество .
Метод. Выполнить шаги 1-3:
(1) При помощи алгоритма 4.3. вычислить FIRST(X) для каждого .
(2) Положить .
(3)
{i = 1; nonstop = true; while (i <= && nonstop) {добавить FIRST(Xi) n {e} к FIRST(u); if (e not in FIRST(Xi) nonstop = false; else i+ = 1; }if (nonstop) {добавить e к FIRST(u); } }Рассмотрим алгоритм вычисления функции FOLLOW.
Алгоритм 4.5. Вычисление FOLLOW для нетерминалов грамматики.
Вход. КС-грамматика G = (N, T, P, S).
Выход. Множество FOLLOW(X) для каждого символа .
Метод. Выполнить шаги 1-4:
(1) Положить для каждого символа .
(2) Добавить $ к FOLLOW(S).
(3) Если в P eсть правило вывода , где , то все элементы из , за исключением e, добавить к FOLLOW(B).
(4) Пока ничего нельзя будет добавить ни к какому множеству FOLLOW(X), выполнять:
если в P есть правило или , , где содержит , то все элементы из FOLLOW(A) добавить к FOLLOW(B).
Пример 4.4. Рассмотрим грамматику из примера 4.3. Для нее
FIRST(E) = FIRST(T) = FIRST(F) = {(, id}FIRST(E') = {+, e}FIRST(T') = {*, e}FOLLOW(E) = FOLLOW(E') = { ), $}FOLLOW(T) = FOLLOW(T') = {+, ), $}FOLLOW(F) = {+, *, ), $}Например, id и левая скобка добавляются к FIRST(F) на шаге 3 при i = 1, поскольку FIRST(id) = {id} и FIRST("(") = {"("} в соответствии с шагом 1. На шаге 3 при i = 1, в соответствии с правилом вывода T -> FT', к FIRST(T) добавляются такжеid и левая скобка. На шаге 2 в FIRST(E') включается e.
Также при вычислении множеств FOLLOW на шаге 2 в FOLLOW(E) включается $. На шаге 3, на основании правила F -> (E), кFOLLOW(E) добавляется также правая скобка. На шаге 4, примененном к правилу E -> TE', в FOLLOW(E') включаются $ и правая скобка. Поскольку , они также попадают и во множество FOLLOW(T). В соответствии с правилом вывода E -> TE', на шаге 3 в FOLLOW(T) включаются и все элементы из FIRST(E'), отличные от e.
Определим теперь функцию FIRSTk(R), где k - натуральное число и .
либо |w| < k и , либо для некоторого .
Если , то , где w - это первые k символов цепочки при и при .
Приведем алгоритм вычисления функции , где .
Определение. Пусть - некоторый алфавит. Если L1 и L2 - подмножества , то положим
Лемма 4.1. Для любой КС-грамматики и любых
Доказательство оставляем читателю в качестве упражнения.
Aлгоритм 4.6. Вычисление функции .
Вход. КС-грамматика и цепочка .
Выход. .
Метод.Так как по последней лемме
то достаточно показать, как найти FIRSTk(X) для .
Если , то очевидно, что FIRSTk(X) = {X}.
Определим множества Fi(X) для каждого и возрастающих значений i >= 0:
(1) Fi(a) = {a} для всех и i >= 0:
(2) и существует правило из P, для которого либо |x| = k, либо |x| < k и .
(3) Допустим, что F0, F1..., Fi-1 уже определены для всех . Тогда
принадлежит P и
(4) Так как для всех A и i, то в конце концов мы дойдем до такого i, что Fi-1(A) = Fi(A) для всех . Тогда положим FIRSTk(A) = Fi(A) для этого значения i.
Дата добавления: 2016-06-13; просмотров: 3073;