SDUTOJ 2099 - 小型Basic编译器问题(模拟)

小型Basic编译器问题

Time Limit: 1000MS  Memory Limit: 65536KB
Problem Description
编写一个TinyBasic语言的解释程序,对于任何一个给出的正确的TinyBasic语言的程序,你的程序能运行它并得到正确的结果。那么,怎样的TinyBasic的程序叫做正确的呢? 

(1)符合TinyBasic语言的语法规则; 
(2)程序执行时会产生一个或多个输出,可以中断(即程序不会进入无限循环状态)。 

TinyBasic语言的语法规则: 

(1)每一行的TinyBasic程序都是下面这样的形式(所有出现的字母均为大写)
[<空格>]<行号><空格><语句> 
其中,[<空格>]中可以有任意个空格,当然也可以没有;
<行号>中一定要有行号,从1开始,依次递增1; 
<空格>中至少有一个空格; 
<语句>应为下面的语句之一:
LET<空格><变量>=<表达式> 
PRINT<空格><变量> 
GOTO<空格><表达式> 
IF<空格><表达式> 
STOP 
(2)定义变量和表达式的规则为 
<变量>必须是单个的大写英文字母,存储一个整数值; 
<表达式>为   //(这个地方不知道为啥题目少了一块。。。请自行YY);
<常量> 范围在[-10000…10000]内的整数常量,比如0,-5,34  
 <变量>+<变量> 两个变量所代表的是有符号整数   
<变量>><变量> 大于号,真为1,假为0 
(3)表达式中和LET语句的=(等号)两边没有空格; 
(4)TinyBasic的程序最多只有100行。 

执行TinyBasic语言程序的规则: 

(1)从程序中的第1行开始执行; 
(2)程序中用到的所有变量的初始值均为0; 
(3)语句连续执行除非碰到IF或GOTO语句; 
(4)5种语句的定义 
LET 给变量赋值。若两个变量相加,相加的结果在[-10000…10000]之内。  
PRINT <变量名>=<值>的格式打印变量的值。左对齐,并单独占用一行;行中无任何多余空格。 
GOTO 跳到行号为<表达式>的值的一行。<表达式>不需要是一个常量;<表达式>的值是程序中的有效行号。  
IF 如果表达式的值非0继续执行下一行;如果表达式的值为0跳过下一行,执行下一行的下一行。在IF语句以下至少还应该有两条语句。
STOP 终止执行。TinyBasic程序一定会执行到STOP语句(如果你的解释程序是正确的话);TinyBasic程序可能包含一个以上的STOP语句;程序的最后一句不一定是STOP语句。
Input
输入数据只有一组,包含一个程序,没有多余的空行,每一行为一条语句,具体要求按上面的解释。你编写的程序要正确地运行该TinyBasic程序。
Output
输出程序的运行结果,文件头尾都不需要多余空行。
Example Input
1 LET A=10
2 LET I=0
3 LET X=I+I
4 LET T=1
5 LET X=X+T
6 PRINT X
7 LET T=1
8 LET I=I+T
9 IF A>I
10 GOTO 3
11 STOP
Example Output
X=1
X=3
X=5
X=7
X=9
X=11
X=13
X=15
X=17
X=19
Hint
 这个题么,,,就是一道纯粹的模拟题,最近刷POJ太累,所以就刷了个水题玩玩,哈哈哈。。。
 具体的模拟过程题目里面已经说的很清楚了,只需要注意几点:
     鉴于题目里面用到的变量全是只有一位的大写字母,所以用一个数组存下所有可能出现的变量,用ASCLL码作为下标;
     STOP作为终结语句,但并不一定在最后,所以要输入结束后再从第一句开始执行;
     所有变量初始值为0,先清空数组;
     其他的只需要按照平时编程的思路走即可,细节不用多说。
上代码:
#include <cstdio>
#include <cstring>
#include <cctype>
using namespace std;
struct node {                                      //结构体数组存指令类型和执行语句
    char sign[25],opera[25];
} p[128];
int value[128];                                    //存储所有变量,ASCLL码作为下标
int run(int j,node a) {
    if (!strcmp(a.sign,"STOP"))                    //结束程序
        return 0;
    if(!strcmp(a.sign,"IF")) {                     //条件为真,执行下一句,否则跳过下一句
        if (value[(int)a.opera[0]]>value[(int)a.opera[2]])
            return j+1;
        return j+2;
    }
    if (!strcmp(a.sign,"GOTO")) {                  //跳转到指定行
        int k,ans=0;
        for (k=0; a.opera[k]; k++)
            ans=ans*10+a.opera[k]-'0';
        return ans;
    }
    if (!strcmp(a.sign,"LET")) {                   //赋值
        if (isalpha(a.opera[2]))                   //类似X=A+B的情况
            value[(int)a.opera[0]]=value[(int)a.opera[2]]+value[(int)a.opera[4]];
        else {                                     //类似X=0的情况
            value[(int)a.opera[0]]=0;
            for (int k=2; a.opera[k]; k++)
                value[(int)a.opera[0]]=value[(int)a.opera[0]]*10+a.opera[k]-'0';
        }
    } else                                         //输出
        printf("%c=%d\n",a.opera[0],value[(int)a.opera[0]]);
    return j+1;
}
int main() {
    int i,j=1;
    memset(value,0,sizeof(value));
    while(~scanf("%d",&i)) {
        scanf("%s",p[i].sign);
        if (strcmp(p[i].sign,"STOP"))
            scanf("%s",p[i].opera);
    }
    while(j)                                        //j为下一次执行的行数,为0则终结
        j=run(j,p[j]);
    return 0;
}

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值