摘要:一般用大写的表示文法的开头,称为开始符号。更多讨论讨论地址是精读手写编译器文法介绍如果你想参与讨论,请点击这里,每周都有新的主题,周末或周一发布。
1 引言
文法用来描述语言的语法规则,所以不仅可以用在编程语言上,也可用在汉语、英语上。
2 精读我们将一块语法规则称为 产生式,使用 “Left → Right” 表示任意产生式,用 “Left => Right” 表示产生式的推导过程,比如对于产生式:
E → i E → E + E
我们进行推导时,可以这样表示:E => E + E => i + E => i + i + E => i + i + i
也有使用 Left : Right 表示产生式的例子,比如 ANTLR。BNF 范式通过 Left ::= Right 表示产生式。
举个例子,比如 SELECT * FROM table 可以被表达为:
S → SELECT * FROM table
当然这是最固定的语法,真实场景中,* 可能被替换为其他单词,而 table 不但可能有其他名字,还可能是个子表达式。
一般用大写的 S 表示文法的开头,称为开始符号。终结符与非终结符
下面为了方便书写,使用 BNF 范式表示文法。
终结符就是语句的终结,读到它表示产生式分析结束,相反,非终结符就是一个新产生式的开始,比如:
::= SELECT FROM ::= [ , ] ::= [ , ]
所有 ::= 号左边的都是非终结符,所以 selectList 是非终结符,解析 selectStatement 时遇到了 selectList 将会进入 selectList 产生式,而解析到普通 SELECT 单词就不会继续解析。
对于有二义性的文法,可以通过 上下文相关文法 方式描述,也就是在产生式左侧补全条件,解决二义性:
aBc -> a1c | a2c dBe -> d3e
一般产生式左侧都是非终结符,大写字母是非终结符,小写字母是终结符。
上面表示,非终结符 B 在 ac 之间时,可以解析为 1 或 2,而在 de 之间时,解析为 3。但我们可以增加一个非终结符让产生式可读性更好:
B -> 1 | 2 C -> 3
这样就将上下文相关文法转换为了上下文无关文法。
上下文无关文法根据是否依赖上下文,文法分为 上下文相关文法 与 上下文无关文法,一般来说 上下文相关文法 都可以转换为一堆 上下文无关文法 来处理,而用程序处理 上下文无关文法 相对轻松。
SQL 的文法就是上下文相关文法,在正式介绍 SQL 文法之前,举一个简单的例子,比如我们描述等号(=)的文法:
SELECT CASE WHEN bee = "red" THEN "ANGRY" ELSE "NEUTRAL" END AS BeeState FROM bees; SELECT * from bees WHERE bee = "red";
上面两个 SQL 中,等号前后的关键字取决于当前是在 CASE WHEN 语句里,还是在 WHERE 语句里,所以我们认为等号所在位置的文法是上下文相关的。
但是当我们将文法粒度变细,将 CASE WHEN 与 WHERE 区块分别交由两块文法解决,将等号这个通用的表达式抽离出来,就可以不关心上下文了,这种方式称为 上下文无关文法。
附上一个 mysql 上下文无关文法集合。
左推导与右推导上面提到的推导符号 => 在实际运行过程中,显然有两种方向左和右:
E + E => ?
从最左边的 E 开始分析,称为左推导,对语法解析来说是自顶向下的方式,常用方法是递归下降。
从最右边的 E 开始分析,称为右推导,对语法解析来说是自底向上的方式,常用方法是移进、规约。
右推导过程比左推导过程复杂,所以如果考虑手写,最好使用左推导的方式。
左推导的分支预测比如 select
::= , |
由于它可以展开:SelectList => SelectList , a => SelectList , b, a => c, b, a。
但程序执行时,读到这里会进入死循环,因为 SelectList 可以被无限展开,这就是左递归问题。
消除左递归消除左递归一般通过转化为右递归的方式,因为左递归完全不消耗 Token,而右递归可以通过消耗 Token 的方式跳出死循环。
Token 见上一期精读 精读《手写 SQL 编译器 - 词法分析》
::= ::= , | null
这其实是一个通用处理,可以抽象出来:
E → E + F E → F
E → FG G → + FG G → null
不过我们也不难发现,通过通用方式消除左递归后的文法更难以阅读,这是因为用死循环的方式解释问题更容易让人理解,但会导致机器崩溃。
笔者建议此处不要生硬的套公式,在套了公式后,再对产生式做一些修饰,让其更具有语义:
提取左公因式::= | ,
即便是上下文无关的文法,通过递归下降方式,许多时候也必须从左向右超前查看 K 个字符才能确定使用哪个产生式,这种文法称为 LL(k)。
但如果每次超前查看的内容都有许多字符相同,会导致第二次开始的超前查看重复解析字符串,影响性能。最理想的情况是,每次超前查看都不会对已确定的字符重复查看,解决方法是提取左公因式。
设想如下的 sql 文法:
::= as | as | |
其实 Text 本身也是比较复杂的产生式,最坏的情况需要对 Text 连续匹配六遍。我们将 Text 公因式提取出来就可以仅匹配一遍,因为无论是何种 Field 产生式,都必定先遇到 Text:
::= ::= | ::= as ::= |
和消除左递归一样,提取左公因式也会降低文法的可读性,需要进行人为修复。不过提取左公因式的修复没办法在文法中处理,在后面的 “函数式” 处理环节是有办法处理的,敬请期待。
结合优先级对 SQL 的文法来说不存在优先级的概念,所以从某种程度来说,SQL 的语法复杂度还不如基本的加减乘除。
3 总结在实现语法解析前,需要使用文法描述 SQL 的语法,文法描述就是语法分析的主干业务代码。
下一篇将介绍语法分析相关知识,帮助你一步步打造自己的 SQL 编译器。
4 更多讨论讨论地址是:精读《手写 SQL 编译器 - 文法介绍》 · Issue #94 · dt-fe/weekly
如果你想参与讨论,请点击这里,每周都有新的主题,周末或周一发布。
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/96170.html
摘要:返回的语法树作为结果被传递到文法中,其结果可能是。每个元素的子节点全部执行完毕,才会生成当前节点的语法树。更多讨论讨论地址是精读手写编译器语法树如果你想参与讨论,请点击这里,每周都有新的主题,周末或周一发布。 1 引言 重回 手写 SQL 编辑器 系列。之前几期介绍了 词法、文法、语法的解析,以及回溯功能的实现,这次介绍如何生成语法树。 基于 《回溯》 一文介绍的思路,我们利用 JS ...
摘要:经过连续几期的介绍,手写编译器系列进入了智能提示模块,前几期从词法到文法语法,再到构造语法树,错误提示等等,都是为智能提示做准备。 1 引言 词法、语法、语义分析概念都属于编译原理的前端领域,而这次的目的是做 具备完善语法提示的 SQL 编辑器,只需用到编译原理的前端部分。 经过连续几期的介绍,《手写 SQL 编译器》系列进入了 智能提示 模块,前几期从 词法到文法、语法,再到构造语法...
阅读 3074·2021-11-08 13:18
阅读 2234·2019-08-30 15:55
阅读 3570·2019-08-30 15:44
阅读 3016·2019-08-30 13:07
阅读 2751·2019-08-29 17:20
阅读 1902·2019-08-29 13:03
阅读 3371·2019-08-26 10:32
阅读 3187·2019-08-26 10:15