另一个答案侧重于判别部分,我将侧重于手动证明。你试过:
Lemma l2: A=B -> False.
apply (fun e:(A=B) => match e with end).
Defined.
使用 Coq 时应该注意并且让我经常感到不舒服的是,Coq 接受定义不明确的定义,并在内部将其重写为类型良好的术语。这允许不那么冗长,因为 Coq 自己添加了一些部分。但另一方面,Coq 使用的术语与我们输入的术语不同。
你的证明就是这种情况。自然,模式匹配e
应该涉及到构造函数eq_refl
,它是该eq
类型的单个构造函数。在这里,Coq 检测到不存在相等性并因此了解如何修改您的代码,但您输入的内容不是正确的模式匹配。
有两种成分可以帮助理解这里发生了什么:
- 的定义
eq
- 完整的模式匹配语法,
as
within
和return
terms
首先,我们可以看一下 的定义eq
。
Inductive eq {A : Type} (x : A) : A -> Prop := eq_refl : x = x.
请注意,此定义与看起来更自然的定义不同(无论如何,更对称)。
Inductive eq {A : Type} : A -> A -> Prop := eq_refl : forall (x:A), x = x.
eq
用第一个定义而不是第二个定义来定义这一点非常重要。特别是,对于我们的问题,重要的是,inx = y
是x
一个参数,而y
是一个索引。也就是说,x
在所有构造函数中是恒定的,而y
在每个构造函数中可以不同。您与 type 有相同的区别Vector.t
。如果添加元素,向量元素的类型不会改变,这就是它作为参数实现的原因。然而,它的大小是可以改变的,这就是它被实现为索引的原因。
现在,让我们看看扩展的模式匹配语法。我在这里对我所理解的内容做一个非常简短的解释。不要犹豫,查看参考手册以获取更安全的信息。该return
子句可以帮助指定每个分支都不同的返回类型。该子句可以使用模式匹配的as
andin
子句中定义的变量,它分别绑定匹配的术语和类型索引。该return
子句将在每个分支的上下文中进行解释,替换as
并in
使用该上下文的变量,逐个对分支进行类型检查,并用于match
从外部角度键入。
这是一个带有as
子句的人为示例:
Definition test n :=
match n as n0 return (match n0 with | 0 => nat | S _ => bool end) with
| 0 => 17
| _ => true
end.
根据 的值n
,我们不会返回相同的类型。的类型test
是forall n : nat, match n with | 0 => nat | S _ => bool end
。但是当 Coq 可以决定我们在哪种情况下匹配时,它可以简化类型。例如:
Definition test2 n : bool := test (S n).
在这里,Coq 知道,无论是什么n
,S n
给定的test
结果都是 type bool
。
对于相等,我们可以做类似的事情,这次使用in
从句。
Definition test3 (e:A=B) : False :=
match e in (_ = c) return (match c with | B => False | _ => True end) with
| eq_refl => I
end.
这里发生了什么 ?本质上,Coq 分别对 the 的分支match
和match
自身的分支进行类型检查。在唯一的分支eq_refl
中,c
等于A
(因为它的定义eq_refl
实例化了与参数相同的值的索引),因此我们声称我们返回了一些类型的值True
,这里I
。但是从外部的角度来看,c
等于B
(因为e
是 type A=B
),这一次return
子句声称match
返回了一些 type 的值False
。我们在这里使用 Coq 的功能来简化我们刚刚看到的类型中的模式匹配test2
。请注意,我们True
在其他情况下使用了B
,但我们并不True
特别需要。我们只需要一些有人居住的类型,这样我们就可以在eq_refl
分支中返回一些东西。
回到 Coq 产生的奇怪术语,Coq 使用的方法做了类似的事情,但在这个例子中,肯定更复杂。特别是,当 CoqIDProp
需要idProp
无用的类型和术语时,它经常使用类型。它们对应于True
并I
在上面使用。
最后,我给出了关于 coq-club 的讨论链接,它确实帮助我理解了如何在 Coq 中键入扩展模式匹配。