module plfa.part3.Compositional where

简介

本章我们会证明指称语义满足可组合性,即我们会填补以下方程中省略号的部分。

ℰ (` x) ≃ ...
ℰ (ƛ M) ≃ ... ℰ M ...
ℰ (M · N) ≃ ... ℰ M ... ℰ N ...

这些方程蕴含了指称语义也可以定义为一个递归函数。 我们会在本章的结尾给出它的定义,并证明它等价于 ℰ。

导入

open import Data.Product.Base using (_×_; Σ; Σ-syntax; ; ∃-syntax; proj₁; proj₂)
  renaming (_,_ to ⟨_,_⟩)
open import Data.Sum.Base using (_⊎_; inj₁; inj₂)
open import Data.Unit.Base using (; tt)
open import plfa.part2.Untyped
  using (Context; _,_; ; _∋_; _⊢_; `_; ƛ_; _·_)
open import plfa.part3.Denotational
  using (Value; _↦_; _`,_; _⊔_; ; _⊑_; _⊢_↓_;
         ⊑-bot; ⊑-fun; ⊑-conj-L; ⊑-conj-R1; ⊑-conj-R2;
         ⊑-dist; ⊑-refl; ⊑-trans; ⊔↦⊔-dist;
         var; ↦-intro; ↦-elim; ⊔-intro; ⊥-intro; sub;
         up-env; ; _≃_; ≃-sym; Denotation; Env)
open plfa.part3.Denotational.≃-Reasoning

λ-抽象的方程

考虑第一个方程

ℰ (ƛ M) ≃ ... ℰ M ...

我们需要定义一个函数,它将 Denotation (Γ , ★) 映射到 Denotation Γ, 我们将此函数命名为 。在应用到 λ-项时,它能够模拟语义中非递归的部分。 具体来说,我们需要考虑规则 ↦-intro⊥-intro⊔-intro,因此 有三个参数:子项 M 的指称 D、环境 γ,以及值 v。如果我们通过对值 v 进行递归来定义 ,那么它正好匹配 ↦-intro⊥-intro⊔-intro 这三条规则。

 : ∀{Γ}  Denotation (Γ , )  Denotation Γ
 D γ (v  w) = D (γ `, v) w
 D γ  = 
 D γ (u  v) = ( D γ u) × ( D γ v)

如果你仔细观察,就会发现 函数看起来像函数式程序员熟悉的 curry 化操作:它将一个以长度为 n + 1 的元组(即环境 Γ , ★)作为参数的函数, 转化为一个以长度为 n 的元组作为参数,并返回一个单参函数的函数。

我们希望通过 函数证明

ℰ (ƛ N) ≃ ℱ (ℰ N)

当从一个更大的值 v 映射到一个更小的值 u 时,函数 会保持成立。 它的证明就是在 ⊑-fun 规则的情况中使用 up-env 引理,对 u ⊑ v 推导过程的直接归纳。

sub-ℱ : ∀{Γ}{N : Γ ,   }{γ v u}
    ( N) γ v
   u  v
    ------------
    ( N) γ u
sub-ℱ d ⊑-bot = tt
sub-ℱ d (⊑-fun lt lt′) = sub (up-env d lt) lt′
sub-ℱ d (⊑-conj-L lt lt₁) =  sub-ℱ d lt , sub-ℱ d lt₁ 
sub-ℱ d (⊑-conj-R1 lt) = sub-ℱ (proj₁ d) lt
sub-ℱ d (⊑-conj-R2 lt) = sub-ℱ (proj₂ d) lt
sub-ℱ {v = v₁  v₂  v₁  v₃} {v₁  (v₂  v₃)}  N2 , N3  ⊑-dist =
   ⊔-intro N2 N3
sub-ℱ d (⊑-trans x₁ x₂) = sub-ℱ (sub-ℱ d x₂) x₁

有了这个小前提的性质,我们就可以证明 λ-方程语义向右的方向。 证明是通过在 sub 规则的情况下使用 sub-ℱ,对语义进行归纳得出的。

