Building a Scanner CS164, Fall 2004 Ras Bodik, CS 164, Fall 2004.

Slides:



Advertisements
Similar presentations
4b Lexical analysis Finite Automata
Advertisements

1 1 CDT314 FABER Formal Languages, Automata and Models of Computation Lecture 3 School of Innovation, Design and Engineering Mälardalen University 2012.
COMP-421 Compiler Design Presented by Dr Ioanna Dionysiou.
From Cooper & Torczon1 The Front End The purpose of the front end is to deal with the input language Perform a membership test: code  source language?
Winter 2007SEG2101 Chapter 81 Chapter 8 Lexical Analysis.
Context-Free Grammars Lecture 7
Prof. Hilfinger CS 164 Lecture 21 Lexical Analysis Lecture 2-4 Notes by G. Necula, with additions by P. Hilfinger.
1 Foundations of Software Design Lecture 23: Finite Automata and Context-Free Grammars Marti Hearst Fall 2002.
1 Chapter 3 Scanning – Theory and Practice. 2 Overview Formal notations for specifying the precise structure of tokens are necessary  Quoted string in.
Automating Construction of Lexers. Example in javacc TOKEN: { ( | | "_")* > | ( )* > | } SKIP: { " " | "\n" | "\t" } --> get automatically generated code.
CS 426 Compiler Construction
1 Scanning Aaron Bloomfield CS 415 Fall Parsing & Scanning In real compilers the recognizer is split into two phases –Scanner: translate input.
Chapter 3 Lexical Analysis
Topic #3: Lexical Analysis
CPSC 388 – Compiler Design and Construction Scanners – Finite State Automata.
Finite-State Machines with No Output
1 Outline Informal sketch of lexical analysis –Identifies tokens in input string Issues in lexical analysis –Lookahead –Ambiguities Specifying lexers –Regular.
Lexical Analysis - An Introduction. The Front End The purpose of the front end is to deal with the input language Perform a membership test: code  source.
Lexical Analysis - An Introduction Copyright 2003, Keith D. Cooper, Ken Kennedy & Linda Torczon, all rights reserved. Students enrolled in Comp 412 at.
Lexical Analysis - An Introduction Copyright 2003, Keith D. Cooper, Ken Kennedy & Linda Torczon, all rights reserved. Students enrolled in Comp 412 at.
Lecture # 3 Chapter #3: Lexical Analysis. Role of Lexical Analyzer It is the first phase of compiler Its main task is to read the input characters and.
Automating Construction of Lexers. Example in javacc TOKEN: { ( | | "_")* > | ( )* > | } SKIP: { " " | "\n" | "\t" } --> get automatically generated code.
COMP 3438 – Part II - Lecture 2: Lexical Analysis (I) Dr. Zili Shao Department of Computing The Hong Kong Polytechnic Univ. 1.
Lexical Analysis I Specifying Tokens Lecture 2 CS 4318/5531 Spring 2010 Apan Qasem Texas State University *some slides adopted from Cooper and Torczon.
Lexical Analyzer (Checker)
1 Course Overview PART I: overview material 1Introduction 2Language processors (tombstone diagrams, bootstrapping) 3Architecture of a compiler PART II:
COP 4620 / 5625 Programming Language Translation / Compiler Writing Fall 2003 Lecture 3, 09/11/2003 Prof. Roy Levow.
CS412/413 Introduction to Compilers Radu Rugina Lecture 4: Lexical Analyzers 28 Jan 02.
CS 536 Fall Scanner Construction  Given a single string, automata and regular expressions retuned a Boolean answer: a given string is/is not in.
Lexical Analysis: Finite Automata CS 471 September 5, 2007.
1 Course Overview PART I: overview material 1Introduction 2Language processors (tombstone diagrams, bootstrapping) 3Architecture of a compiler PART II:
Joey Paquet, 2000, Lecture 2 Lexical Analysis.
Overview of Previous Lesson(s) Over View  Symbol tables are data structures that are used by compilers to hold information about source-program constructs.
Lexical Analysis.
1st Phase Lexical Analysis
CS412/413 Introduction to Compilers and Translators Spring ’99 Lecture 2: Lexical Analysis.
Prof. Necula CS 164 Lecture 31 Lexical Analysis Lecture 3-4.
Chapter 2 Scanning. Dr.Manal AbdulazizCS463 Ch22 The Scanning Process Lexical analysis or scanning has the task of reading the source program as a file.
Overview of Previous Lesson(s) Over View  A token is a pair consisting of a token name and an optional attribute value.  A pattern is a description.
CS 404Ahmed Ezzat 1 CS 404 Introduction to Compiler Design Lecture 1 Ahmed Ezzat.
LECTURE 5 Scanning. SYNTAX ANALYSIS We know from our previous lectures that the process of verifying the syntax of the program is performed in two stages:
Lecture 2 Compiler Design Lexical Analysis By lecturer Noor Dhia
Department of Software & Media Technology
Topic 3: Automata Theory 1. OutlineOutline Finite state machine, Regular expressions, DFA, NDFA, and their equivalence, Grammars and Chomsky hierarchy.
WELCOME TO A JOURNEY TO CS419 Dr. Hussien Sharaf Dr. Mohammad Nassef Department of Computer Science, Faculty of Computers and Information, Cairo University.
CS 3304 Comparative Languages
CS510 Compiler Lecture 2.
Lecture 2 Lexical Analysis
Chapter 3 Lexical Analysis.
Chapter 2 :: Programming Language Syntax
Lecture 2 Lexical Analysis Joey Paquet, 2000, 2002, 2012.
Chapter 2 Scanning – Part 1 June 10, 2018 Prof. Abdelaziz Khamis.
Finite-State Machines (FSMs)
Lexical analysis Finite Automata
Compilers Welcome to a journey to CS419 Lecture5: Lexical Analysis:
RegExps & DFAs CS 536.
Finite-State Machines (FSMs)
Recognizer for a Language
Hierarchy of languages
Lexical Analysis Lecture 3-4 Prof. Necula CS 164 Lecture 3.
CS 3304 Comparative Languages
Lecture 4: Lexical Analysis & Chomsky Hierarchy
4b Lexical analysis Finite Automata
CS 3304 Comparative Languages
Chapter 2 :: Programming Language Syntax
4b Lexical analysis Finite Automata
Instructor: Aaron Roth
Lexical Analysis.
Chapter 2 :: Programming Language Syntax
Lecture 5 Scanning.
Presentation transcript:

