这只是部分答案。
OP提出的问题是:在非常规递归类型的情况下如何定义fold
/ ?cata
因为我不相信自己能做到这一点,所以我会求助于 Coq。让我们从一个简单的常规递归列表类型开始。
Inductive List (A : Type) : Type :=
| Empty: List A
| Cons : A -> List A -> List A
.
这里没有什么花哨的东西,List A
是根据List A
. (记住这一点 - 我们会回到它。)
呢cata
?让我们查询归纳原理。
> Check List_rect.
forall (A : Type) (P : List A -> Type),
P (Empty A) ->
(forall (a : A) (l : List A), P l -> P (Cons A a l)) ->
forall l : List A, P l
让我们来看看。以上利用依赖类型:P
依赖于列表的实际值。让我们在P list
常量类型的情况下手动简化它B
。我们获得:
forall (A : Type) (B : Type),
B ->
(forall (a : A) (l : List A), B -> B) ->
forall l : List A, B
可以等效地写为
forall (A : Type) (B : Type),
B ->
(A -> List A -> B -> B) ->
List A -> B
除了foldr
“当前列表”也传递给二进制函数参数之外 - 不是主要区别。
现在,在 Coq 中,我们可以用另一种稍微不同的方式定义一个列表:
Inductive List2 : Type -> Type :=
| Empty2: forall A, List2 A
| Cons2 : forall A, A -> List2 A -> List2 A
.
它看起来是同一类型,但有很大的不同。在这里,我们不是List A
根据 来定义类型List A
。相反,我们是根据 定义一个类型List2 : Type -> Type
函数List2
。这样做的要点是递归引用List2
不必被应用到A
——事实上我们这样做只是一个事件。
无论如何,让我们看看归纳原理的类型:
> Check List2_rect.
forall P : forall T : Type, List2 T -> Type,
(forall A : Type, P A (Empty2 A)) ->
(forall (A : Type) (a : A) (l : List2 A), P A l -> P A (Cons2 A a l)) ->
forall (T : Type) (l : List2 T), P T l
让我们像以前一样删除List2 T
参数,基本上假设它是不变的。P
P
forall P : forall T : Type, Type,
(forall A : Type, P A ) ->
(forall (A : Type) (a : A) (l : List2 A), P A -> P A) ->
forall (T : Type) (l : List2 T), P T
等效改写:
forall P : (Type -> Type),
(forall A : Type, P A) ->
(forall (A : Type), A -> List2 A -> P A -> P A) ->
forall (T : Type), List2 T -> P T
大致对应,用 Haskell 表示法
(forall a, p a) -> -- Empty
(forall a, a -> List2 a -> p a -> p a) -> -- Cons
List2 t -> p t
还不错——基本情况现在必须是一个多态函数,就像Empty
在 Haskell 中一样。这有点道理。类似地,归纳案例必须是多态函数,就像Cons
是一样。还有一个额外的List2 a
参数,但如果我们愿意,我们可以忽略它。
现在,以上仍然是常规类型上的一种折叠/ cata 。非正规的呢?我会学习
data List a = Empty | Cons a (List (a,a))
在 Coq 中变成:
Inductive List3 : Type -> Type :=
| Empty3: forall A, List3 A
| Cons3 : forall A, A -> List3 (A * A) -> List3 A
.
用归纳原理:
> Check List3_rect.
forall P : forall T : Type, List3 T -> Type,
(forall A : Type, P A (Empty3 A)) ->
(forall (A : Type) (a : A) (l : List3 (A * A)), P (A * A) l -> P A (Cons3 A a l)) ->
forall (T : Type) (l : List3 T), P T l
删除“依赖”部分:
forall P : (Type -> Type),
(forall A : Type, P A) ->
(forall (A : Type), A -> List3 (A * A) -> P (A * A) -> P A ) ->
forall (T : Type), List3 T -> P T
在 Haskell 表示法中:
(forall a. p a) -> -- empty
(forall a, a -> List3 (a, a) -> p (a, a) -> p a ) -> -- cons
List3 t -> p t
除了额外的List3 (a, a)
论点,这是一种折叠。
最后,OP类型呢?
data List a = Empty | Cons a (List (List a))
唉,Coq 不接受这个类型
Inductive List4 : Type -> Type :=
| Empty4: forall A, List4 A
| Cons4 : forall A, A -> List4 (List4 A) -> List4 A
.
因为内部List4
事件并不处于严格的积极位置。这可能暗示我应该停止懒惰并使用 Coq 来完成这项工作,并开始自己思考所涉及的 F 代数...... ;-)