自己动手写basic解释器(六)

 

自己动手写basic解释器

刺猬@http://blog.csdn.net/littlehedgehog

 





注: 文章basic解释源码摘自梁肇新先生的《编程高手箴言》(据他所说这个代码也是网上摘录的),源码解读参考《java编程艺术》。《java编程艺术》里面自然是java版了(可能旭哥更加适合点儿),我这里还是解读的C版basic解释器代码。





4、for
接着看for,for循环这里借助了栈来实现,很不错的方法。

  1. /* execute a FOR loop
  2.  * for 循环  其主要格式 文章第一篇已经给出  
  3.  * for i=1 to 10
  4.  * next i
  5.  * 下面就引用此例了
  6.  */
  7. void exec_for()
  8. {
  9.     struct for_stack i;     //申请一个栈元素  到时候加入
  10.     int value;

  11.     get_token();  /*  获取标号  这里获取到变量i */
  12.     if (!isalpha(*token))  //变量必定是字符型
  13.     {
  14.         serror(4);
  15.         return;
  16.     }

  17.     i.var = toupper(*token) - 'A';  /* 我们是把变量放在hash表中的  所以这里来计算变量在hash表中位置   */

  18.     get_token();  /* 这里得到了等号 */
  19.     if (*token!='=')  
  20.     {
  21.         serror(3);
  22.         return;
  23.     }
  24.     get_exp(&value);  /* 初始值  比如这里是1 */

  25.     variables[i.var]=value;     //这里把初始值放在变量数组中

  26.     get_token();

  27.     if (tok != TO) serror(9);  /* 读取to单词 */
  28.     get_exp(&i.target);  /* 取得最终要达到的数值  比如这里是10 */

  29.     /* if loop can execute at least once, push into on stack */
  30.     if (value<=i.target)  {
  31.         i.loc = prog;       //记录要执行的语句  这里是for循环的里面要执行的语句
  32.         fpush(i);       //压栈
  33.     }
  34.     else  /* otherwise, skip loop code altogether */
  35.         while (tok!=NEXT)  get_token();     //每到next之前  都输入for循环要执行的语句   所以一直执行
  36. }
通过压栈可以记录for循环要执行的一系列语句。压栈代码如下,同样很简单:

  1. void fpush(struct for_stack i)
  2. {
  3.     if (ftos>FOR_NEST)
  4.     serror(10);
  5.     fstack[ftos]=i;
  6.     ftos++;
  7. }
for循环依靠next关键字来进入下一轮循环,并且要把循环变量加一

  1. /* execute a NEXT statement */
  2. void next()
  3. {
  4.     struct for_stack i;

  5.     i = fpop();  /* 这里我们读出当年压栈的记录 */

  6.     variables[i.var]++;  /* 变量加一 */
  7.     if (variables[i.var]>i.target)  return;  /* 这个自然是作完了 */
  8.     fpush(i);   /* 没做完事情,记录当前变量  压栈 */
  9.     prog = i.loc;  /* 记得么  loc记录了for循环的第一个要执行的语句 这样我们就顺利地转入下次循环开始执行 */
  10. }


出栈代码也很简单,就是控制栈顶指针而已。

  1. struct for_stack fpop()
  2. {
  3.     ftos--;
  4.     if (ftos<0)  serror(11);
  5.     return (fstack[ftos]);
  6. }

通过栈不仅可以解决for循环的问题,而且gosub这个关键字也是通过栈来解决的。回想一下,我们在函数调用过程,cpu会先把当前的eip所指向的地址压栈,函数执行完后我们再出栈以跳回原执行地址。

  1. /* execute a GOSUB command 
  2.  * 这个类似c语言中的函数调用 */
  3. void gosub()
  4. {
  5.     char *loc;

  6.     get_token();
  7.     /* find the label to call */
  8.     loc = find_label(token);
  9.     if (loc=='/0')
  10.         serror(7);  /* label not defined */
  11.     else  
  12.     {
  13.         gpush(prog);  /* 当前执行的地址压栈 */
  14.         prog = loc;  /* 重新把要执行的地址赋值给prog */
  15.     }
  16. }
值得注意的是我们这里又重新定义了一个栈以及相关的操作,而不是采用的for循环的栈。如下
  1. char *gstack[SUB_NEST];  /* stack for gosub  就是一个字符指针数组*/
  2. int gtos;  /* index to top of GOSUB  栈顶指针 */

相关的函数也很简单:
  1. /* GOSUB stack push function */
  2. void gpush(char *s)
  3. {
  4.     gtos++;

  5.     if (gtos==SUB_NEST)  
  6.     {
  7.         serror(12);
  8.         return;
  9.     }

  10.     gstack[gtos] = s;
  11. }


  12. /* GOSUB stack pop function */
  13. char *gpop()
  14. {
  15.     if (gtos==0)  {
  16.         serror(13);
  17.         return 0;
  18.     }
  19.     return gstack[gtos--];
  20. }


go_sub的处理体现出了程序语言运行的精髓,通过压栈保留当前信息(当前地址),必要时出栈又获取相应的信息
(想要返回的地址) 。cpu就是这样执行的,当然我们设计解释器也可以参照这个模型,来看看如何从一个"函数"(Basic中应该称作sub,子过程)中返回。

  1. /* return from GOSUB */
  2. void greturn()
  3. {
  4.     prog = gpop();   //出栈,并把栈顶元素(即原来的地址)付给prog
  5. }






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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值