Building a Scanner CS164, Fall 2004 Ras Bodik, CS 164, Fall 2004

Administrativia Extra credit for bugs in project assignments in starter kits and handouts TAs are final arbiters of what’s a bug only the first student to report the bug gets credit Ras Bodik, CS 164, Fall 2004

What does a lexer do?

Recall: The Structure of a Compiler Decaf program (stream of characters) scanner stream of tokens parser Abstract Syntax Tree (AST) checker AST with annotations (types, declarations) code gen maybe x86 Ras Bodik, CS 164, Fall 2004

Recall: Lexical Analysis The input is just a sequence of characters. Example: if (i == j) z = 0; else z = 1; More accurately, the input is string: \tif (i == j)\n\t\tz = 0;\n\telse\n\t\tz = 1; Goal: find lexemes and map them to tokens: partition input string into substrings (called lexemes), and classify them according to their role (role = token) Ras Bodik, CS 164, Fall 2004

partitioned into these lexemes: mapped to a sequence of tokens Notes: Continued Lexer input: \tif(i==j)\n\t\tz = 0;\n\telse\n\t\tz = 1; partitioned into these lexemes: \t if ( i == j ) \n\t\t z = 0 ; \n\t else \n\t\t z = 1 ; mapped to a sequence of tokens IF, LPAR, ID(“i”), EQUALS, ID(“j”) … Notes: whitespace lexemes are dropped, not mapped to tokens is this the same fatal mistake as in FORTRAN? (see Lecture 1) some tokens have attributes: the lexeme and/or line number why do we need them? Ras Bodik, CS 164, Fall 2004

A token is a syntactic category What’s a Token? A token is a syntactic category In English: noun, verb, adjective, … In a programming language: Identifier, Integer, Keyword, Whitespace, … Parser relies on the token distinctions: identifiers are treated differently than keywords but all identifiers are treated the same, regardless of what lexeme created them Ras Bodik, CS 164, Fall 2004

What are lexemes? Webster: cs164: “items in the vocabulary of a language” cs164: same: items in the vocabulary of a language: numbers, keywords, identifiers, operators, etc. strings into which the input string is partitioned. Ras Bodik, CS 164, Fall 2004