ℰƛ→ℱℰ : ∀{Γ}{γ : Env Γ}{N : Γ ,   }{v : Value}
    (ƛ N) γ v
    ------------
    ( N) γ v
ℰƛ→ℱℰ (↦-intro d) = d
ℰƛ→ℱℰ ⊥-intro = tt
ℰƛ→ℱℰ (⊔-intro d₁ d₂) =  ℰƛ→ℱℰ d₁ , ℰƛ→ℱℰ d₂ 
ℰƛ→ℱℰ (sub d lt) = sub-ℱ (ℰƛ→ℱℰ d) lt

λ-抽象的「反演引理」是上面定理的一种特例。反演引理在证明「归约会保持指称不变」时会很有用。

lambda-inversion : ∀{Γ}{γ : Env Γ}{N : Γ ,   }{v₁ v₂ : Value}
   γ  ƛ N  v₁  v₂
    -----------------
   (γ `, v₁)  N  v₂
lambda-inversion{v₁ = v₁}{v₂ = v₂} d = ℰƛ→ℱℰ{v = v₁  v₂} d

λ-抽象的语义方程向左的方向甚至比向右的方向更容易证明,我们只需对值 v 进行归纳:

ℱℰ→ℰƛ : ∀{Γ}{γ : Env Γ}{N : Γ ,   }{v : Value}
    ( N) γ v
    ------------
    (ƛ N) γ v
ℱℰ→ℰƛ {v = } d = ⊥-intro
ℱℰ→ℰƛ {v = v₁  v₂} d = ↦-intro d
ℱℰ→ℰƛ {v = v₁  v₂}  d1 , d2  = ⊔-intro (ℱℰ→ℰƛ d1) (ℱℰ→ℰƛ d2)

因此,指称语义对 λ-抽象来说是可组合的,正如函数 所证明的那样:

lam-equiv : ∀{Γ}{N : Γ ,   }
    (ƛ N)   ( N)
lam-equiv γ v =  ℰƛ→ℱℰ , ℱℰ→ℰƛ 

函数应用的方程

接下来我们填补关于函数应用的方程中的省略号。

ℰ (M · N) ≃ ... ℰ M ... ℰ N ...

为此我们需要定义一个函数,它接受语境 Γ 中的两个指称,并产生语境 Γ 中的另一个指称。我们将此函数命名为 ,以模拟应用语义 L · M 中非递归的部分。我们无法像处理 那样简单地对值 v 进行递归来定义函数, 因为 ↦-elim 这类的规则可应用于任何值。相反,我们将通过直接处理 ↦-elim⊥-intro 规则而忽略 ⊔-intro 的方式来定义 。这使得证明的向右方向变得更加困难, 而 ⊔-intro 的情况说明了为什么 ⊑-dist 规则很重要。

这样我们就定义了 D₁ 应用于 D₂,记作 D₁ ● D₂,来包含 ⊥-intro 规则中所有等价于 的值 w,以及对于 ↦-elim 规则,在 D₁ 中的条目 v ↦ w 的所有输出值 w,前提是它的输入值 vD₂ 中。

infixl 7 _●_

_●_ : ∀{Γ}  Denotation Γ  Denotation Γ  Denotation Γ
(D₁  D₂) γ w = w    Σ[ v  Value ]( D₁ γ (v  w) × D₂ γ v )

如果你仔细观察一下,就会发现 _●_ 运算符看起来像函数式程序员熟悉的 apply 操作:它接受两个参数,并将第一个参数应用于第二个参数。

接下来我们考虑应用的反演引理,它同样是应用的语义方程的向右方向。 下面是它的证明:

ℰ·→●ℰ : ∀{Γ}{γ : Env Γ}{L M : Γ  }{v : Value}
    (L · M) γ v
    ----------------
   ( L   M) γ v
ℰ·→●ℰ (↦-elim{v = v′} d₁ d₂) = inj₂  v′ ,  d₁ , d₂  
ℰ·→●ℰ {v = } ⊥-intro = inj₁ ⊑-bot
ℰ·→●ℰ {Γ}{γ}{L}{M}{v} (⊔-intro{v = v₁}{w = v₂} d₁ d₂)
    with ℰ·→●ℰ d₁ | ℰ·→●ℰ d₂
... | inj₁ lt1 | inj₁ lt2 = inj₁ (⊑-conj-L lt1 lt2)
... | inj₁ lt1 | inj₂  v₁′ ,  L↓v12 , M↓v3   =
      inj₂  v₁′ ,  sub L↓v12 lt , M↓v3  
      where lt : v₁′  (v₁  v₂)  v₁′  v₂
            lt = (⊑-fun ⊑-refl (⊑-conj-L (⊑-trans lt1 ⊑-bot) ⊑-refl))
... | inj₂  v₁′ ,  L↓v12 , M↓v3   | inj₁ lt2 =
      inj₂  v₁′ ,  sub L↓v12 lt , M↓v3  
      where lt : v₁′  (v₁  v₂)  v₁′  v₁
            lt = (⊑-fun ⊑-refl (⊑-conj-L ⊑-refl (⊑-trans lt2 ⊑-bot)))
... | inj₂  v₁′ ,  L↓v12 , M↓v3   | inj₂  v₁′′ ,  L↓v12′ , M↓v3′   =
      let L↓⊔ = ⊔-intro L↓v12 L↓v12′ in
      let M↓⊔ = ⊔-intro M↓v3 M↓v3′ in
      inj₂  v₁′  v₁′′ ,  sub L↓⊔ ⊔↦⊔-dist , M↓⊔  
ℰ·→●ℰ {Γ}{γ}{L}{M}{v} (sub d lt)
    with ℰ·→●ℰ d
... | inj₁ lt2 = inj₁ (⊑-trans lt lt2)
... | inj₂  v₁ ,  L↓v12 , M↓v3   =
      inj₂  v₁ ,  sub L↓v12 (⊑-fun ⊑-refl lt) , M↓v3  

我们通过对语义进行归纳来证明:

  • 对于 ↦-elim 的情况,我们有 γ ⊢ L ↓ (v′ ↦ v)γ ⊢ M ↓ v′, 只需证明 (ℰ L ● ℰ M) γ v 即可。

  • 对于 ⊥-intro 的情况,我们有 v = ⊥,于是可推出 v ⊑ ⊥

  • 对于 ⊔-intro 的情况,我们有 ℰ (L · M) γ v₁ℰ (L · M) γ v₂, 且需要证明 (ℰ L ● ℰ M) γ (v₁ ⊔ v₂)。根据归纳假设,我们有 (ℰ L ● ℰ M) γ v₁(ℰ L ● ℰ M) γ v₂,于是我们需要考虑四种子情况:

    • 假设 v₁ ⊑ ⊥v₂ ⊑ ⊥,那么 v₁ ⊔ v₂ ⊑ ⊥

    • 假设 v₁ ⊑ ⊥γ ⊢ L ↓ v₁′ ↦ v₂γ ⊢ M ↓ v₁′, 根据 sub 规则我们有 γ ⊢ L ↓ v₁′ ↦ (v₁ ⊔ v₂),因为 v₁′ ↦ (v₁ ⊔ v₂) ⊑ v₁′ ↦ v₂

    • 假设 γ ⊢ L ↓ v₁′ ↦ v₁γ ⊢ M ↓ v₁′v₂ ⊑ ⊥, 根据 sub 规则我们有 γ ⊢ L ↓ v₁′ ↦ (v₁ ⊔ v₂),因为 v₁′ ↦ (v₁ ⊔ v₂) ⊑ v₁′ ↦ v₁

    • 假设 γ ⊢ L ↓ v₁′′ ↦ v₁, γ ⊢ M ↓ v₁′′γ ⊢ L ↓ v₁′ ↦ v₂γ ⊢ M ↓ v₁′, 这种情况是最有趣的部分。通过使用两次 ⊔-intro,我们有 γ ⊢ L ↓ (v₁′ ↦ v₂) ⊔ (v₁′′ ↦ v₁)γ ⊢ M ↓ (v₁′ ⊔ v₁′′)。但它对 ℰ L ● ℰ M 来说与我们的需要尚不匹配, 因为 L 的结果必须是一个输入条目为 v₁′ ⊔ v₁′′。 因此我们需要使用 sub 规则来得到 γ ⊢ L ↓ (v₁′ ⊔ v₁′′) ↦ (v₁ ⊔ v₂); 使用 ⊔↦⊔-dist 引理(感谢 ⊑-dist 规则)来证明:

        (v₁′ ⊔ v₁′′) ↦ (v₁ ⊔ v₂) ⊑ (v₁′ ↦ v₂) ⊔ (v₁′′ ↦ v₁)

      这样我们就证明了该情况所有的子情况。

  • 对于情况 sub,我们有 Γ ⊢ L · M ↓ v₁v ⊑ v₁。 根据归纳假设,我们有 (ℰ L ● ℰ M) γ v₁,于是我们需要考虑两种子情况:

    • 假设 v₁ ⊑ ⊥,我们可推出 v ⊑ ⊥
    • 假设 Γ ⊢ L ↓ v′ → v₁Γ ⊢ M ↓ v′,我们可根据规则 sub 推出 Γ ⊢ L ↓ v′ → v,因为 v′ → v ⊑ v′ → v₁

向左的方向可通过在前提 (ℰ L ● ℰ M) γ v 下进行情况分析来证明。 对于情况 v ⊑ ⊥,我们可通过规则 ⊥-intro 得到 Γ ⊢ L · M ↓ ⊥。 否则,我们可直接通过规则 ↦-elim 得出结论:

●ℰ→ℰ· : ∀{Γ}{γ : Env Γ}{L M : Γ  }{v}
   ( L   M) γ v
    ----------------
    (L · M) γ v
●ℰ→ℰ· {γ}{v} (inj₁ lt) = sub ⊥-intro lt
●ℰ→ℰ· {γ}{v} (inj₂  v₁ ,  d1 , d2  ) = ↦-elim d1 d2

这样我们就通过 函数的相关定理,证明了对于函数应用来说,语义是可组合的。

app-equiv : ∀{Γ}{L M : Γ  }
    (L · M)  ( L)  ( M)
app-equiv γ v =  ℰ·→●ℰ , ●ℰ→ℰ· 

我们还需要一个变量的反演引理:若 Γ ⊢ x ↓ v,则 v ⊑ γ x。 它的证明就是对语义的直接归纳:

var-inv :  {Γ v x} {γ : Env Γ}
    (` x) γ v
    -------------------
   v  γ x
