130242014076+陈旭+第2次实验

Posted 软件工程2班-陈旭

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了130242014076+陈旭+第2次实验相关的知识,希望对你有一定的参考价值。

软件体系结构的第二次实验(解释器风格与管道过滤器风格)

一、实验目的

  1.熟悉体系结构的风格的概念

  2.理解和应用管道过滤器型的风格。

  3、理解解释器的原理

  4、理解编译器模型

二、实验环境

  硬件: 

  软件:Python或任何一种自己喜欢的语言

三、实验内容

  1、实现“四则运算”的简易翻译器。

  结果要求:

    1)实现加减乘除四则运算,允许同时又多个操作数,如:2+3*5-6 结果是11

    2)被操作数为整数,整数可以有多位

    3)处理空格

    4)输入错误显示错误提示,并返回命令状态“CALC”

     

      图1    实验结果示例

  加强练习:

    1、有能力的同学,可以尝试实现赋值语句,例如x=2+3*5-6,返回x=11。(注意:要实现解释器的功能,而不是只是显示)

    2、尝试实现自增和自减符号,例如x++ 

    3、采用管道-过滤器(Pipes and Filters)风格实现解释器

 

                        图2  管道-过滤器风格

 

 

                     图 3  编译器模型示意图

  本实验,实现的是词法分析和语法分析两个部分。

四、实验步骤:

 代码如下:

  1 package com.brainstrong.uamis.util;
  2 
  3 import java.io.InputStream;
  4 import java.util.ArrayList;
  5 import java.util.List;
  6 import java.util.Scanner;
  7 import java.util.StringTokenizer;
  8 
  9 /**
 10  * JAVA实现四则运算解释器
 11  *
 12  * @author 旭旭
 13  * @create 2017-10-27 19:43
 14  **/
 15 public class Calc {
 16 
 17     public static void main(String[] args) {
 18         Scanner scanner = new Scanner(System.in);
 19         String exp = scanner.nextLine();
 20         System.out.println(calc(exp));
 21     }
 22 
 23     /**
 24      *
 25      * @param exp 四则表达式
 26      * @return
 27      */
 28     public static double calc(String exp){
 29         //1.把表示负数的-号换成@号
 30         exp = negativeToAtChar(exp);
 31         //2.数字的分类
 32         List<Double> numbers = splitNumExp(exp);
 33         //3.运算符的分离
 34         List<Character> ops = splitOpfromExp(exp);
 35         //4.先乘除
 36         for (int i = 0; i < ops.size(); i++) {
 37             //判断,运算符是否是乘除
 38             char op = ops.get(i);
 39             //是,取出,运算
 40             if (op == \'*\' || op == \'/\') {
 41                 //取出来,运算
 42                 ops.remove(i);//后面的数据往前顺序移动
 43                 //运算
 44                 //从数字容器中取出对应运算符的两个数字
 45                 double d1 = numbers.remove(i);
 46                 double d2 = numbers.remove(i);
 47 
 48                 if (op == \'*\') {
 49                     d1 *= d2;
 50                 }else {
 51                     d1 /= d2;
 52                 }
 53 
 54                 //把运算结果放入数字容器中i的位置
 55                 numbers.add(i, d1);//原来i位置(包括)后面的数据依次往后顺移
 56                 i--;
 57             }
 58         }
 59         //5.后加减
 60         while (!ops.isEmpty() ) {
 61             char op = ops.remove(0);
 62             double d1 = numbers.remove(0);
 63             double d2 = numbers.remove(0);
 64             //运算
 65             if (op == \'+\') {
 66                 d1 += d2;
 67             } else {
 68                 d1 -= d2;
 69             }
 70             //把运算结果插入到数字容器中0的位置
 71             numbers.add(0, d1);
 72         }
 73         //6.容器中的第一个数据就是结果
 74         return numbers.get(0);
 75     }
 76 
 77     /**
 78      * 从表达式中分离表达式和运算符
 79      * @param exp
 80      * @return
 81      */
 82     private static List<Character> splitOpfromExp(String exp) {
 83         List<Character> ops = new ArrayList<Character>();
 84         StringTokenizer st = new StringTokenizer(exp, "1234567890.@");
 85         while (st.hasMoreTokens()) {
 86             char c = st.nextElement().toString().trim().charAt(0);
 87             ops.add(c);
 88         }
 89         return ops;
 90     }
 91 
 92     /**
 93      * 分离出数字
 94      * @param exp
 95      * @return
 96      */
 97     private static List<Double> splitNumExp(String exp) {
 98         List<Double> numbers = new ArrayList<Double>();
 99         StringTokenizer st = new StringTokenizer(exp, "+-*/");
100         while (st.hasMoreTokens()) {
101             String numStr = st.nextElement().toString().trim();
102             if (numStr.charAt(0) == \'@\') {
103                 numStr = "-" + numStr.substring(1);
104             }
105             numbers.add(Double.parseDouble(numStr));
106         }
107         return numbers;
108     }
109 
110     /**
111      * 把-号换成@号
112      * @param exp
113      * @return
114      */
115     private static String negativeToAtChar(String exp) {
116         for (int i = 0; i < exp.length(); i++) {
117             char c = exp.charAt(i);
118             if (c == \'-\') {
119                 //判断是否是负数
120                 if (i == 0) {
121                     //第一个位置肯定是负数
122                     exp = "@"+exp.substring(1);
123                 }else {
124                     //不是第一个位置
125                     //判断前一个位置是否是运算符
126                     char cprev = exp.charAt(i - 1);
127                     if (cprev == \'+\' || cprev == \'-\' || cprev == \'*\' || cprev == \'/\') {
128                         exp = exp.substring(0, i)+"@"+exp.substring(i+1);
129                     }
130                 }
131             }
132         }
133         return exp;
134     }
135 
136 }

 

以上是关于130242014076+陈旭+第2次实验的主要内容,如果未能解决你的问题,请参考以下文章

130242014076-“电商系统登录功能模块”需求分析与设计实验课小结

第九次作业

130242014057 周陈清 第2次实验

OS第2次实验报告:创建进程

操作系统第2次实验报告:创建进程

操作系统第2次实验报告:创建进程