How to build a scanner for Decaf?

Instead, we’ll build a lexer generator Writing the lexer Not by hand tedious, repetitious, error-prone, non-maintainable Instead, we’ll build a lexer generator once we have the generator, we’ll only describe the lexemes and their tokens … that is, provide Decaf’s lexical specification (the What) … and generate code that performs the partitioning generated code hides repeated code (the How) Ras Bodik, CS 164, Fall 2004

Code generator: key benefit The scanner generator allows the programmer to focus on: What the lexer should do, rather than How it should be done. what: declarative programming how: imperative programming Ras Bodik, CS 164, Fall 2004

Imperative scanner (in Java) Let’s first build the scanner in Java, by hand: to see how it is done, and where’s the repetitious code that we want to hide A simple scanner will do. Only four tokens: TOKEN Lexeme ID a sequence of one or more letters or digits starting with a letter EQUALS “==“ PLUS “+” TIMES “*” Ras Bodik, CS 164, Fall 2004

Imperative scanner c=nextChar(); if (c == ‘=‘) { c=nextChar(); if (c == ‘=‘) {return EQUALS;}} if (c == ‘+’) { return PLUS; } if (c == ‘*’) { return TIMES; } if (c is a letter) { c=NextChar(); while (c is a letter or digit) { c=NextChar(); } undoNextChar(c); return ID; } this code contains a real bug (i.e., not introduced intentionally) by generating code, we wish to avoid such “mechanical” bugs suitable for simple languages few simple tokens (no C-like comments) no need to add, modify tokens token interactions may hurt later (like strings in Java) we’ll add tokens in PA5 performance important (rarely so) also, there is java tokenizer Ras Bodik, CS 164, Fall 2004

Imperative scanner You could write your entire scanner in this style and for small scanners this style is appropriate This code looks simple and clean, but try to add tokens that start with the same string: “if” and “iffy” C-style comments: /* anything here /* nested comments */ */ string literals with escape sequences: “…\t … \”…” error handling, e.g., badly formed strings (see PA2) Look at StreamTokenizer.nextToken() for an example of real imperative scanner in Eclipse, type Ctrl+Shift+T. Enter StreamTokenizer. Press F4. In the Hierarchy view, select method nextToken. Ras Bodik, CS 164, Fall 2004

What is the need for undoNextChar()? Maximal munch rule What is the need for undoNextChar()? it performs look-ahead, to determine whether the ID lexeme can be grown further This is an example of maximal much rule: this rule followed by all scanners the rule: the input character stream is partitioned into lexemes that are as large as possible Ex.: in Java, “iffy” is not partitioned into “if” (the IF keyword) and “fy” (ID), but into “iffy” (ID) Ras Bodik, CS 164, Fall 2004

Imperative Lexer: what vs. how c=nextChar(); if (c == ‘=‘) { c=nextChar(); if (c == ‘=‘) {return EQUALS;}} if (c == ‘+’) { return PLUS; } if (c == ‘*’) { return TIMES; } if (c is a letter) { c=NextChar(); while (c is a letter or digit) { c=NextChar(); } undoNextChar(c); return ID; }  little logic, much plumbing Ras Bodik, CS 164, Fall 2004

Identifying the plumbing (the how) c=nextChar(); if (c == ‘=‘) { c=nextChar(); if (c == ‘=‘) {return EQUALS;}} if (c == ‘+’) { return PLUS; } if (c == ‘*’) { return TIMES; } if (c is a letter) { c=NextChar(); while (c is a letter or digit) { c=NextChar(); } undoNextChar(c); return ID; }  characters read always the same way Ras Bodik, CS 164, Fall 2004

Identifying the plumbing (the how) c=nextChar(); if (c == ‘=‘) { c=nextChar(); if (c == ‘=‘) {return EQUALS;}} if (c == ‘+’) { return PLUS; } if (c == ‘*’) { return TIMES; } if (c is a letter) { c=NextChar(); while (c is a letter or digit) { c=NextChar(); } undoNextChar(c); return ID; }  tokens are always return-ed Ras Bodik, CS 164, Fall 2004