var-inv (var) = ⊑-refl
var-inv (⊔-intro d₁ d₂) = ⊑-conj-L (var-inv d₁) (var-inv d₂)
var-inv (sub d lt) = ⊑-trans lt (var-inv d)
var-inv ⊥-intro = ⊑-bot

为了完善语义方程,我们为变量建立以下方程:

var-equiv : ∀{Γ}{x : Γ  }   (` x)   γ v  v  γ x)
var-equiv γ v =  var-inv ,  lt  sub var lt) 

合同性

本章的主要工作已经完成:我们建立了语义方程来展示指称语义是如何组合的。 在本节和下一节中,我们将利用这些方程来证明一些推论:指称相等满足合同性(congruence), 并证明指称具有可组合性(compositionality property), 该性质指出在同一语境中,两个指称相等的项会产生两个指称相等的程序 。

我们首先证明指称相等对 λ-抽象满足合同性:ℰ N ≃ ℰ N′ 蕴含 ℰ (ƛ N) ≃ ℰ (ƛ N′)。我们用 lam-equiv 方程将这个问题简化为 是否满足合同性:

ℱ-cong : ∀{Γ}{D D′ : Denotation (Γ , )}
   D  D′
    -----------
    D   D′
ℱ-cong{Γ} D≃D′ γ v =
    x  ℱ≃{γ}{v} x D≃D′) ,  x  ℱ≃{γ}{v} x (≃-sym D≃D′)) 
  where
  ℱ≃ : ∀{γ : Env Γ}{v}{D D′ : Denotation (Γ , )}
      D γ v    D  D′   D′ γ v
  ℱ≃ {v = } fd dd′ = tt
  ℱ≃ {γ}{v  w} fd dd′ = proj₁ (dd′ (γ `, v) w) fd
  ℱ≃ {γ}{u  w} fd dd′ =  ℱ≃{γ}{u} (proj₁ fd) dd′ , ℱ≃{γ}{w} (proj₂ fd) dd′ 

