6 上下文无关语言
6.1 上下文无关语言的泵引理
6.1.1 直观感受
回忆一下正则语言的泵引理,他告诉我们,如果存在某个字符串,它足够长,长到在这个语言的 DFA 中导致了一个环,那么我们就可以将这个环泵进泵出,从而找到一个属于这个语言的无穷的字符串序列。
对于上下文无关语言来说,情况会稍微有些复杂。
我们总是能够找到一个足够长的字符串的 两个 部分,将它们泵进泵出。也就是说,如果我们将这个字符串中的这两个部分任意地重复相同的次数,就可以得到依旧在这个语言当中的另外的字符串。
6.1.2 形式化描述
下面,我们给出 上下文无关语言的泵引理 (CFL Pumping Lemma) 的形式化描述:
定理 6.1
对于每一个上下文无关语言 ,存在一个整数 ,使得 ,,满足:
- ;
- ;
- 。
这个引理的证明就要用到上下文无关文法的乔姆斯基范式 (CNF) 了。
考虑 的 CNF 一个文法,令这个文法有 个不同的变量,取 。
取 ,有 引理 1:产出 的解析树一定存在一条长度至少为 的路径。
先证引理 1:反证法。如果一个 CNF 文法的解析树的所有路径长度都 的话,那么这个解析树能够产生的字符串的长度最长也不过 ,就像:
因为 CNF 的解析树是一个二叉树,而高度不超过 的二叉树最多有 个叶子结点,每个叶子结点后续最多产生一个终结符,因此产生的字符串长度最多为 ,这与 是矛盾的。
引理 1 证完了,我们回到泵引理的证明。现在,我们知道了 的解析树存在一条路径,上面有至少 个变量。(因为再加上一个终结符,长度至少为 )。
考虑某条最长的路径,由于只存在 个不同的变量,所以在低 个非叶结点中,根据鸽笼原理,我们能够找到两个结点,它们标记的是同一个变量,不妨记为 ,则解析树形如:
其中,,因为它们不能同时为空,这是由 CNF 文法中不含空产生式所决定的。
且 ,因为我们只选取了最长的路径的低 个非叶结点,即这颗子树的树高最多为 ,从而产出的字符串最多为叶子结点的最大个数 。
有了这颗树之后,泵引理的结论就很明显了。泵 0 次的结果如下:
泵 2 次的结果如下:
泵 3 次的结果如下:
这些情况都在原语言中,因为我们能够找到一棵对应的合法的解析树。造成这些解析树不同的原因就是当遇到变量 的时候,我们可以有两条候选的推导方式使用,一条是 ,一条是 。
6.1.3 泵引理的使用
是一个上下文无关语言,因为我们可以数两样东西(这是刚接触上下文无关文法的时候讲过的一个直觉,当然,你也可以直接通过给出文法来证明它是上下文无关语言)。
但是 不是上下文无关语言,因为我们不能同时做两对计数,也就是说不能同时数三件东西。不过这只是一个直觉,不能够用作证明。
我们可以通过泵引理来证明。假设 是一个上下文无关语言,令 是 的泵引理常数。
考虑 ,我们可以写作 ,其中 且 。
第一种情况: 中没有 。
- 它们之中至少有一个是 ,从而 中之多有一个 ,这是不可能的,因为这样 就无法泵进泵出了。
第二种情况: 至少有一个 。
- 的长度太短了( ),从而无法覆盖 中所有的三个 字串。
- 于是, 中至少有一个 子串 (没有被 覆盖到),以及至少一个 字串不足 个 (因为 删去而被丢掉了)。
- 从而,。
6.2 上下文无关语言的判定性质
6.2.1 判定性质概述
如常,当我们讨论一个上下文无关语言的时候,我们其实是在说这个语言的一种表示,比如说一个上下文无关文法,或者说一个根据终止状态或空栈接收语言的下推自动机。
存在一些算法来判定:
- 字符串 是否在上下文无关语言 中;
- 上下文无关语言 是否为空;
- 上下文无关语言 是否无限。
不过,也有很多的问题,在正则语言中可以判定,但是上下文无关语言却不可以。
比如说:两个上下文无关语言是否相同?两个上下文无关语言是否不想交?
对于正则语言,我们可以用乘积自动机来判定,但是上下文无关语言并没有很好的手段。
不过,为了证明不存在这样的算法,需要一些图灵机和判定醒的理论,这里就不详细展开了。
6.2.2 判断是否为空
我们其实已经做过这件事情(test emptiness)了,之前学习上下文无关文法的时候,我们学过去除无用变量的算法。
如果起始符号是一个无用变量,那么这个上下文无关语言为空,否则不为空。
6.2.3 判断字符串是否属于语言
我们想要知道字符串 是否在上下文无关文法 描述的上下文无关语言 中。
假设 是 CNF 形式,否则将给定的文法转化成 CNF 形式,其中 是一个特殊情况,可以通过判断起始符号是否可空来解决(这一点其实之前将上下文无关文法的化简时候已经讲过了,这里不再赘述)。
CYK 算法 是一个很好的动态规划的例子,并且可以在 的时间内判断是否属于 (test membership),其中 。
CYK 算法:令 ,我们将会构造一个 的三角形的变量阵列:
对 ,即产生的字符串的长度进行归纳。
最终,判断 是否在 中即可。
基础情况:
归纳:。
比如说对于文法 ,判断 是否在这个文法描述的语言中。
算法过程如下:
最后我们发现,,从而该字符串并不在这个文法所描述的语言中。
6.2.4 判断是否无穷
判断无穷性 (test infiniteness) 的想法和正则语言本质上是类似的,使用泵引理常数 ,如果该语言存在一个长度在 到 之间的字符串,那么这个语言就是无穷的,否则就是有穷的。
6.3 上下文无关语言的闭包性质
上下文无关语言在并集、拼接和星闭包下都是封闭的。并且在反转、同态和逆同态下也是封闭的。但在交集和差集操作下并不封闭。
6.3.1 并集下的闭包性质
令 和 分别是上下问无关文法 和 下的上下文无关语言。不妨设 和 没有共同的变量(变量的名字不改变语言,可以通过重命名的方式做到没有共同的变量这一点)。令 和 分别是 和 的起始符号。
基于 和 的产生式集合与符号集合来构建 的文法,增加一个新的起始符号 ,增加一个新的产生式 即可。
在新的文法中,所有的推导都是从 开始的,因此第一步总是将 用 或者 替代。在第一种情况中,结果一定是 中的一个字符串,第二种情况中,结果一定是 中的一个字符串,从而新文法产生的语言是原来语言的并集。
6.3.2 拼接下的闭包性质
还是令 和 分别是上下问无关文法 和 下的上下文无关语言。不妨设 和 没有共同的变量(变量的名字不改变语言,可以通过重命名的方式做到没有共同的变量这一点)。令 和 分别是 和 的起始符号。
基于 和 的产生式集合与符号集合来构建 的文法,增加一个新的起始符号 ,增加一个新的产生式 。
这样的话,每一个从 开始的推导都会得出一个 中的字符串接着一个 中的字符串。
6.3.3 星闭包下的闭包性质
令上下文无关语言 具有文法 和起始符号 ,通过给 引入新的起始变量 以及产生式 的方式构建 的文法。
一个从 开始的最右推导会产生一个由 个或者多个 构成的句型,每个 都可以产生 中的某个字符串,从而能够导出 。
6.3.4 翻转操作下的闭包性质
如果 是一个具有文法 的上下文无关语言,通过翻转每一个产生式体的方式来构造 的文法。
比如说令 有 ,则 有 。
6.3.5 同态操作下的闭包性质
令 是一个具有文法 的上下文无关语言,令 是一个在 的终结符上的同态。通过将每个终结符 替换成 的方式为 构造一个文法。
比如说 有产生式 , 定义为 ,则 的文法具有产生式 。
6.3.6 交集与差集
和正则语言不同的是,上下文无关语言类在交集操作下并不封闭。我们知道, 不是一个上下文无关语言(可以通过泵引理证明)。
然而 是(构造文法证明: )。
同理, 也是,但 并不是。
我们其实可以证明一个更一般的结论:任何在差集下封闭的语言,在交集下也封闭。
证明:。
于是,上下文无关文法在差集下也不封闭了。(否则和它在交集下不封闭就矛盾了)
不过,一个上下文无关语言和一个正则语言相交依旧是一个上下文无关语言。虽然这并不能称作一个闭包性质,但也很不错了。
这个性质的证明过程包含了将一个 DFA 和一个 PDA 同时运行,它们组合起来依旧是 PDA。这里 PDA 是根据终止状态接收的。
形式化构造:
令 DFA 的转移函数为 ,PDA 的转移方程为 。组合起来的 PDA 的状态形如 ,其中 是一个 的状态, 是一个 的状态。
其实和之前的乘积自动机类似,只不过增加了一个栈而已。需要注意的是这里 可以是 ,在这种情况下 。
组合起来的 PDA 的终止状态形如 ,满足 是 的一个终止状态, 是 的一个接收状态。
初始状态是由两个初始状态组成的 。
简单归纳便可得到:
从而证明新的到的 PDA 所推导的语言是之前的 DFA 和 PDA 所推导语言的交集。
6.3.7 逆同态下的闭包性质
先回忆一下逆同态是什么,令 是一个同态, 是一个字母表为 的输出语言的语言,则 。
比如说令 ,,则 。
当时,我们是通过构造 DFA 的方式来证明正则语言的该性质的,因为逆同态有些特殊,是字符串映射成字符,不是字符映射成字符串,所以使用正则表达式不太方便。
这里也是类似的,我们不使用上下文无关文法,而是通过构造新的 PDA 的方式来证明这个性质。
令 是由某个 PDA 所定义的语言,下面构建 PDA 接收 。
整体上是模拟 的,不过维护了一个用于存放 的结果字符串的缓冲区作为它状态的一部分。
可以通过缓冲区将原本接收一个字符串模拟成接收一个字符。
的形式化构造:
- 状态形如 ,其中:
- 是 的一个状态;
- 是对于某个 的后缀, 是 定义域中的某个元素。
- 这样的话, 只会有有限个可能的值。
- 的栈符号还是原来 的栈符号。
- 的起始状态是 。
- 的输入符号是 可作用于的符号,
- 的终止状态形如 ,其中 是 的一个终止状态。
下面定义它的转移函数:
- 对于 的任意输入符号 和任意栈符号 ,。
- 当缓冲区为空的时候, 可以重新加载。
- ,如果 ,其中 是 的一个输入符号或者 。
- 通过缓冲区来模拟 ,结合第一个转移方程,就可以将 接收一个字符串的行为模拟成 接收一个字符的行为,从而实现逆同态。
下面我们只需要证明 。
关键点是: 发生转移 当且仅当 发生转移 ,其中 。
两个方向的证明都是对于移动的次数进行归纳。
一旦我们又了这个结论,我们就可以将它限制到 为空且 是终止状态的情况。这就相当于是在说 接收 当且仅当 接收 。也就是说 定义的语言是 定义的语言的逆同态。