PL/0编译器(java version) – Scanner.java
1: package compiler;
2:
3: import java.io.BufferedReader;
4: import java.io.FileNotFoundException;
5: import java.io.FileReader;
6: import java.util.Arrays;
7:
8: public class Scanner {
9:
10: public int lineCnt=0;
11: private char curCh = ' ';
12: private String line;
13: public int lineLength = 0;
14: public int chCount = 0;
15: private int[] ssym;
16: private BufferedReader in;
17:
18: public Scanner(String filePath) {
19: try {
20: in = new BufferedReader(new FileReader(filePath));
21: } catch (FileNotFoundException ex) {
22: ex.printStackTrace();
23: System.out.println("***File not exist!***");
24: }
25: //设置单字符
26: ssym = new int[256];
27: Arrays.fill(ssym, Symbol.nul);
28: ssym['+'] = Symbol.plus;
29: ssym['-'] = Symbol.minus;
30: ssym['*'] = Symbol.mul;
31: ssym['/'] = Symbol.div;
32: ssym['('] = Symbol.lparen;
33: ssym[')'] = Symbol.rparen;
34: ssym['='] = Symbol.eql;
35: ssym[','] = Symbol.comma;
36: ssym['.'] = Symbol.peroid;
37: ssym[';'] = Symbol.semicolon;
38:
39: }
40:
41: //读取一个字符,为减少磁盘I/O次数,每次读取一行
42: void getch() {
43: if (chCount == lineLength) {
44: try {//如果读到行末尾,就重新读入一行
45: String tmp="";
46: while (tmp.equals("")) {
47: tmp=in.readLine().trim()+' '; //除去空行
48: }
49: line=tmp;
50: lineCnt++;
51: } catch (Exception e) {
52: // throw new Error("***program imcomplete!***");
53: e.printStackTrace();
54: System.out.println("***reading character meet with error!***");
55: }
56: lineLength = line.length();
57: chCount = 0;
58: System.out.println(line);
59: }
60: curCh = line.charAt(chCount++);
61: }
62:
63: //词法分析,获取一个词法分析符号,是词法分析器的重点
64: public Symbol getsym() {
65: Symbol sym;
66: while (curCh == ' ') {
67: getch();
68: }
69: if ((curCh >= 'a' && curCh <= 'z')||(curCh >= 'A' && curCh <= 'Z')) {
70: sym = matchKeywordOrIdentifier(); //关键字或者一般标识符
71: } else if (curCh >= '0' && curCh <= '9') {
72: sym = matchNumber(); //数字
73: } else {
74: sym = matchOperator(); //操作符
75: }
76: return sym;
77: }
78:
79: private Symbol matchKeywordOrIdentifier() {
80: StringBuffer sb = new StringBuffer();
81: do{
82: sb.append(curCh);
83: getch();
84: }while((curCh >= 'a' && curCh <= 'z')||(curCh>='A'&&curCh<='Z') || (curCh >= '0' && curCh <= '9'));
85:
86: String token = sb.toString();
87: int index = Arrays.binarySearch(Symbol.word, token); //搜索是不是保留字
88: Symbol sym = null;
89: if (index < 0) {
90: sym = new Symbol(Symbol.ident); //一般标识符
91: sym.id = token;
92: } else {
93: sym = new Symbol(Symbol.wsym[index]); //保留字对应的符号值0-31
94: }
95: return sym;
96: }
97:
98: private Symbol matchNumber() {
99: //统计数字位数
100: Symbol sym = new Symbol(Symbol.number);
101: do {
102: sym.num = 10 * sym.num + curCh - '0'; // 获取数字的值
103: getch();
104: } while (curCh >= '0' && curCh <= '9'); //!!!
105:
106: return sym;
107: }
108:
109: private Symbol matchOperator() {
110: Symbol sym = null;
111: switch (curCh) {
112: case ':': // 赋值符号
113: getch();
114: if (curCh == '=') {
115: sym = new Symbol(Symbol.becomes);
116: getch();
117: } else {
118: sym = new Symbol(Symbol.nul); //不能识别的符号
119: }
120: break;
121: case '<': //小于或者小于等于
122: getch();
123: if (curCh == '=') {
124: sym = new Symbol(Symbol.leq); //是<=
125: getch();
126: } else if (curCh == '>') {
127: sym = new Symbol(Symbol.neq); //是<>
128: getch();
129: } else {
130: sym = new Symbol(Symbol.lss); //是<
131: }
132: break;
133: case '>': //大于或者大于等于
134: getch();
135: if (curCh == '=') {
136: sym = new Symbol(Symbol.geq); //大于等于
137: getch();
138: } else {
139: sym = new Symbol(Symbol.gtr); //大于
140: }
141: break;
142: default:
143: sym = new Symbol(ssym[curCh]);
144: if (sym.symtype != Symbol.peroid) {
145: getch();
146: }
147: }
148: return sym;
149: }
150: }
.csharpcode, .csharpcode pre
{
font-size: small;
color: black;
font-family: consolas, "Courier New", courier, monospace;
background-color: #ffffff;
/*white-space: pre;*/
}
.csharpcode pre { margin: 0em; }
.csharpcode .rem { color: #008000; }
.csharpcode .kwrd { color: #0000ff; }
.csharpcode .str { color: #006080; }
.csharpcode .op { color: #0000c0; }
.csharpcode .preproc { color: #cc6633; }
.csharpcode .asp { background-color: #ffff00; }
.csharpcode .html { color: #800000; }
.csharpcode .attr { color: #ff0000; }
.csharpcode .alt
{
background-color: #f4f4f4;
width: 100%;
margin: 0em;
}
.csharpcode .lnum { color: #606060; }
PL/0编译器(java version) – Scanner.java的更多相关文章
- PL/0编译器(java version) - MainFrame.java
1: /* 2: * To change this license header, choose License Headers in Project Properties. 3: * To chan ...
- PL/0编译器实践---后记
花了几天时间,把清华版的<编译原理>一书中的PL/0编译器实践了一遍.颇有收获,记录如下: 理解代码的技巧,如何理解一份代码,比如这个程序,其逻辑相对于一般程序就比较复杂了,如何翻译,虚拟 ...
- PL/0编译器(java version)–Praser.java
1: package compiler; 2: 3: import java.io.IOException; 4: import java.util.BitSet; 5: 6: /** 7: ...
- PL/0编译器(java version)–PL0.java
1: package compiler; 2: 3: import java.io.BufferedWriter; 4: import java.io.FileWriter; 5: 6: /* ...
- PL/0编译器(java version) – SymbolTable.java
1: package compiler; 2: //竟然没有对符号表检查大小,会溢出的. 3: 4: import java.io.IOException; 5: 6: public clas ...
- PL/0编译器(java version) - Interpreter.java
1: package compiler; 2: 3: import java.io.BufferedReader; 4: import java.io.BufferedWriter; 5: imp ...
- PL/0编译器(java version) - Err.java
1: package compiler; 2: 3: import java.io.BufferedWriter; 4: 5: public class Err { 6: 7: publi ...
- PL/0编译器(java version) – Symbol.java
1: package compiler; 2: 3: /** 4: * 采用全局变量sym来存储符号码,并用全局变量id和num来传递语义值 5: * 6: * @author jiangnan ...
- PL/0编译器(java version)–Pcode.java
1: package compiler; 2: 3: /** 4: * //虚拟机指令 5: * 6: * @author jiangnan 7: * 8: */ 9: public class ...
随机推荐
- Scala入门详解
object作为Scala中的一个关键字,相当于Java中的public static class这样的一个修饰符,也就说object中的成员都是静态的! 所以我们在这个例子中的main方法是静态的, ...
- HTML5+JS 《五子飞》游戏实现(七)游戏试玩
前面第一至第六章我们已经把<五子飞>游戏的基本工作都已经讲得差不多了,这一章主要是把所有的代码分享给大家,然后小伙伴们也可以玩一玩. 至于人机对战的我们放到后面讲进行分析. 试玩地址:ht ...
- IL指令大全(转)
名称 说明 Add 将两个值相加并将结果推送到计算堆栈上. Add.Ovf 将两个整数相加,执行溢出检查,并且将结果推送到计算堆栈上. Add.Ovf.Un 将两个无符号整数值相加,执行溢出检查,并且 ...
- Android闹钟开发与展示Demo
前言: 看过了不少安卓闹钟开发的例子,都是点到为止,都不完整,这次整一个看看. 一.闹钟的设置不需要数据库,但是展示闹钟列表的时候需要,所以需要数据库: public class MySQLiteOp ...
- Sqlsever
Sqlsever: 获取主键当前最大值: select ident_current('tablename');
- 利用ZTree链接数据库实现 [权限管理]
最近想研究权限管理,看群里有人发了ZTrees模板,我看了下,觉得笔easyUI操作起来更灵活些,于是就开始研究了. 刚开始从网上找了找了个Demo,当然这个并没有实现权限啥的,但实现了前台调用Aja ...
- EditPlus v4.5 简体中文
优秀的代码编辑器! 下载地址: EditPlus v4.00 build 465 简体中文汉化增强版 http://yunpan.cn/cVCSIZsKK7VFF 访问fe58 http://p ...
- python表达式操作符【学习python必知必会】
运算符 描述 实例 yield x 生成器函数发送协议 lambda args: expression 生成匿名函数 x if y else z 三元选择表达式(c系列有的 python也要有 ...
- 一键系统优化15项脚本,适用于Centos6.x
#!/bin/sh ################################################ #Author:nulige # qqinfo:1034611705 # Date ...
- Bete冲刺第二阶段
Bete冲刺第二阶段 今日工作: web: 修复了a版本中接口数据返回错误的问题.通知对数据表新增了几个字段.并且新增了一个通知查询的接口. ios: 修正网络连接bug 招募功能界面完成但是跳转问题 ...