Identifying the plumbing (the how) c=nextChar(); if (c == ‘=‘) { c=nextChar(); if (c == ‘=‘) {return EQUALS;}} if (c == ‘+’) { return PLUS; } if (c == ‘*’) { return TIMES; } if (c is a letter) { c=NextChar(); while (c is a letter or digit) { c=NextChar(); } undoNextChar(c); return ID; }  the lookahead is explicit Ras Bodik, CS 164, Fall 2004

Identifying the plumbing (the how) c=nextChar(); if (c == ‘=‘) { c=nextChar(); if (c == ‘=‘) {return EQUALS;}} if (c == ‘+’) { return PLUS; } if (c == ‘*’) { return TIMES; } if (c is a letter) { c=NextChar(); while (c is a letter or digit) { c=NextChar(); } undoNextChar(c); return ID; }  must build decision tree out of nested if’s (yuck!) Ras Bodik, CS 164, Fall 2004

Can we hide the plumbing? That is, can we avoid having to spell out the details of calls to the input method? write the return in front of tokens? code the look-ahead code explicitly? use if’s and while’s to indicate the decision tree? In short, can we make the code look like the specification table? TOKEN Lexeme ID a sequence of one or more letters or digits starting with a letter EQUALS “==“ PLUS “+” TIMES “*” Ras Bodik, CS 164, Fall 2004

Separate out the how (plumbing) The code actually follows a simple pattern: read next character and compare it with some predetermined character if there is a match, jump to a different line of code repeat this until you return a token. Is there a programming language that can encode this concisely? yes, finite automata! read, compare read, compare return a token line of code line of code Ras Bodik, CS 164, Fall 2004

Separate out the what c=nextChar(); if (c == ‘=‘) { c=nextChar(); if (c == ‘=‘) {return EQUALS;}} if (c == ‘+’) { return PLUS; } if (c == ‘*’) { return TIMES; } if (c is a letter) { c=NextChar(); while (c is a letter or digit) { c=NextChar(); } undoNextChar(c); return ID; } Ras Bodik, CS 164, Fall 2004

Part 1: declarative (the what) A declarative scanner Part 1: declarative (the what) describe each token as a finite automaton must be supplied for each scanner, of course (it specifies the lexical properties of the input language) Part 2: imperative (the how) connect these automata into a scanner automaton common to all scanners (like a library) responsible for the mechanics of scanning show a DFA-based lexer for two simple tokens [this is going from recognizer to a lexer] identify logic, plumbing, link them to imperative lexer where do actions come from, why are token definitions clearer (less distributed across code) why they are now easy to separate show what’s run-time (plumbing) and what will be generated this lexer could be written by hand it’s already modifiable [can show an example, relate to complication with imperative lexer?] gaining some advantages already (which ones?) Ras Bodik, CS 164, Fall 2004

DFAs

Deterministic finite automata (DFA) We’ll use DFA’s as recognizers: given an input string, the DFA will recognize whether the string is a valid lexeme “recognize” means answer true or false Example: Is “xyz” an identifier? The DFA will says yes. DFA’s alone insufficient to build a scanner: DFA’s recognize if a string is, say, an identifier but alone can’t partition the input program into lexemes so we’ll need to use them in a special way Ras Bodik, CS 164, Fall 2004

Deterministic Finite Automata Example: Decaf Identifiers sequences of one or more letters or underscores or digits, starting with a letter or underscore: letter | ‘_’ | digit letter | ‘_’ S A Ras Bodik, CS 164, Fall 2004

Example: Integer Literals DFA that accepts integer literals with an optional + or - sign: digit B digit digit + S A - Ras Bodik, CS 164, Fall 2004

And another (more abstract) example Alphabet {0,1} What strings does this recognize? 1 1 Ras Bodik, CS 164, Fall 2004

Finite-Automata State Graphs The start state A final state Nodes are states. Edges (arrows) are transitions. Each edge labeled with a single character. a single edge labeled "letter" stands for 52 edges labeled 'a', 'b', ..., 'z', 'A', ..., 'Z'. the label "letter,digit" stands for 62 edges labeled 'a',...'Z','0',...'9'. S is the start state every FSM has exactly one (labeled with "S"), A is a final state drawn using a double circle, and non-final states are drawn using single circles. A FSM may have more than one final state. a A transition Ras Bodik, CS 164, Fall 2004

In state s1 on input “a” go to state s2 Finite Automata Transition s1 a s2 Is read In state s1 on input “a” go to state s2 If end of input If in accepting state => accept Otherwise => reject If no transition possible (got stuck) => reject Ras Bodik, CS 164, Fall 2004

