"Давайте создадим компилятор!" - читать интересную книгу автора (Креншоу Джек)Оператор IFПосле этого небольшого пояснения метода мы наконец готовы начать программирование синтаксического анализатора для условного оператора. Фактически, мы уже почти сделали это! Как обычно я буду использовать наш односимвольный подход, с символом "i" вместо «IF» и "e" вместо «ENDIF» (также как и END... это двойственная природа не вызывает никакого беспорядка). Я также пока полностью пропущу символ для условия ветвления, который мы все еще должны определить. Код для DoIf: {–} { Recognize and Translate an IF Construct } procedure Block; Forward; procedure DoIf; var L: string; begin Match('i'); L := NewLabel; Condition; EmitLn('BEQ ' + L); Block; Match('e'); PostLabel(L); end; {–} Добавьте эту подпрограмму в вашу программу и измените Block так, чтобы он ссылался на нее как показано ниже: {–} { Recognize and Translate a Statement Block } procedure Block; begin while not(Look in ['e']) do begin case Look of 'i': DoIf; 'o': Other; end; end; end; {–} Обратите внимание на обращение к процедуре Condition. В конечном итоге мы напишем подпрограмму, которая сможет анализировать и транслировать любое логическое условие которое мы ей дадим. Но это уже тема для отдельной главы (фактически следующей). А сейчас давайте просто заменим ее макетом, который выдает некоторый текст. Напишите следующую подпрограмму: {–} { Parse and Translate a Boolean Condition } { This version is a dummy } Procedure Condition; begin EmitLn('lt;conditiongt;'); end; {–} Вставьте эту процедуру в вашу программу как раз перед DoIf. Теперь запустите программу. Испробуйте строку типа: aibece Как вы можете видеть, синтаксический анализатор, кажется, распознает конструкцию и вставляет объектный код в правильных местах. Теперь попробуйте набор вложенных IF: aibicedefe Он начинает все более походить на настоящий, не так ли? Теперь, когда у нас есть общая идея (и инструменты такие как нотация и процедуры NewLabel и PostLabel) проще пареной репы расширить синтаксический анализатор для поддержки и других конструкций. Первое (а также и одно из самых сложных) это добавление условия ELSE в IF. В БНФ это выглядит так: IF lt;conditiongt; lt;blockgt; [ ELSE lt;blockgt;] ENDIF Сложность возникает просто потому, что здесь присутствует необязательное условие, которого нет в других конструкциях. Соответствующий выходной код должен быть таким: IF lt;conditiongt; BEQ L1 lt;blockgt; BRA L2 L1: lt;blockgt; L2: ... Это приводит нас к следующей синтаксически управляемой схеме перевода: IF lt;conditiongt; { L1 = NewLabel; L2 = NewLabel; Emit(BEQ L1) } lt;blockgt; ELSE { Emit(BRA L2); PostLabel(L1) } lt;blockgt; ENDIF { PostLabel(L2) } Сравнение этого со случаем IF без ELSE дает нам понимание того, как обрабатывать обе эти ситуации. Код ниже выполняет это. (Обратите внимание, что использую "l" вместо «ELSE» так как "e" имеет другое назначение): {–} { Recognize and Translate an IF Construct } procedure DoIf; var L1, L2: string; begin Match('i'); Condition; L1 := NewLabel; L2 := L1; EmitLn('BEQ ' + L1); Block; if Look = 'l' then begin Match('l'); L2 := NewLabel; EmitLn('BRA ' + L2); PostLabel(L1); Block; end; Match('e'); PostLabel(L2); end; {–} Вы получили его. Законченый анализатор/транслятор в 19 строк кода. Сейчас протестируйте его. Испробуйте что-нибудь типа: aiblcede Работает? Теперь, только для того, чтобы убедиться, что мы ничего не испортили и случай с IF без ELSE тоже будет обрабатываться, введите aibece Теперь испробуйте несколько вложенных IF. Испытайте что-нибудь на ваш выбор, включая несколько неправильных утверждений. Только запомните, что 'e' не является допустимым оператором «other». |
|
|