ℱ-cong 的证明使用了 ℱ≃ 来处理「当且仅当」的两个方向。 该引理直接通过对值 v 进行归纳得证。

现在我们通过直白的等式推理证明 λ-抽象满足合同性:

lam-cong : ∀{Γ}{N N′ : Γ ,   }
    N   N′
    -----------------
    (ƛ N)   (ƛ N′)
lam-cong {Γ}{N}{N′} N≃N′ =
  start
     (ƛ N)
  ≃⟨ lam-equiv 
     ( N)
  ≃⟨ ℱ-cong N≃N′ 
     ( N′)
  ≃⟨ ≃-sym lam-equiv 
     (ƛ N′)
  

接下来我们证明指称相等对于应用也满足合同性:ℰ L ≃ ℰ L′ℰ M ≃ ℰ M′ 蕴含 ℰ (L · M) ≃ ℰ (L′ · M′)。方程 app-equiv 将其归约为 运算符是否满足合同性的问题。

●-cong : ∀{Γ}{D₁ D₁′ D₂ D₂′ : Denotation Γ}
   D₁  D₁′  D₂  D₂′
   (D₁  D₂)  (D₁′  D₂′)
●-cong {Γ} d1 d2 γ v =   x  ●≃ x d1 d2) ,
                          x  ●≃ x (≃-sym d1) (≃-sym d2)) 
  where
  ●≃ : ∀{γ : Env Γ}{v}{D₁ D₁′ D₂ D₂′ : Denotation Γ}
     (D₁  D₂) γ v    D₁  D₁′    D₂  D₂′
     (D₁′  D₂′) γ v
  ●≃ (inj₁ v⊑⊥) eq₁ eq₂ = inj₁ v⊑⊥
  ●≃ {γ} {w} (inj₂  v ,  Dv↦w , Dv  ) eq₁ eq₂ =
    inj₂  v ,  proj₁ (eq₁ γ (v  w)) Dv↦w , proj₁ (eq₂ γ v) Dv  