A finite automaton is a 5-tuple (, Q, , q, F) where: Formal Definition A finite automaton is a 5-tuple (, Q, , q, F) where: An input alphabet  A set of states Q A start state q A set of final states F  Q  is the state transition function: Q x   Q (i.e., encodes transitions state input state) 5-tuple means “consists of” it is a function that takes two arguments -- a state and a character -- and returns a state Ras Bodik, CS 164, Fall 2004

Language defined by DFA The language defined by a DFA is the set of strings accepted by the DFA. in the language of the identifier DFA shown above: x, tmp2, XyZzy, position27. not in the language of the identifier DFA shown above: 123, a?, 13apples. Question 1: Write an automaton that accepts Java identifiers one or more letters, digits, or underscores, starting with a letter or an underscore. Question 2: Write a finite-state machine that accepts only Java identifiers that do not end with an underscore Ras Bodik, CS 164, Fall 2004

the declarative scanner

Part 1: create a DFA for each token ID: letter letter | digit “+” PLUS: “*” TIMES: “=” “=” EQUALS: Ras Bodik, CS 164, Fall 2004

Part 2: allow actions on DFA transitions the action can be one of "put back one character" or "return token XYZ", such DFA is called a transducer it translates input string to an output string Ras Bodik, CS 164, Fall 2004

Step 2: example of extending a DFA The DFA recognizing identifiers is modified to: Look-ahead is added for lexemes of variable length in our case, only ID needs lookahead A note on action “return ID” resets the scanner back into start state S (recall that scanner is called by parser; each time, one token is returned) letter | digit any char except letter or digit letter S action: put back 1 char return ID Ras Bodik, CS 164, Fall 2004

Step 2: Combine the extended DFA’s The algorithm: merge start nodes of the DFA’s. F3 return PLUS “+” letter | digit put back 1 char; return ID F4 “*” letter S return TIMES any char except letter or digit ID F3 “=” return EQUALS “=” F1 Ras Bodik, CS 164, Fall 2004

Towards a realistic scanner Building a scanner out of DFA’s (as shown) is simple but doesn’t quite work Consider a fifth token type, for the assignment operator TOKEN Lexeme ID a sequence of one or more letters or digits starting with a letter EQUALS “==“ ASSIGN “=” PLUS “+” TIMES “*” Ras Bodik, CS 164, Fall 2004

Correct scanner F3 F4 S ID F3 F1 return PLUS “+” letter | digit put back 1 char; return ID “*” letter S return TIMES any char except letter or digit ID F3 “=” any char except “=” put back 1 char; return ASSIGN return EQUALS “=” F1 Ras Bodik, CS 164, Fall 2004

But above algorithm produces F3 return PLUS “+” letter | digit F4 put back 1 char; return ID “*” letter S return TIMES any char except letter or digit “=” F3 “=” return ASSIGN return EQUALS “=” Bad: not a DFA should perform look-ahead Fix it with NFA’s! F1 Ras Bodik, CS 164, Fall 2004

NFAs

Deterministic vs. Nondeterministic Automata Deterministic Finite Automata (DFA) one transition per input characater per state no -moves Nondeterministic Finite Automata (NFA) allows multiple outgoing transitions for one input can have -moves Both: finite automata have finite memory Need only to encode the current state NFA’s can be in multiple states at once (stay tuned) Ras Bodik, CS 164, Fall 2004

The operation of the automaton is not completely defined by the input A simple NFA example Alphabet: { 0, 1 } The operation of the automaton is not completely defined by the input On input “11” the automaton could be in either state 1 Ras Bodik, CS 164, Fall 2004

Another kind of transition: -moves  Epsilon Moves Another kind of transition: -moves  A B Machine can move from state A to state B without reading input Ras Bodik, CS 164, Fall 2004

Execution of Finite Automata A DFA can take only one path through the state graph Completely determined by input NFAs can choose whether to make -moves which of multiple transitions for a single input to take so we think of an NFA as being in one of multiple states (see next example) Ras Bodik, CS 164, Fall 2004

An NFA can get into multiple states Acceptance of NFAs An NFA can get into multiple states 1 Input: 1 1 Rule: NFA accepts if it can get in a final state Ras Bodik, CS 164, Fall 2004

