Dependent types¶
Reference material
This page is mostly based on the introduction of dependent types in the HoTT Book (Sections 1.2–1.6), immediately introducing corresponding formalizations in Rzk and noting some differences.
We now proceed to look at the primitives in Rzk for working with dependent types.
Functions¶
The type (x : A) → B x
is the type of (dependent)
functions with an argument of type A
and, for each input x
,
the output type B x
.
As a simple example of a dependent function, consider the identity function:
#define identity
: (A : U) → (x : A) → A
:= \ A x → x
Since we are not using x
in the type of identity
,
we can simply write the type of the argument,
without providing its name:
#define identity₁
: (A : U) → A → A
:= \ A x → x
We can write this definition differently,
by putting (A : U)
into parameters (before :
),
and omitting it in the lambda abstraction:
#define identity₂
( A : U)
: A → A
:= \ x → x
We could also move x
into parameters as well,
although this probably does not increase readability anymore:
#define identity₃
( A : U)
( x : A)
: A
:= x
Another, less trivial example of a dependent function is the one that swaps the arguments of another function:
#define swap
( A B C : U)
: (A → B → C) → (B → A → C)
:= \ f → \ b a → f a b
Product types¶
Rzk does not have built-in product types, since they are a special case of Σ-types, which we will discuss soon. For now, we give definition of product types:
#define prod
( A B : U)
: U
:= Σ (_ : A), B
The type prod A B
corresponds to the product type \(A \times B\).
The Unit
type corresponds to the type \(\mathbf{1}\).
The intended elements of prod A B
are only pairs (a, b) : prod A B
where a : A
and b : B
. Similarly, intended element of Unit
is only unit
. However, formally, this is not immediately true and instead
is a theorem that we can prove.
Remark on type formers¶
Formally, we have the following constituents of the definition for product types and function types (for comparison):
-
Type formation:
prod A B
is a type wheneverA
andB
are typesA → B
is a type wheneverA
andB
are types
-
Constructors (introduction rules):
(x , y)
is a term of typeprod A B
wheneverx : A
andy : B
\ x → y
is a term of typeA → B
whenever for anyx : A
we havey : B
-
Eliminators (elimination rules):
-
Given
z : prod A B
, we can project the first and second components:first z : A
andsecond z : B
-
it is also possible to pattern match (deconstruct) in a function argument or when introducing a parameter, e.g.
#define swap-prod₁ ( A B : U) : prod A B → prod B A := \ (x , y) → (y , x) #define swap-prod₂ ( A B : U) ( (x , y) : prod A B) : prod B A := ( y , x)
-
more generally, eliminators come in a form of an induction principle, which we will discuss below and can be defined in Rzk in terms of pattern matching or
first
andsecond
:#define ind-prod ( A B : U) ( C : prod A B → U) ( f : (a : A) → (b : B) → C (a , b)) : (z : prod A B) → C z := \ (a , b) → f a b
-
Given
f : A → B
, we can apply it to an argument of typea : A
:f a : B
Built-in eliminators in Rzk
Built-in eliminators in Rzk need to be always fully applied (e.g.
first
without an argument is invalid syntax!). Technically, this corresponds with the "second presentation" of type theory in Appendix A.2 of the HoTT Book. In practice, this is not always convenient for users, as we often want to curry some of these built-ins, so wrapper functions are introduced (by users), for example:#define pr₁ ( A B : U) : prod A B → A := \ p → first p
-
-
Computation rules:
- Projecting from a pair is computed as follows for any
x : A
andy : B
:first (x , y) ≡ x
second (x , y) ≡ y
- Applying an lambda abstraction is computed by substituting the argument into a body:
(\ x → y) a ≡ y{x ↦ a}
whena : A
and for allx : A
,y : B
.
- Projecting from a pair is computed as follows for any
-
Uniqueness principle (optional):
- For any
z : prod A B
, we havez ≡ (first z, second z)
- This holds definitionally for product types and Σ-types in Rzk, but is provable in a weaker (propositional) form in HoTT Book
- For any two functions
f : A → B
andg : A → B
, we havef ≡ g
iff for anyx : A
we havef x ≡ g x
- This is taken in a weaker form as an Axiom 2.9.3 in HoTT Book and can also be postulated or assumed locally in Rzk (e.g. see Function extensionality in the sHoTT project)
- For any
Recursion principle¶
Following the HoTT Book, for each type former we can formalize its recursion principle.
A recursion principle for type T
is a function that allows to produce
a result of arbitrary type C
from a value of type T
:
#define rec-T
( C : U)
-- ... (parameters to the recursion principle)
: T → C
For example, for the product type prod A B
, recursion principle looks like this:
#define rec-prod
( A B : U)
( C : U)
( f : A → B → C)
: prod A B → C
:= \ (a , b) → f a b
For the Unit
type, recursion principle is trivial:
#define rec-Unit
( C : U)
( c : C)
: Unit → C
:= \ unit → c
Induction principle¶
To define a dependent function out of a type, we use its induction principle,
which can be seen as a dependent version of the recursion principle.
An induction principle for type T
is a function that allows to produce
a result of arbitrary type C z
from a value z : T
:
#define ind-T
( C : T → U)
-- ... (parameters to the induction principle)
: (z : T) → C z
For example, for the product type prod A B
, induction principle looks like this:
#define ind-prod
( A B : U)
( C : prod A B → U)
( f : (a : A) → (b : B) → C (a , b))
: (z : prod A B) → C z
:= \ (a , b) → f a b
We can use ind-prod
to prove the uniqueness principle for products.
Here we use the identity type, which we will cover later, but for now it is
sufficient to know that there is always an element refl_{x} : x =_{A} x
for any x : A
.
#define uniq-prod
( A B : U)
( z : prod A B)
: (first z, second z) =_{prod A B} z
:= ind-prod A B
( \ z' → (first z', second z') =_{prod A B} z') -- C
( \ a b → refl_{(a , b)})
-- C (a, b)
-- ≡ ( \ z' → (first z', second z') =_{prod A B} z') (a, b)
-- ≡ (first (a, b), second (a, b)) =_{prod A B} (a, b)
-- ≡ (a, second (a, b)) =_{prod A B} (a, b)
-- ≡ (a, b) =_{prod A B} (a, b)
z
Since in Rzk the uniqueness principle is builtin, a simpler proof also works:
#define uniq-prod'
( A B : U)
( z : prod A B)
: (first z, second z) =_{prod A B} z
:= refl_{z} -- works in Rzk, not in HoTT Book, since in Rzk we have (first z, second z) ≡ z
For the Unit
type, induction principle is trivial:
#define ind-Unit
( C : Unit → U)
( c : C unit)
: (z : Unit) → C z
:= \ unit → c
Unlike rec-Unit
, induction principle for Unit
is not useless,
since it allows, for example, to prove the uniqueness principle:
#define uniq-Unit
( z : Unit)
: unit =_{Unit} z
:= ind-Unit
( \ z' → unit =_{Unit} z')
( refl_{unit})
z
Again, since Rzk has a builtin uniqueness principle for Unit
, a simpler proof also works:
#define uniq-Unit'
( z : Unit)
: unit =_{Unit} z
:= refl_{z} -- works in Rzk, not in HoTT Book, since in Rzk we have unit ≡ z
Dependent pair types (Σ-types)¶
A straightforward generalization of product types to dependent pairs Σ (a : A), B a
where A
is a type and B : A → U
is a type family indexed in A
.
The indended values of Σ (a : A), B a
are pairs (a , b)
of
terms a : A
and b : B a
. Note that the type of the second component
may depend on the value of the first component.
When the type family B
is constant, e.g. (\ _ → C)
,
then Σ (a : A), B a
becomes exactly the product type prod A C
.
To eliminate dependent pairs, we use first
, second
, or pattern
matching on pairs. However, the types of projections are less obvious compared
to the case of product types.
Projections¶
The first projection can be easily defined in terms of pattern matching:
#define pr₁
( A : U)
( B : A → U)
: (Σ (a : A), B a) → A
:= \ (a , _) → a
However, second projection requires some care. For instance, we might try this:
-- NOTE: incorrect definition
#define pr₂
( A : U)
( B : A → U)
: (Σ (a : A), B a) → B a -- ERROR!
:= \ (_ , b) → b
We get the undefined variable
error since a
is not visible outside of Σ-type definition.
To access it, we need a dependent function:
#define pr₂
( A : U)
( B : A → U)
: (z : Σ (a : A), B a) → B (pr₁ A B z)
:= \ (_ , b) → b
In Rzk, it is sometimes more convenient to talk about Σ-types as "total" types (as in "total spaces"):
#define total-type
( A : U)
( B : A → U)
: U
:= Σ (a : A), B a
We can use pattern matching in the function type and this new definition to write second projection slightly differently:
#define pr₂'
( A : U)
( B : A → U)
( (a, b) : total-type A B)
: B a
:= b
Recursion and induction principles¶
The recursion principle for Σ-types is a simple generalization of the recursion principle for product types:
#define rec-Σ
( A : U)
( B : A → U)
( C : U)
( f : (a : A) → B a → C)
: total-type A B → C
:= \ (a , b) → f a b
The induction principle is, again, a generalization of the recursion principle to dependent types:
#define ind-Σ
( A : U)
( B : A → U)
( C : total-type A B → U)
( f : (a : A) → (b : B a) → C (a , b))
: (z : total-type A B) → C z
:= \ (a , b) → f a b
As before, using ind-Σ
we may prove the uniqueness principle, now for Σ-types:
#define uniq-Σ
( A : U)
( B : A → U)
( z : total-type A B)
: (pr₁ A B z, pr₂ A B z) =_{total-type A B} z
:= ind-Σ A B
( \ z → (pr₁ A B z, pr₂ A B z) =_{total-type A B} z)
( \ a b → refl_{(a , b)})
z
And again, Rzk can accept a simpler proof, since uniqueness for Σ-types is already built into it:
#define uniq-Σ'
( A : U)
( B : A → U)
( z : total-type A B)
: (pr₁ A B z, pr₂ A B z) =_{total-type A B} z
:= refl_{z} -- works in Rzk, but not in HoTT Book
Type-theoretic "axiom" of choice¶
Using ind-Σ
we can also prove a type-theoretic axiom of choice:
#define AxiomOfChoice
: U
:= (A : U)
→ (B : U)
→ (R : A → B → U)
→ ((x : A) → Σ (y : B), R x y)
→ (Σ (f : A → B), (x : A) → R x (f x))
You are encouraged to try proving this yourself first.
If you encounter problems, try looking for the proof in the HoTT Book Section 1.6 (page 32).
If you still have issues formalizing it in Rzk, you may peek here:
Proof of the type theoretic axiom of choice
#define ac : AxiomOfChoice
:= \ A B R g → ( \ a → first (g a) , \ x → second (g x))
-- g : (x : A) → Σ (y : B), R x y
-- x : A
-- g x : Σ (y : B), R x y
-- second (g x) : R x (first (g x))
-- f : A → B
-- f := \ a → first (g a)
--
-- R x (f x)
-- == R x ((\ a → first (g a)) x)
-- == R x (first (g x))
Coproducts¶
Given types \(A\) and \(B\) a coproduct type \(A + B\) corresponds intuitively to a disjoint union of \(A\) and \(B\) (in set theory). We also have a nullary version: \(\mathbf{0}\) (empty type).
In Rzk, empty type and coproduct types do not exist, but a weaker version can be postulated.
Postulating the empty type¶
For example, an empty type can be postulated as follows:
Since there should be no values of type Void
,
the induction principle corresponds to the principle that from falsehood anything follows.
A non-dependent version of that corresponds to the recursion principle,
which we can define in terms of ind-Void
:
#define rec-Void
( C : U)
: Void → C
:= ind-Void (\ _ → C)
Postulating the coproduct type¶
Similarly, we can postulate the coproduct:
#postulate coprod
( A B : U)
: U
There are two ways to create a term of type coprod A B
—
inject a term from A
or a term of B
:
To eliminate a coproduct, we have to provide two handlers — one for the left case and one for the right:
#postulate ind-coprod
( A B : U)
( C : coprod A B → U)
( l : (a : A) → C (inl A B a))
( r : (b : B) → C (inr A B b))
: (z : coprod A B) → C z
Since we are postulating the induction principle, we also have to provide the computational rules explicitly. However, in Rzk, we can only postulate propositional computational rules:
#postulate ind-coprod-inl
( A B : U)
( C : coprod A B → U)
( l : (a : A) → C (inl A B a))
( r : (b : B) → C (inr A B b))
( a : A)
: ind-coprod A B C l r (inl A B a) = l a
#postulate ind-coprod-inr
( A B : U)
( C : coprod A B → U)
( l : (a : A) → C (inl A B a))
( r : (b : B) → C (inr A B b))
( b : B)
: ind-coprod A B C l r (inr A B b) = r b
We can now define recursion for coproducts as a special case of induction:
#define rec-coprod
( A B : U)
( C : U)
( l : A → C)
( r : B → C)
: coprod A B → C
:= ind-coprod A B (\ _ → C) l r
The uniqueness principle for coproducts says
that any coproduct is either an inl
or an inr
.
Proving the uniqueness is fairly straightforward, except
we have to provide some intermediate types explicitly:
#define uniq-coprod
( A B : U)
( z : coprod A B)
: coprod
( Σ (a : A), inl A B a = z)
( Σ (b : B), inr A B b = z)
:= ind-coprod A B
( \ z' → coprod
( Σ (a : A), inl A B a = z')
( Σ (b : B), inr A B b = z'))
( \ a' → inl
( Σ (a : A), (inl A B a = inl A B a'))
( Σ (b : B), (inr A B b = inl A B a'))
( a' , refl))
( \ b' → inr
( Σ (a : A), (inl A B a = inr A B b'))
( Σ (b : B), (inr A B b = inr A B b'))
( b' , refl))
z
Booleans¶
#postulate ind-Bool
( C : Bool → U)
( f : C false)
( t : C true)
: (z : Bool) → C z
#postulate ind-Bool-false
( C : Bool → U)
( f : C false)
( t : C true)
: ind-Bool C f t false = f
#postulate ind-Bool-true
( C : Bool → U)
( f : C false)
( t : C true)
: ind-Bool C f t true = t
#define rec-Bool
( C : U)
( f t : C)
: Bool → C
:= ind-Bool (\ _ → C) f t
#define uniq-Bool
( z : Bool)
: coprod (z = false) (z = true)
:= ind-Bool
( \ z' → coprod (z' = false) (z' = true))
( inl (false = false) (false = true) refl)
( inr (true = false) (true = true) refl)
z
#define not
: Bool → Bool
:= rec-Bool Bool true false
Unfortunately, because computation rules are postulated in a weak form, they do not compute automatically and have to be used explicitly, so the following proof does not work:
#define not-not-is-identity
: (z : Bool) → not (not z) = z
:= ind-Bool
( \ z → not (not z) = z)
( refl)
( refl)
There is a way to fix the proof, but we'll need to learn more about the identity types before we can do that.
Natural numbers¶
#postulate ℕ : U
#postulate zero : ℕ
#postulate succ (n : ℕ) : ℕ
#postulate ind-ℕ
( C : ℕ → U)
( base : C zero)
( step : (n : ℕ) → C n → C (succ n))
: (n : ℕ) → C n
#postulate ind-ℕ-zero
( C : ℕ → U)
( base : C zero)
( step : (n : ℕ) → C n → C (succ n))
: ind-ℕ C base step zero = base
#postulate ind-ℕ-succ
( C : ℕ → U)
( base : C zero)
( step : (n : ℕ) → C n → C (succ n))
( n : ℕ)
: ind-ℕ C base step (succ n) = step n (ind-ℕ C base step n)
#define rec-ℕ
( C : U)
( base : C)
( step : (n : ℕ) → C → C)
: ℕ → C
:= ind-ℕ (\ _ → C) base step
#define double-ℕ
: ℕ → ℕ
:= rec-ℕ ℕ zero (\ _ m → succ (succ m))
#define compute-ind-ℕ-zero
( C : ℕ → U)
( base : C zero)
( step : (n : ℕ) → C n → C (succ n))
: C zero
:= base
#define compute-ind-ℕ-one
( C : ℕ → U)
( base : C zero)
( step : (n : ℕ) → C n → C (succ n))
: C (succ zero)
:= step zero (compute-ind-ℕ-zero C base step)
#define compute-ind-ℕ-two
( C : ℕ → U)
( base : C zero)
( step : (n : ℕ) → C n → C (succ n))
: C (succ (succ zero))
:= step (succ zero) (compute-ind-ℕ-one C base step)
#compute compute-ind-ℕ-two (\ _ → ℕ) zero (\ _ m → succ (succ m))