同样,「当且仅当」的两个方向需要通过一个引理来证明,而该引理则通过对 (D₁ ● D₂) γ v 进行情况分析来证明。

根据 的合同性,我们可以直接通过等式推理证明应用也满足合同性。

app-cong : ∀{Γ}{L L′ M M′ : Γ  }
    L   L′
    M   M′
    -------------------------
    (L · M)   (L′ · M′)
app-cong {Γ}{L}{L′}{M}{M′} L≅L′ M≅M′ =
  start
     (L · M)
  ≃⟨ app-equiv 
     L   M
  ≃⟨ ●-cong L≅L′ M≅M′ 
     L′   M′
  ≃⟨ ≃-sym app-equiv 
     (L′ · M′)
  

可组合性

可组合性(Compositionality Property)说明了在同一语境下, 两个指称相等的项会产生两个指称相等的程序。为准确起见,我们定义了 「语境」和「在语境下」的含义。

语境(Context)是一个带洞的程序。以下数据定义 Ctx 将这个概念表示了出来。 我们使用两个变量语境来索引 Ctx 数据类型:一个表示洞,另一个表示填洞所产生的项。

data Ctx : Context  Context  Set where
  ctx-hole : ∀{Γ}  Ctx Γ Γ
  ctx-lam :  ∀{Γ Δ}  Ctx (Γ , ) (Δ , )  Ctx (Γ , ) Δ
  ctx-app-L : ∀{Γ Δ}  Ctx Γ Δ  Δ    Ctx Γ Δ
  ctx-app-R : ∀{Γ Δ}  Δ    Ctx Γ Δ  Ctx Γ Δ
  • 构造子 ctx-hole 表示洞,且在此情况中洞的变量语境 与填洞所产生的项的变量语境相同。
  • 构造子 ctx-lam 接受一个 Ctx 并在顶部添加 λ-抽象来产生一个更大的 Ctx。 洞的变量语境保持不变,而我们从结果项的语境中删除一个变量,因为它被此 λ-抽象约束。
  • 应用有两种构造,ctx-app-Lctx-app-Rctx-app-L 对应洞位于左侧项(运算符)内的情况,ctx-app-R 对应洞位于右侧项(运算数)内的情况。