NFA vs. DFA (1) NFA’s and DFA’s are equally powerful each NFA can be translated into a corresponding DFA (one that recognizes same strings) formally, NFAs and DFAs recognize the same set of languages (called regular languages) But NFA’s are more convenient they allow easy merges of automata, which helps in scanner construction And DFAs are easier to implement There are no choices to consider in PA2, you will use NFA’s we’ll give you NFA recognizer code Ras Bodik, CS 164, Fall 2004

For a given language the NFA can be simpler than the DFA NFA vs. DFA (2) For a given language the NFA can be simpler than the DFA 1 NFA 1 DFA DFA can be exponentially larger than NFA Ras Bodik, CS 164, Fall 2004

full declarative scanner

With NFA’s, we can develop the scanner: The algorithm With NFA’s, we can develop the scanner: Construct an automaton for each lexeme (as before) Merge them into the scanner automaton (as before) Don’t extend the automata as before, instead whenever you reach a final state: remember position in input (so that you can undo reads) keep reading more characters, moving to other states whenever you get stuck (cannot make a move on next char): return to the last final state (i.e., undo the reads) return the token associated with this final state Ras Bodik, CS 164, Fall 2004

Notice that reading past final state implements look-ahead Notes Notice that reading past final state implements look-ahead This look-ahead is unbounded can return any number of characters Can you think of two lexemes that will require the scanner to return a large amount of characters? “large” means we can write an input that will make the necessary look-ahead arbitrarily large. Ras Bodik, CS 164, Fall 2004

Practical concerns Ambiguity Discarding whitespace Error inputs problem: scanner may reach multiple final states at once ex.: “if” matches both ID and IF (the keyword) solution: prioritize tokens (IF wins over ID) Discarding whitespace solution: final state for white-space lexeme is special: don’t return a token, but jump to (common) start state Error inputs problem: discard illegal lexemes and print an error message simple solution (discard char by char): add a lexeme that matches any character, giving it lowest priority; it will match when no other will Ras Bodik, CS 164, Fall 2004

We have a full declarative scanner imperative part, stored in the library (the third step of the algorithm) this is the run-time: it performs look-ahead, moves, input matching, returning tokens declarative part think of it as configuring the run-time of the scanner configuration done by specifying an automaton for each token Ras Bodik, CS 164, Fall 2004

Programming the declarative scanner configuring the run-time can be done by hand This code creates a DFA for the EQUALS token: Node start = new Node(), middle = new Node(); Node final = new FinalNode(EQUALS); start.addEdge(middle, ‘=‘); middle.addEdge(final, ‘=‘); this is an improvement over the imperative scanner more readable, maintainable, les error-prone but our goal is to avoid writing even this, we’ll write a code generator it will translate regular expressions (our textual program) into NFA’s Ras Bodik, CS 164, Fall 2004

regular expressions

Regular Expressions Automaton is a good “visual” aid but is not suitable as a specification (its textual description is too clumsy) regular expressions are a suitable specification a compact way to define a language that can be accepted by an automaton. used as the input to a scanner generator define each token, and also define white-space, comments, etc these do not correspond to tokens, but must be recognized and ignored. Ras Bodik, CS 164, Fall 2004

Example: Pascal identifier Lexical specification (in English): a letter, followed by zero or more letters or digits. Lexical specification (as a regular expression): letter . (letter | digit)* | means "or" . means "followed by" * means zero or more instances of () are used for grouping Often, the "followed by" dot is omitted, and just writing two things next to each other means that one follows the other. For example: letter (letter | digit)* Ras Bodik, CS 164, Fall 2004

Operands of a regular expression Operands are same as labels on the edges of an FSM single characters, or the special character  (the empty string) "letter" is a shorthand for a | b | c | ... | z | A | ... | Z "digit“ is a shorthand for 0 | 1 | … | 9 sometimes we put the characters in quotes necessary when denoting | . * Ras Bodik, CS 164, Fall 2004

Precedence of | . * operators. Regular Expression Operator Analogous Arithmetic Operator Precedence | plus lowest . times middle * exponentiation highest does not define the same language as the expression given above. Since the dot operator has higher precedence than the | operator (and the * operator has the highest precedence of all), this expression is the same as: (letter.letter) | (digit*) and it means "either two letters, or zero or more digits". Consider regular expressions: letter.letter | digit* letter.(letter | digit)* Ras Bodik, CS 164, Fall 2004

