- 快召唤伙伴们来围观吧
- 微博 QQ QQ空间 贴吧
- 文档嵌入链接
- 复制
- 微信扫一扫分享
- 已成功复制到剪贴板
编译器设计03
展开查看详情
1 .CS 153: Concepts of Compiler Design August 31 Class Meeting Department of Computer Science San Jose State University Fall 2017 Instructor: Ron Mak www.cs.sjsu.edu/~mak 1
2 .2 Pascal-Specific Front End Classes PascalParserTD is a subclass of Parser and implements the parse() and getErrorCount () methods for Pascal. TD for “ top down ” PascalScanner is a subclass of Scanner and implements the extractToken () method for Pascal. Strategy Design Pattern
3 .3 The Pascal Parser Class The initial version of method parse() does hardly anything, but it forces the scanner into action and serves our purpose of doing end-to-end testing . public void parse () throws Exception { Token token; long startTime = System.currentTimeMillis (); while (!((token = nextToken ()) instanceof EofToken )) {} // Send the parser summary message. float elapsedTime = ( System.currentTimeMillis () - startTime )/1000f; sendMessage (new Message( PARSER_SUMMARY , new Number[] { token.getLineNumber (), getErrorCount (), elapsedTime })); } What does this while loop do? frontend.pascal.PascalParserTD
4 .4 The Pascal Scanner Class The initial version of method extractToken () doesn ’ t do much either, other than create and return either a default token or the EOF token . protected Token extractToken () throws Exception { Token token; char currentChar = currentChar (); // Construct the next token. The current character determines the // token type. if ( currentChar == EOF) { token = new EofToken (source); } else { token = new Token(source); } return token; } Remember that the Scanner method nextToken () calls the abstract method extractToken () . Here, the Scanner subclass PascalScanner implements method extractToken () . frontend.pascal.PascalScanner
5 .5 The Token Class The Token class ’ s default extract() method extracts just one character from the source. This method will be overridden by the various token subclasses. It serves our purpose of doing end-to-end testing . protected void extract() throws Exception { text = Character.toString ( currentChar ()); value = null; nextChar (); // consume current character } frontend.Token
6 .6 The Token Class , cont’d A character (or a token) is “ consumed ” after it has been read and processed, and the next one is about to be read. If you forget to consume, you will loop forever on the same character or token.
7 .7 A Front End Factory Class A language-specific parser goes together with a scanner for the same language. But we don ’ t want the framework classes to be tied to a specific language. Framework classes should be language-independent. We use a factory class to create a matching parser-scanner pair . Factory Method Design Pattern
8 .8 A Front End Factory Class, cont ’ d Good: Arguments to the createParser () method enable it to create and return a parser bound to an appropriate scanner . Variable parser doesn ’ t have to know what kind of parser subclass the factory created. Once again, the idea is to maintain loose coupling . Parser parser = FrontendFactory.createParser ( … ); “ Coding to the interface. ”
9 .9 A Front End Factory Class, cont ’ d Good: Bad: Why is this bad? Now variable parser is tied to a specific language. Parser parser = FrontendFactory.createParser ( … ); PascalParserTD parser = new PascalParserTD ( … )
10 .10 A Front End Factory Class, cont ’ d public static Parser createParser (String language , String type , Source source ) throws Exception { if ( language.equalsIgnoreCase (" Pascal ") && type.equalsIgnoreCase (" top-down ")) { Scanner scanner = new PascalScanner (source); return new PascalParserTD (scanner); } else if (! language.equalsIgnoreCase ("Pascal")) { throw new Exception("Parser factory: Invalid language + language + "); } else { throw new Exception("Parser factory: Invalid type + type + "); } } frontend.FrontendFactory
11 .11 Initial Back End Subclasses The CodeGenerator and Executor subclasses will only be (do-nothing) stubs for now. Strategy Design Pattern
12 .12 The Code Generator Class All the process() method does for now is send the COMPILER_SUMMARY message. number of instructions generated (none for now) code generation time (nearly no time at all for now) public void process ( ICode iCode , SymTab symTab ) throws Exception { long startTime = System.currentTimeMillis (); float elapsedTime = ( System.currentTimeMillis () - startTime )/1000f; int instructionCount = 0; // Send the compiler summary message. sendMessage (new Message( COMPILER_SUMMARY , new Number[] { instructionCount , elapsedTime })); } backend.compiler.CodeGenerator
13 .13 The Executor Class All the process() method does for now is send the INTERPRETER_SUMMARY message. number of statements executed (none for now) number of runtime errors (none for now) execution time (nearly no time at all for now) public void process ( ICode iCode , SymTab symTab ) throws Exception { long startTime = System.currentTimeMillis (); float elapsedTime = ( System.currentTimeMillis () - startTime )/1000f; int executionCount = 0; int runtimeErrors = 0; // Send the interpreter summary message. sendMessage (new Message( INTERPRETER_SUMMARY , new Number[] { executionCount , runtimeErrors , elapsedTime })); } backend.interpreter.Executor
14 .14 A Back End Factory Class public static Backend createBackend (String operation ) throws Exception { if ( operation.equalsIgnoreCase (" compile ") { return new CodeGenerator (); } else if ( operation.equalsIgnoreCase (" execute ")) { return new Executor (); } else { throw new Exception("Backend factory: Invalid operation + operation + "); } } backend.BackendFactory
15 .15 End-to-End: Program Listings Here ’ s the heart of the main Pascal class ’ s constructor: source = new Source (new BufferedReader (new FileReader ( filePath ))); source.addMessageListener (new SourceMessageListener ()); parser = FrontendFactory.createParser ("Pascal", "top-down", source); parser.addMessageListener (new ParserMessageListener ()); backend = BackendFactory.createBackend (operation); backend.addMessageListener (new BackendMessageListener ()); parser.parse (); iCode = parser.getICode (); symTab = parser.getSymTab (); backend.process ( iCode , symTab ); source.close (); The front end parser creates the intermediate code and the symbol table of the intermediate tier. The back end processes the intermediate code and the symbol table .
16 .16 Listening to Messages Class Pascal has inner classes that implement the MessageListener interface. private static final String SOURCE_LINE_FORMAT = "%03d %s"; private class SourceMessageListener implements MessageListener { public void messageReceived (Message message) { MessageType type = message.getType (); Object body[] = (Object []) message.getBody (); switch (type) { case SOURCE_LINE : { int lineNumber = (Integer) body[0]; String lineText = (String) body[1]; System.out.println ( String.format (SOURCE_LINE_FORMAT, lineNumber , lineText )); break; } } } } Demo
17 .17 Is it Really Worth All this Trouble? Major software engineering challenges: Managing change . Managing complexity . To help manage change , use the open-closed principle . Close the code for modification. Open the code for extension. Closed: The language-independent framework classes. Open : The language-specific subclasses.
18 .18 Is it Really Worth All this Trouble? cont’d Techniques to help manage complexity : Partitioning Loose coupling Incremental development Always build upon working code. Good object-oriented design with design patterns.
19 .19 Source Files from the Book Download the Java source code from each chapter of the book: http://www.cs.sjsu.edu/~mak/CS153/sources/ You will not survive this course if you use a simple text editor like Notepad to view and edit the Java code. The complete Pascal interpreter in Chapter 12 contains 127 classes and interfaces .
20 .20 Integrated Development Environment (IDE) You can use either Eclipse or NetBeans . Eclipse is preferred because later you will be able to use an ANTLR plug-in. Learn how to create projects, edit source files, single-step execution, set breakpoints, examine variables, read stack dumps, etc.
21 .21 Pascal-Specific Front End Classes
22 .22 The Payoff Now that we have … Source language-independent framework classes Pascal-specific subclasses Mostly just placeholders for now An end-to-end test (the program listing generator) … we can work on the individual components Without worrying (too much) about breaking the rest of the code.
23 .23 Front End Framework Classes
24 .24 Pascal-Specific Subclasses
25 .25 Pascal-Specific Token Classes Each class PascalWordToken , PascalNumberToken , PascalStringToken , PascalSpecial-SymbolToken , and PascalErrorToken is is a subclass of class PascalToken . PascalToken is a subclass of class Token . Each Pascal token subclass overrides the default extract() method of class Token . The default method could only create single-character tokens. Loosely coupled. Highly cohesive.
26 .PascalTokenType Details 26
27 .PascalTokenType Details 26
28 .28 PascalTokenType, cont ’ d The static set RESERVED_WORDS contains all of Pascal ’ s reserved word strings in lower case: " and " , " array " , " begin " , etc. We can test whether a token is a reserved word: // Set of lower-cased Pascal reserved word text strings. public static HashSet <String> RESERVED_WORDS = new HashSet <String> (); static { PascalTokenType values[] = PascalTokenType.values (); for ( int i = AND .ordinal (); i <= WITH .ordinal (); ++ i ) { RESERVED_WORDS.add (values[ i ]. getText (). toLowerCase ()); } } if ( RESERVED_WORDS.contains ( text.toLowerCase ())) … frontend.pascal.PascalTokenType frontend.pascal.tokens.PascalWordToken
29 .28 PascalTokenType, cont ’ d The static set RESERVED_WORDS contains all of Pascal ’ s reserved word strings in lower case: " and " , " array " , " begin " , etc. We can test whether a token is a reserved word: // Set of lower-cased Pascal reserved word text strings. public static HashSet <String> RESERVED_WORDS = new HashSet <String> (); static { PascalTokenType values[] = PascalTokenType.values (); for ( int i = AND .ordinal (); i <= WITH .ordinal (); ++ i ) { RESERVED_WORDS.add (values[ i ]. getText (). toLowerCase ()); } } if ( RESERVED_WORDS.contains ( text.toLowerCase ())) … frontend.pascal.PascalTokenType frontend.pascal.tokens.PascalWordToken