将项插入到某个语境下的操作由以下 plug 函数定义,它是通过对语境进行递归来定义的:

plug : ∀{Γ}{Δ}  Ctx Γ Δ  Γ    Δ  
plug ctx-hole M = M
plug (ctx-lam C) N = ƛ plug C N
plug (ctx-app-L C N) L = (plug C L) · N
plug (ctx-app-R L C) M = L · (plug C M)

我们接下来陈述并证明组合性原则:给定两个项 MN,若它们的指称相等, 那么将它们插入到任意语境 C 中都会产生两个指称相等的程序:

compositionality : ∀{Γ Δ}{C : Ctx Γ Δ} {M N : Γ  }
    M   N
    ---------------------------
    (plug C M)   (plug C N)
compositionality {C = ctx-hole} M≃N =
  M≃N
compositionality {C = ctx-lam C′} M≃N =
  lam-cong (compositionality {C = C′} M≃N)
compositionality {C = ctx-app-L C′ L} M≃N =
  app-cong (compositionality {C = C′} M≃N) λ γ v    x  x) ,  x  x) 
compositionality {C = ctx-app-R L C′} M≃N =
  app-cong  γ v    x  x) ,  x  x) ) (compositionality {C = C′} M≃N)

它可以通过直接对语境 C 进行归纳,用我们之前建立的合同性质 lam-congapp-cong 来证明。

指称语义作为函数来定义

建立了 var-equivlam-equivapp-equiv 三个方程后, 我们就能将指称语义定义为一个在输入项 M 上递归的函数。其实, 我们可以定义以下函数 ⟦ M ⟧,利用辅助的柯里化函数 , 并分别在 λ-抽象和应用的情况下应用 函数来将项映射到指称。

⟦_⟧ : ∀{Γ}  (M : Γ  )  Denotation Γ
 ` x  γ v = v  γ x
 ƛ N  =   N 
 L · M  =  L    M 

ℰ M⟦ M ⟧ 指称相等的证明可通过 var-equivlam-equivapp-equiv 三个方程以及 的合同性引理直接归纳得出。

ℰ≃⟦⟧ :  {Γ} {M : Γ  }   M   M 
ℰ≃⟦⟧ {Γ} {` x} = var-equiv
ℰ≃⟦⟧ {Γ} {ƛ N} =
  let ih = ℰ≃⟦⟧ {M = N} in
     (ƛ N)
  ≃⟨ lam-equiv 
     ( N)
  ≃⟨ ℱ-cong (ℰ≃⟦⟧ {M = N}) 
      N 
  ≃⟨⟩
     ƛ N 
  
ℰ≃⟦⟧ {Γ} {L · M} =
    (L · M)
  ≃⟨ app-equiv 
    L   M
  ≃⟨ ●-cong (ℰ≃⟦⟧ {M = L}) (ℰ≃⟦⟧ {M = M}) 
    L    M 
  ≃⟨⟩
     L · M 
  

Unicode

本章使用了以下 Unicode:

ℱ  U+2131  SCRIPT CAPITAL F (\McF)
●  U+2131  BLACK CIRCLE (\cib)