刚刚学习编译原理,这是第二章的代码的改写,源码
http://www.blogjava.net/Files/zhaochengming/compiler%20section%202%20parse.rar
1.KeyWord.java 关键字对象类
package parse;
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public class KeyWord
{
private String lexptr;
private int token;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public KeyWord(String lexptr, int token)
{
this.lexptr = lexptr;
this.token = token;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public String getLexptr()
{
return lexptr;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void setLexptr(String lexptr)
{
this.lexptr = lexptr;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public int getToken()
{
return token;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void setToken(int token)
{
this.token = token;
}
}
2.SymTable.java 符号表对象类
package parse;
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
import java.util.ArrayList;
import java.util.List;
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
/** *//**
* 符号表
* @author Administrator
*
*/
data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public class SymTable
{
private static List<KeyWord> SYMTABLE = new ArrayList<KeyWord>();
// 符号表最大容量
private static final int SYMMAX = 999;
// 字符串最大长度
public static final int STRMAX = 100;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
private static final KeyWord KEYWORDS[] =
{
new KeyWord(null,0),
new KeyWord("DIV", Token.DIV),
new KeyWord("ADD", Token.ADD),
new KeyWord("SUB", Token.SUB),
new KeyWord("MOD", Token.MOD),
new KeyWord("MUL", Token.MUL),
};
// 初始化symtable
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
static
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
for(KeyWord keyword : KEYWORDS)
{
insert(keyword.getLexptr(), keyword.getToken());
}
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
/** *//**
* 在符号表中插入新的符号
* @param lexptr
* @param token
* @return
*/
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public static int insert(String lexptr, int token)
{
int lexptrlen = 0;
if(lexptr != null) lexptrlen = lexptr.length();
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(SYMTABLE.size() >= SYMMAX)
{
throw new IllegalStateException("符号表满。");
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(lexptrlen >= STRMAX)
{
throw new IllegalStateException("lexptr超过最大长度。");
}
SYMTABLE.add(new KeyWord(lexptr, token));
return SYMTABLE.size() - 1;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
/** *//**
* 返回lexptr在符号表中的下标
* @param lexptr
* @return
*/
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public static int lookup(String lexptr)
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
for (int i = 1; i < SYMTABLE.size(); i++)
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(SYMTABLE.get(i).getLexptr().equals(lexptr))
{
return i;
}
}
return 0;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public static KeyWord getKeyWord(int index)
{
return SYMTABLE.get(index);
}
}
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
3.Lex.java 词法分析对象类
package parse;
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public class Lex
{
private int last = -1;
private int pos = 0; // 当前位置
private ParseReader pr;
private String buf;
private int tokenVal;
private Parser parser;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public Lex(ParseReader pr, Parser parse)
{
this.pr = pr;
this.parser = parse;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public int lexan()
{
int t;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
while (true)
{
t = getChar();
if(t == ' ' || t == '\r');
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
else if(isDigit(t))
{
tokenVal = 0;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
while(isDigit(t))
{
tokenVal = tokenVal* 10 + (t - '0');
t = getChar();
}
ungetc(t);
return Token.NUM;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else if (isAlpha(t))
{
int p = -1, b = 0;
StringBuilder sb = new StringBuilder();
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
while (isDigit(t) || isAlpha(t))
{
sb.append((char)t);
t = getChar();
b++;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(b >= SymTable.STRMAX)
{
parser.error("词法分析错误");
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(t != Token.EOF)
{
ungetc(t);
}
p = SymTable.lookup(sb.toString());
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(p == 0)
{
p = SymTable.insert(sb.toString(), Token.ID);
}
tokenVal = p;
return SymTable.getKeyWord(tokenVal).getToken();
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else if (t == Token.EOF)
{
return Token.DONE;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else
{
tokenVal = Token.NULL;
return t;
}
}
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public int getTokenVal()
{
return tokenVal;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public int getChar()
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(last < 0)
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(buf != null && pos < buf.length())
{
return buf.charAt(pos++);
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(pr.isEof())
{
return Token.EOF;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else
{
buf = pr.readLine();
pos = 0;
return getChar();
}
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} else
{
int c = last;
last = -1;
return c;
}
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void ungetc(int c)
{
last = c;
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public boolean isDigit(int t)
{
return t >= '0' && t <= '9';
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public boolean isAlpha (int t)
{
return ((t >= 'A' && t <= 'Z') || (t >= 'a' && t <= 'z'));
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public int getLineNo()
{
return pr.getLineNo();
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public int getPos()
{
return pos;
}
}
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
4.Parser.java 语法分析对象类
package parse;
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
import java.io.FileNotFoundException;
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public class Parser
{
private StringBuilder temp = new StringBuilder();
private Lex lex;
int lookahead;
data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public Parser(ParseReader pr)
{
lex = new Lex(pr,this);
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void parse()
{
lookahead = lex.lexan();
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
while (lookahead != Token.DONE)
{
expr(); match(';');
}
out("后缀表达式为: "+temp.toString());
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
private void expr()
{
int t;
term();
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
while(true)
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
switch(lookahead)
{
case '+':
case '-':
case Token.ADD:
case Token.SUB:
t = lookahead;
match(lookahead); term(); emit(t, Token.NULL);
continue;
default:
return;
}
}
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
private void term()
{
int t;
factor();
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
while(true)
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
switch(lookahead)
{
case '*':
case '/':
case '%':
case Token.MOD:
case Token.DIV:
case Token.MUL:
t = lookahead;
match(lookahead); factor(); emit(t, Token.NULL);
continue;
default:
return;
}
}
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
private void factor()
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
switch(lookahead)
{
case '(':
match('('); expr(); match(')'); break;
case Token.NUM:
emit(Token.NUM, lex.getTokenVal()); match(Token.NUM); break;
case Token.ID:
emit(Token.ID, lex.getTokenVal()); match(Token.ID); break;
default:
error("语法分析错误");
}
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
private void match(int t)
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
if(lookahead == t)
{
lookahead = lex.lexan();
} else error("语法错误");
}
data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void emit(int t, int token)
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
switch(t)
{
case '+':
case '-':
case '*':
case '%':
case '/':
temp.append(" ").append((char)t).append(" ");
out(String.valueOf((char)t));break;
case Token.DIV:
temp.append(" DIV ");
out("DIV");break;
case Token.MOD:
temp.append(" MOD ");
out("MOD");break;
case Token.MUL:
temp.append(" MUL ");
out("MUL");break;
case Token.ADD:
temp.append(" ADD ");
out("ADD");break;
case Token.SUB:
temp.append(" SUB ");
out(" SUB ");break;
case Token.ID:
temp.append(SymTable.getKeyWord(token).getLexptr());
out(SymTable.getKeyWord(token).getLexptr());
break;
default:
temp.append(" ").append(token).append(" ");
out("token " + Token.name(t) + ", tokenval "+token);
}
}
data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void out(String msg)
{
System.out.println(msg);
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public static void main(String args[])
{
Parser pr;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
try
{
pr = new Parser(new ParseReaderRAF("c:/a.txt"));
pr.parse();
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} catch (FileNotFoundException e)
{
e.printStackTrace();
}
}
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public void error(String msg)
{
System.err.println("ERROR:[" + msg + "] LineNumber : " + lex.getLineNo() +", position :" + lex.getPos());
System.exit(0);
}
}
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
5.ParseReader 词法分析读字符流接口
package parse;
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public interface ParseReader
{
public String readLine();
public boolean isEof();
public int getLineNo();
}
6.ParseReaderRAF 词法分析读字符流RandomAccessFile对象
package parse;
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.RandomAccessFile;
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
data:image/s3,"s3://crabby-images/16507/1650758e64773369e558bf6a35239aa629f2eb9d" alt=""
public class ParseReaderRAF implements ParseReader
{
private RandomAccessFile raf;
private int lineno = 0;
private String buf;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public ParseReaderRAF(String filepath) throws FileNotFoundException
{
this.raf = new RandomAccessFile(new File(filepath), "r");
}
data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public int getLineNo()
{
return lineno;
}
data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public boolean isEof()
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
try
{
if(raf.getFilePointer() >= raf.length())
return true;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} catch (IOException e)
{
e.printStackTrace();
}
return false;
}
data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
public String readLine()
{
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
try
{
buf = raf.readLine();
//System.out.println(buf);
lineno++;
data:image/s3,"s3://crabby-images/4989c/4989c5aa5aeee035dc328aff8277d531300533ab" alt=""
} catch (Exception e)
{
e.printStackTrace();
}
return buf;
}
data:image/s3,"s3://crabby-images/a0398/a0398c5eaea7654f53f3ad01f4ef86b30b77f7b1" alt=""
}
data:image/s3,"s3://crabby-images/370e0/370e053b28c0d1e5a884270fad646284f2d183b3" alt=""
</script>