Анимация
JavaScript
|
Главная Библионтека последовательно. Вставьте новую процедуру: { Skip Over an End-of-Line } procedure NewLine; begin while Look = CR do begin GetChar; if Look = LF then GetChar; SkipWhite; end; end; Заметьте, что мы видели эту процедуру раньше в виде процедуры Fin. Я изменил имя, так как новое кажется более соответствующим фактическому назначению. Я также изменил код чтобы учесть множественные переносы и строки только с пробелами. Следующим шагом будет вставка вызовов NewLine везде, где мы посчитаем перенос допустимым. Как я подчеркивал ранее, этот момент может очень различаться для разных языков. В TINY я решил разрешить их практически в любом месте. Это означает, что нам нужно вызывать NewLine в начале (не в конце как с SkipWhite) процедур GetName, GetNum и Match. Для процедур, которые имеют циклы While, таких как TopDecl, нам нужен вызов NewLine в начале процедуры и в конце каждого цикла. Таким способом мы можем быть уверены, что NewLine вызывается в начале каждого прохода через цикл. Если вы все это сделали, испытайте программу и проверьте, что она действительно обрабатывает пробелы и переносы. Если это так, тогда мы готовы работать с многосимвольными токенами и ключевыми словами. Для начала, добавьте дополнительные объявления (скопированные почти дословно из главы 7): { Type Declarations } type Symbol = string[8]; SymTab = array[1..1000] of Symbol; TabPtr = SymTab; { Variable Declarations } var Look : char; { Lookahead Character } Token: char; { Encoded Token } Value: string[16]; { Unencoded Token } ST: Array[A..Z] of char; { Definition of Keywords and Token Types } const NKW = 9; NKW1 = 10; const KWlist: array[1..NKW] of Symbol = (IF, ELSE, ENDIF, WHILE, ENDWHILE, VAR, BEGIN, END, PROGRAM); const KWcode: string[NKW1] = xilewevbep; Затем добавьте три процедуры, также из седьмой главы: { Table Lookup } function Lookup(T: TabPtr; s: string; n: integer): integer; var i: integer; found: Boolean; begin found := false; i := n; while (i > 0) and not found do if s = T*[i] then found := true else dec(i); Lookup := i; end; { Get an Identifier and Scan it for Keywords } procedure Scan; begin GetName; Token := KWcode[Lookup(Addr(KWlist), Value, NKW) + 1]; end; { Match a Specific Input String } procedure MatchString(x: string); begin if Value <> x then Expected( + x + ); end; Теперь мы должны сделать довольно много тонких изменений в оставшихся процедурах. Сначала мы должны изменить функцию GetName на процедуру, снова как в главе 7: { Get an Identifier } procedure GetName; begin NewLine; if not IsAlpha(Look) then Expected(Name); Value := ; while IsAlNum(Look) do begin Value := Value + UpCase(Look); GetChar; end; SkipWhite; end; Обратите внимание, что эта процедура оставляет свой результат в глобальной строковой переменной Value. Затем, мы должны изменить каждую обращение к GetName чтобы отразить ее новую форму. Они происходят в Factor, Assignment и Decl: { Parse and Translate a Math Factor } procedure BoolExpression; Forward; procedure Factor; begin if Look = ( then begin MatchCC); BoolExpression; Match()); end else if IsAlpha(Look) then begin GetName; LoadVar(Value[1]); end else LoadConst(GetNum); end; { Parse and Translate an Assignment Statement } procedure Assignment; var Name: char; begin Name := Value[1]; MatchC = ); BoolExpression; Store(Name); end; { Parse and Translate a Data Declaration } procedure Decl; begin GetName; Alloc(Value[1]); while Look = , do begin Match(,); GetName; Alloc(Value[1]); end; end; (Заметьте, что мы все еще разрешаем только односимвольные имена переменных поэтому мы используем здесь простое решение и просто используем первый символ строки.) Наконец, мы должны внести изменения, позволяющие использовать Token вместо Look как символа для проверки и вызывать Scan в подходящих местах. По большей части это включает удаление вызовов Match, редкие замены вызовов Match на вызовы MatchString, и замену вызовов NewLine на вызовы Scan. Вот затронутые подпрограммы: { Recognize and Translate an IF Construct } procedure Block; Forward; procedure DoIf; var L1, L2: string; begin 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 [ 47 ] 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 |