More examples Describe (in English) the language defined by each of the following regular expressions: letter (letter | digit*) digit digit* "." digit digit* Ras Bodik, CS 164, Fall 2004

Example: Integer Literals An integer literal with an optional sign can be defined in English as: “(nothing or + or -) followed by one or more digits” The corresponding regular expression is: (+|-|).(digit.digit*) A new convenient operator ‘+’ same precedence as ‘*’ digit.digit* is the same as digit+ which means "one or more digits" so “integer literals with optional sign is” can be written as (+|-|epsilon).digit+ Ras Bodik, CS 164, Fall 2004

Language Defined by a Regular Expression Recall: language = set of strings Language defined by an automaton the set of strings accepted by the automaton Launguage defined by a regular expression the set of strings that match the expression. Regular Exp. Corresponding Set of Strings  {""} a {"a"} a.b.c {"abc"} a | b | c {"a", "b", "c"} (a | b | c)* {"", "a", "b", "c", "aa", "ab", ..., "bccabb" ...} Ras Bodik, CS 164, Fall 2004

Translating RE to NFA’s key idea: find hierarchy in the regular expression find nested RE’s define translation for individual operation compose the translations of nested RE’s RE’s as AST translation via AST traversal Ras Bodik, CS 164, Fall 2004

Regular Expressions to NFA (1) For each kind of rexp, define an NFA Notation: NFA for rexp M M For   For input a a Ras Bodik, CS 164, Fall 2004

Regular Expressions to NFA (2) For A . B A B  For A | B A B  Ras Bodik, CS 164, Fall 2004

Regular Expressions to NFA (3) For A*  A    Ras Bodik, CS 164, Fall 2004

Example of RegExp -> NFA conversion Consider the regular expression (1|0)*1 The NFA is  1   C E 1 A B  G H  I J   D F   Ras Bodik, CS 164, Fall 2004

Automatically translating RE’s to NFA’s we need an AST that represents the RE the translation is then a bottom up traversal of the AST like in Java pretty printing options: properly, you write AST designed for RE operators and a parser from RE syntax to the RE AST in PA2, we’ll reuse Java syntax and Java AST overload Java operators it’s a hack, but it allows us to implement the RE-NFA translation quickly Ras Bodik, CS 164, Fall 2004

odds and ends

Feel free to implement these two in PA2 Practical issues PA2, we’ll use NFAs But DFAs are often faster because they can be implemented with tables Next few slides NFA to DFA conversion table implementation of DFA’s Feel free to implement these two in PA2 experiment with how much faster your scanner is than the NFA-based scanner Ras Bodik, CS 164, Fall 2004

Add a transition S a S’ to DFA iff NFA to DFA. The Trick Simulate the NFA Each state of DFA = a non-empty subset of states of the NFA Start state = the set of NFA states reachable through -moves from NFA start state Add a transition S a S’ to DFA iff S’ is the set of NFA states reachable from the states in S after seeing the input a considering -moves as well Ras Bodik, CS 164, Fall 2004

NFA -> DFA Example  1   1      1 1 1 C E A B G H I J D F G H I J   D F   FGABCDHI 1 ABCDHI 1 1 EJGABCDHI Ras Bodik, CS 164, Fall 2004

An NFA may be in many states at any time How many different states ? NFA to DFA. Remark An NFA may be in many states at any time How many different states ? If there are N states, the NFA must be in some subset of those N states How many non-empty subsets are there? 2N - 1 = finitely many Ras Bodik, CS 164, Fall 2004

A DFA can be implemented by a 2D table T Implementation A DFA can be implemented by a 2D table T One dimension is “states” Other dimension is “input symbols” For every transition Si a Sk define T[i,a] = k DFA “execution” If in state Si and input a, read T[i,a] = k and skip to state Sk Very efficient Ras Bodik, CS 164, Fall 2004

Table Implementation of a DFA T 1 S 1 1 U 1 S T U Ras Bodik, CS 164, Fall 2004

Implementation (Cont.) NFA -> DFA conversion is at the heart of tools such as flex or jlex But, DFAs can be huge In practice, flex-like tools trade off speed for space in the choice of NFA and DFA representations Ras Bodik, CS 164, Fall 2004

the end Add slides for exercises how the project differs from the lecture what students need to know for the exam agenda for sections