语义分析之二:地址回填问题的”拉链“式解决方案

前一篇文章介绍了语义分析过程涉及到中间代码生成过程,整个生成思路都是静态的,但是如果遇到较为复杂的逻辑问题,比如控制流出口地址填写问题、外部变量引用的地址填写回填,这便涉及到边编译边回填逻辑地址的问题。遇到这种动态控制流逻辑问题,语义分析阶段采用了一种巧妙的”拉链“式回填方案,时空效率都不错的解决方案。

说着很玄乎,其实”拉链“式解决方案是链表思想的灵活使用,这一思路可以说蛮有灵性的,所以特地写了这篇文章,用来为日后可能的需求提供思路。说了这么多,下面便引入控制流逻辑地址回填问题。

//假设要翻译 if a<b or c<d and e>f ...这段程序

//理想的情况是翻译成下面这样
(1) if a<b goto E.true //E.true代表该if...else...逻辑的正向逻辑段入口地址,但是翻译当前bool表达式显然是暂且还不能确定E.true的
(2) goto (3)
(3) if c<d goto (5)
(4) goto E.false       //E.false代表if...else...的反向逻辑段入口地址
(5) is e>f goto E.true
(6) goto E.false //到bool表达式翻译完成时,才能确定原来逻辑块的正向段入口地址是77...

可以看到上面的出现E.trueE.false的地方是需要回填的逻辑地址,这里比较常见的思路可能是额外为每个逻辑地址保存一个数组用来记录需要的所有地址,比如定义一个E.true-speicific-array并将(1)(2)填入该数组中。这种做法显然是很笨的,有没有比较好的方法?这便是”拉链“式解决方案,借鉴自链表的思想,为所有需要回填E.true的指令行首尾串在一起。如下所示

10) … goto E.true
       …
 (20) … goto E.true
       …
 (30) … goto E.true

//首尾相连,采用倒序的方式串在一起,这样不借助额外的数据结构便实现了定位的目的,并且动态增长也是很简单的

  (10) … goto (0)
         …
  (20) … goto (10)
         …
  (30) … goto (20

不过既然说到了这里,那么通过”拉链“策略便可以自然地引出符号表的概念。对于上面逻辑地址的回填其实还可以让编译程序通过临时变量来保存并回填,那么如果对于goto语句这种段跳转的逻辑,显然便是需要更严格的数据结构来进行相应的数据加工了。

现在如果存在一个语句goto procedure_P;代表跳转到procedure_P段,那么如何翻译该程序段。显然编译程序在没有扫描到段P时,只能翻译成(jump, -, -, P),然后等待段P确定后再回填,但回填后难道就把这个段P的入口地址信息给丢弃吗?那如果之后又遇到了goto procedure_P;又该怎么办呢?所以显然需要保存这个段P的入口地址信息,这个用来保存标识符(变量、函数、过程段等)相关属性的表格便是符号表。


拉链式回填方案和符号表的配合使用

  • 8
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
⒈ 题目 分析对象的BNF定义如下: 〈算术表达〉∷=〈项〉|〈算术表达〉+〈项〉|〈算术表达〉-〈项〉 〈项〉∷=〈因〉|〈项〉*〈因〉|〈项〉/〈因〉 〈因〉∷=〈变量〉│(〈算术表达〉) 〈变量〉∷=〈字母〉 〈字母〉∷=A|B|C|D|E|F|G|H|I|J|K|L|M|N|O|P|Q|R|S|T|U|V|W|X|Y|Z (a)总程序过程 (b) ( E过程)算术表达处理 (c)项处理(T过程) (d)因处理过程)(F过程) (e) (f) 图2-7-5 递归下降法分析表达之框图 (a) ZC 过程;(b) E 过程;(c) T 过程; (d) F 过程;(e) 函数过程 SYM ;(f) 过程 ADVANCE ⒉ 算法 用递归下降法分析上述算术表达的框图,如图2-7-5所示。这里,ZC过程为总控程序,主要完成: ⑴ 通知外界键入算术表达; ⑵ 控制E过程分析算术表达; ⑶ 根据分析结果之正误,分别通知外界不同的信息。 ZC过程被设计成可以分析无穷多个算术表达。E、T和F三个过程分别对应〈算术表达〉、〈项〉和〈因〉三个产生的处理。它们用到两个公共过程。一个是函数过程SYM,它负责从输入字符串ST中取出下一个字符,并存入SYM中等待分析。另一个过程ADVANCE负责剔除ST中的首字符。 算法的书写和实现也请参考课堂教学所给出的方法和实例,不一定照搬以上的框图。 ⒋ 小结 ⑴ 实习前的准备 按实习目的和要求,用PASCAL语言编写一个语法分析程序,同时考虑相应的数据结构。 ⑵ 调试 调试例子应包括符合语法规则的算术表达,以及分析程序能够判别的若干错例。 ⑶ 输出 对于所输入的算术表达,不论对错,都应有明确的信息告诉外界。 ⑷编写上机实习报告。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值